LCOV - code coverage report
Current view: top level - src/backend/utils/adt - tsquery_op.c (source / functions) Hit Total Coverage
Test: PostgreSQL 18devel Lines: 144 160 90.0 %
Date: 2025-01-18 04:15:08 Functions: 19 20 95.0 %
Legend: Lines: hit not hit

          Line data    Source code
       1             : /*-------------------------------------------------------------------------
       2             :  *
       3             :  * tsquery_op.c
       4             :  *    Various operations with tsquery
       5             :  *
       6             :  * Portions Copyright (c) 1996-2025, PostgreSQL Global Development Group
       7             :  *
       8             :  *
       9             :  * IDENTIFICATION
      10             :  *    src/backend/utils/adt/tsquery_op.c
      11             :  *
      12             :  *-------------------------------------------------------------------------
      13             :  */
      14             : 
      15             : #include "postgres.h"
      16             : 
      17             : #include "lib/qunique.h"
      18             : #include "tsearch/ts_utils.h"
      19             : #include "utils/fmgrprotos.h"
      20             : #include "varatt.h"
      21             : 
      22             : Datum
      23          18 : tsquery_numnode(PG_FUNCTION_ARGS)
      24             : {
      25          18 :     TSQuery     query = PG_GETARG_TSQUERY(0);
      26          18 :     int         nnode = query->size;
      27             : 
      28          18 :     PG_FREE_IF_COPY(query, 0);
      29          18 :     PG_RETURN_INT32(nnode);
      30             : }
      31             : 
      32             : static QTNode *
      33         108 : join_tsqueries(TSQuery a, TSQuery b, int8 operator, uint16 distance)
      34             : {
      35         108 :     QTNode     *res = (QTNode *) palloc0(sizeof(QTNode));
      36             : 
      37         108 :     res->flags |= QTN_NEEDFREE;
      38             : 
      39         108 :     res->valnode = (QueryItem *) palloc0(sizeof(QueryItem));
      40         108 :     res->valnode->type = QI_OPR;
      41         108 :     res->valnode->qoperator.oper = operator;
      42         108 :     if (operator == OP_PHRASE)
      43          48 :         res->valnode->qoperator.distance = distance;
      44             : 
      45         108 :     res->child = (QTNode **) palloc0(sizeof(QTNode *) * 2);
      46         108 :     res->child[0] = QT2QTN(GETQUERY(b), GETOPERAND(b));
      47         108 :     res->child[1] = QT2QTN(GETQUERY(a), GETOPERAND(a));
      48         108 :     res->nchild = 2;
      49             : 
      50         108 :     return res;
      51             : }
      52             : 
      53             : Datum
      54          36 : tsquery_and(PG_FUNCTION_ARGS)
      55             : {
      56          36 :     TSQuery     a = PG_GETARG_TSQUERY_COPY(0);
      57          36 :     TSQuery     b = PG_GETARG_TSQUERY_COPY(1);
      58             :     QTNode     *res;
      59             :     TSQuery     query;
      60             : 
      61          36 :     if (a->size == 0)
      62             :     {
      63           0 :         PG_FREE_IF_COPY(a, 1);
      64           0 :         PG_RETURN_POINTER(b);
      65             :     }
      66          36 :     else if (b->size == 0)
      67             :     {
      68           0 :         PG_FREE_IF_COPY(b, 1);
      69           0 :         PG_RETURN_POINTER(a);
      70             :     }
      71             : 
      72          36 :     res = join_tsqueries(a, b, OP_AND, 0);
      73             : 
      74          36 :     query = QTN2QT(res);
      75             : 
      76          36 :     QTNFree(res);
      77          36 :     PG_FREE_IF_COPY(a, 0);
      78          36 :     PG_FREE_IF_COPY(b, 1);
      79             : 
      80          36 :     PG_RETURN_TSQUERY(query);
      81             : }
      82             : 
      83             : Datum
      84          24 : tsquery_or(PG_FUNCTION_ARGS)
      85             : {
      86          24 :     TSQuery     a = PG_GETARG_TSQUERY_COPY(0);
      87          24 :     TSQuery     b = PG_GETARG_TSQUERY_COPY(1);
      88             :     QTNode     *res;
      89             :     TSQuery     query;
      90             : 
      91          24 :     if (a->size == 0)
      92             :     {
      93           0 :         PG_FREE_IF_COPY(a, 1);
      94           0 :         PG_RETURN_POINTER(b);
      95             :     }
      96          24 :     else if (b->size == 0)
      97             :     {
      98           0 :         PG_FREE_IF_COPY(b, 1);
      99           0 :         PG_RETURN_POINTER(a);
     100             :     }
     101             : 
     102          24 :     res = join_tsqueries(a, b, OP_OR, 0);
     103             : 
     104          24 :     query = QTN2QT(res);
     105             : 
     106          24 :     QTNFree(res);
     107          24 :     PG_FREE_IF_COPY(a, 0);
     108          24 :     PG_FREE_IF_COPY(b, 1);
     109             : 
     110          24 :     PG_RETURN_TSQUERY(query);
     111             : }
     112             : 
     113             : Datum
     114          48 : tsquery_phrase_distance(PG_FUNCTION_ARGS)
     115             : {
     116          48 :     TSQuery     a = PG_GETARG_TSQUERY_COPY(0);
     117          48 :     TSQuery     b = PG_GETARG_TSQUERY_COPY(1);
     118             :     QTNode     *res;
     119             :     TSQuery     query;
     120          48 :     int32       distance = PG_GETARG_INT32(2);
     121             : 
     122          48 :     if (distance < 0 || distance > MAXENTRYPOS)
     123           0 :         ereport(ERROR,
     124             :                 (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
     125             :                  errmsg("distance in phrase operator must be an integer value between zero and %d inclusive",
     126             :                         MAXENTRYPOS)));
     127          48 :     if (a->size == 0)
     128             :     {
     129           0 :         PG_FREE_IF_COPY(a, 1);
     130           0 :         PG_RETURN_POINTER(b);
     131             :     }
     132          48 :     else if (b->size == 0)
     133             :     {
     134           0 :         PG_FREE_IF_COPY(b, 1);
     135           0 :         PG_RETURN_POINTER(a);
     136             :     }
     137             : 
     138          48 :     res = join_tsqueries(a, b, OP_PHRASE, (uint16) distance);
     139             : 
     140          48 :     query = QTN2QT(res);
     141             : 
     142          48 :     QTNFree(res);
     143          48 :     PG_FREE_IF_COPY(a, 0);
     144          48 :     PG_FREE_IF_COPY(b, 1);
     145             : 
     146          48 :     PG_RETURN_TSQUERY(query);
     147             : }
     148             : 
     149             : Datum
     150          42 : tsquery_phrase(PG_FUNCTION_ARGS)
     151             : {
     152          42 :     PG_RETURN_DATUM(DirectFunctionCall3(tsquery_phrase_distance,
     153             :                                         PG_GETARG_DATUM(0),
     154             :                                         PG_GETARG_DATUM(1),
     155             :                                         Int32GetDatum(1)));
     156             : }
     157             : 
     158             : Datum
     159          12 : tsquery_not(PG_FUNCTION_ARGS)
     160             : {
     161          12 :     TSQuery     a = PG_GETARG_TSQUERY_COPY(0);
     162             :     QTNode     *res;
     163             :     TSQuery     query;
     164             : 
     165          12 :     if (a->size == 0)
     166           0 :         PG_RETURN_POINTER(a);
     167             : 
     168          12 :     res = (QTNode *) palloc0(sizeof(QTNode));
     169             : 
     170          12 :     res->flags |= QTN_NEEDFREE;
     171             : 
     172          12 :     res->valnode = (QueryItem *) palloc0(sizeof(QueryItem));
     173          12 :     res->valnode->type = QI_OPR;
     174          12 :     res->valnode->qoperator.oper = OP_NOT;
     175             : 
     176          12 :     res->child = (QTNode **) palloc0(sizeof(QTNode *));
     177          12 :     res->child[0] = QT2QTN(GETQUERY(a), GETOPERAND(a));
     178          12 :     res->nchild = 1;
     179             : 
     180          12 :     query = QTN2QT(res);
     181             : 
     182          12 :     QTNFree(res);
     183          12 :     PG_FREE_IF_COPY(a, 0);
     184             : 
     185          12 :     PG_RETURN_POINTER(query);
     186             : }
     187             : 
     188             : static int
     189         432 : CompareTSQ(TSQuery a, TSQuery b)
     190             : {
     191         432 :     if (a->size != b->size)
     192             :     {
     193         208 :         return (a->size < b->size) ? -1 : 1;
     194             :     }
     195         224 :     else if (VARSIZE(a) != VARSIZE(b))
     196             :     {
     197         152 :         return (VARSIZE(a) < VARSIZE(b)) ? -1 : 1;
     198             :     }
     199          72 :     else if (a->size != 0)
     200             :     {
     201          72 :         QTNode     *an = QT2QTN(GETQUERY(a), GETOPERAND(a));
     202          72 :         QTNode     *bn = QT2QTN(GETQUERY(b), GETOPERAND(b));
     203          72 :         int         res = QTNodeCompare(an, bn);
     204             : 
     205          72 :         QTNFree(an);
     206          72 :         QTNFree(bn);
     207             : 
     208          72 :         return res;
     209             :     }
     210             : 
     211           0 :     return 0;
     212             : }
     213             : 
     214             : Datum
     215          78 : tsquery_cmp(PG_FUNCTION_ARGS)
     216             : {
     217          78 :     TSQuery     a = PG_GETARG_TSQUERY_COPY(0);
     218          78 :     TSQuery     b = PG_GETARG_TSQUERY_COPY(1);
     219          78 :     int         res = CompareTSQ(a, b);
     220             : 
     221          78 :     PG_FREE_IF_COPY(a, 0);
     222          78 :     PG_FREE_IF_COPY(b, 1);
     223             : 
     224          78 :     PG_RETURN_INT32(res);
     225             : }
     226             : 
     227             : #define CMPFUNC( NAME, CONDITION )              \
     228             : Datum                                           \
     229             : NAME(PG_FUNCTION_ARGS) {                        \
     230             :     TSQuery  a = PG_GETARG_TSQUERY_COPY(0);     \
     231             :     TSQuery  b = PG_GETARG_TSQUERY_COPY(1);     \
     232             :     int res = CompareTSQ(a,b);                  \
     233             :                                                 \
     234             :     PG_FREE_IF_COPY(a,0);                       \
     235             :     PG_FREE_IF_COPY(b,1);                       \
     236             :                                                 \
     237             :     PG_RETURN_BOOL( CONDITION );                \
     238             : }   \
     239             : /* keep compiler quiet - no extra ; */          \
     240             : extern int no_such_variable
     241             : 
     242          90 : CMPFUNC(tsquery_lt, res < 0);
     243          68 : CMPFUNC(tsquery_le, res <= 0);
     244          66 : CMPFUNC(tsquery_eq, res == 0);
     245          62 : CMPFUNC(tsquery_ge, res >= 0);
     246          68 : CMPFUNC(tsquery_gt, res > 0);
     247           0 : CMPFUNC(tsquery_ne, res != 0);
     248             : 
     249             : TSQuerySign
     250         132 : makeTSQuerySign(TSQuery a)
     251             : {
     252             :     int         i;
     253         132 :     QueryItem  *ptr = GETQUERY(a);
     254         132 :     TSQuerySign sign = 0;
     255             : 
     256         348 :     for (i = 0; i < a->size; i++)
     257             :     {
     258         216 :         if (ptr->type == QI_VAL)
     259         174 :             sign |= ((TSQuerySign) 1) << (((unsigned int) ptr->qoperand.valcrc) % TSQS_SIGLEN);
     260         216 :         ptr++;
     261             :     }
     262             : 
     263         132 :     return sign;
     264             : }
     265             : 
     266             : static char **
     267         408 : collectTSQueryValues(TSQuery a, int *nvalues_p)
     268             : {
     269         408 :     QueryItem  *ptr = GETQUERY(a);
     270         408 :     char       *operand = GETOPERAND(a);
     271             :     char      **values;
     272         408 :     int         nvalues = 0;
     273             :     int         i;
     274             : 
     275         408 :     values = (char **) palloc(sizeof(char *) * a->size);
     276             : 
     277        1272 :     for (i = 0; i < a->size; i++)
     278             :     {
     279         864 :         if (ptr->type == QI_VAL)
     280             :         {
     281         636 :             int         len = ptr->qoperand.length;
     282             :             char       *val;
     283             : 
     284         636 :             val = palloc(len + 1);
     285         636 :             memcpy(val, operand + ptr->qoperand.distance, len);
     286         636 :             val[len] = '\0';
     287             : 
     288         636 :             values[nvalues++] = val;
     289             :         }
     290         864 :         ptr++;
     291             :     }
     292             : 
     293         408 :     *nvalues_p = nvalues;
     294         408 :     return values;
     295             : }
     296             : 
     297             : static int
     298         520 : cmp_string(const void *a, const void *b)
     299             : {
     300         520 :     const char *sa = *((char *const *) a);
     301         520 :     const char *sb = *((char *const *) b);
     302             : 
     303         520 :     return strcmp(sa, sb);
     304             : }
     305             : 
     306             : Datum
     307         204 : tsq_mcontains(PG_FUNCTION_ARGS)
     308             : {
     309         204 :     TSQuery     query = PG_GETARG_TSQUERY(0);
     310         204 :     TSQuery     ex = PG_GETARG_TSQUERY(1);
     311             :     char      **query_values;
     312             :     int         query_nvalues;
     313             :     char      **ex_values;
     314             :     int         ex_nvalues;
     315         204 :     bool        result = true;
     316             : 
     317             :     /* Extract the query terms into arrays */
     318         204 :     query_values = collectTSQueryValues(query, &query_nvalues);
     319         204 :     ex_values = collectTSQueryValues(ex, &ex_nvalues);
     320             : 
     321             :     /* Sort and remove duplicates from both arrays */
     322         204 :     qsort(query_values, query_nvalues, sizeof(char *), cmp_string);
     323         204 :     query_nvalues = qunique(query_values, query_nvalues, sizeof(char *),
     324             :                             cmp_string);
     325         204 :     qsort(ex_values, ex_nvalues, sizeof(char *), cmp_string);
     326         204 :     ex_nvalues = qunique(ex_values, ex_nvalues, sizeof(char *), cmp_string);
     327             : 
     328         204 :     if (ex_nvalues > query_nvalues)
     329          80 :         result = false;
     330             :     else
     331             :     {
     332             :         int         i;
     333         124 :         int         j = 0;
     334             : 
     335         160 :         for (i = 0; i < ex_nvalues; i++)
     336             :         {
     337         316 :             for (; j < query_nvalues; j++)
     338             :             {
     339         228 :                 if (strcmp(ex_values[i], query_values[j]) == 0)
     340          36 :                     break;
     341             :             }
     342         124 :             if (j == query_nvalues)
     343             :             {
     344          88 :                 result = false;
     345          88 :                 break;
     346             :             }
     347             :         }
     348             :     }
     349             : 
     350         204 :     PG_RETURN_BOOL(result);
     351             : }
     352             : 
     353             : Datum
     354         100 : tsq_mcontained(PG_FUNCTION_ARGS)
     355             : {
     356         100 :     PG_RETURN_DATUM(DirectFunctionCall2(tsq_mcontains,
     357             :                                         PG_GETARG_DATUM(1),
     358             :                                         PG_GETARG_DATUM(0)));
     359             : }

Generated by: LCOV version 1.14