Line data Source code
1 : /*------------------------------------------------------------------------- 2 : * 3 : * test_parser.c 4 : * Simple example of a text search parser 5 : * 6 : * Copyright (c) 2007-2025, PostgreSQL Global Development Group 7 : * 8 : * IDENTIFICATION 9 : * src/test/modules/test_parser/test_parser.c 10 : * 11 : *------------------------------------------------------------------------- 12 : */ 13 : #include "postgres.h" 14 : 15 : #include "fmgr.h" 16 : 17 2 : PG_MODULE_MAGIC; 18 : 19 : /* 20 : * types 21 : */ 22 : 23 : /* self-defined type */ 24 : typedef struct 25 : { 26 : char *buffer; /* text to parse */ 27 : int len; /* length of the text in buffer */ 28 : int pos; /* position of the parser */ 29 : } ParserState; 30 : 31 : typedef struct 32 : { 33 : int lexid; 34 : char *alias; 35 : char *descr; 36 : } LexDescr; 37 : 38 : /* 39 : * functions 40 : */ 41 4 : PG_FUNCTION_INFO_V1(testprs_start); 42 4 : PG_FUNCTION_INFO_V1(testprs_getlexeme); 43 4 : PG_FUNCTION_INFO_V1(testprs_end); 44 4 : PG_FUNCTION_INFO_V1(testprs_lextype); 45 : 46 : Datum 47 10 : testprs_start(PG_FUNCTION_ARGS) 48 : { 49 10 : ParserState *pst = (ParserState *) palloc0(sizeof(ParserState)); 50 : 51 10 : pst->buffer = (char *) PG_GETARG_POINTER(0); 52 10 : pst->len = PG_GETARG_INT32(1); 53 10 : pst->pos = 0; 54 : 55 10 : PG_RETURN_POINTER(pst); 56 : } 57 : 58 : Datum 59 92 : testprs_getlexeme(PG_FUNCTION_ARGS) 60 : { 61 92 : ParserState *pst = (ParserState *) PG_GETARG_POINTER(0); 62 92 : char **t = (char **) PG_GETARG_POINTER(1); 63 92 : int *tlen = (int *) PG_GETARG_POINTER(2); 64 92 : int startpos = pst->pos; 65 : int type; 66 : 67 92 : *t = pst->buffer + pst->pos; 68 : 69 92 : if (pst->pos < pst->len && 70 82 : (pst->buffer)[pst->pos] == ' ') 71 : { 72 : /* blank type */ 73 36 : type = 12; 74 : /* go to the next non-space character */ 75 72 : while (pst->pos < pst->len && 76 72 : (pst->buffer)[pst->pos] == ' ') 77 36 : (pst->pos)++; 78 : } 79 : else 80 : { 81 : /* word type */ 82 56 : type = 3; 83 : /* go to the next space character */ 84 330 : while (pst->pos < pst->len && 85 310 : (pst->buffer)[pst->pos] != ' ') 86 274 : (pst->pos)++; 87 : } 88 : 89 92 : *tlen = pst->pos - startpos; 90 : 91 : /* we are finished if (*tlen == 0) */ 92 92 : if (*tlen == 0) 93 10 : type = 0; 94 : 95 92 : PG_RETURN_INT32(type); 96 : } 97 : 98 : Datum 99 10 : testprs_end(PG_FUNCTION_ARGS) 100 : { 101 10 : ParserState *pst = (ParserState *) PG_GETARG_POINTER(0); 102 : 103 10 : pfree(pst); 104 10 : PG_RETURN_VOID(); 105 : } 106 : 107 : Datum 108 2 : testprs_lextype(PG_FUNCTION_ARGS) 109 : { 110 : /* 111 : * Remarks: - we have to return the blanks for headline reason - we use 112 : * the same lexids like Teodor in the default word parser; in this way we 113 : * can reuse the headline function of the default word parser. 114 : */ 115 2 : LexDescr *descr = (LexDescr *) palloc(sizeof(LexDescr) * (2 + 1)); 116 : 117 : /* there are only two types in this parser */ 118 2 : descr[0].lexid = 3; 119 2 : descr[0].alias = pstrdup("word"); 120 2 : descr[0].descr = pstrdup("Word"); 121 2 : descr[1].lexid = 12; 122 2 : descr[1].alias = pstrdup("blank"); 123 2 : descr[1].descr = pstrdup("Space symbols"); 124 2 : descr[2].lexid = 0; 125 : 126 2 : PG_RETURN_POINTER(descr); 127 : }