/* -------------------------------------------------------------------------
 *
 * tsvector_op.c
 *	  operations over tsvector
 *
 * Portions Copyright (c) 1996-2012, PostgreSQL Global Development Group
 *
 *
 * IDENTIFICATION
 *	  src/backend/utils/adt/tsvector_op.c
 *
 * -------------------------------------------------------------------------
 */

#include "postgres.h"
#include "knl/knl_variable.h"

#include "catalog/namespace.h"
#include "catalog/pg_type.h"
#include "commands/trigger.h"
#include "executor/spi.h"
#include "funcapi.h"
#include "mb/pg_wchar.h"
#include "miscadmin.h"
#include "tsearch/ts_utils.h"
#include "utils/builtins.h"
#include "utils/lsyscache.h"
#include "utils/rel.h"
#include "utils/rel_gs.h"

typedef struct {
    WordEntry* arrb;
    WordEntry* arre;
    char* values;
    char* operand;
} CHKVAL;

typedef struct StatEntry {
    uint32 ndoc; /* zero indicates that we already was here
                  * while walking throug the tree */
    uint32 nentry;
    struct StatEntry* left;
    struct StatEntry* right;
    uint32 lenlexeme;
    char lexeme[1];
} StatEntry;

#define STATENTRYHDRSZ (offsetof(StatEntry, lexeme))

typedef struct {
    int4 weight;

    uint32 maxdepth;

    StatEntry** stack;
    uint32 stackpos;

    StatEntry* root;
} TSVectorStat;

#define STATHDRSIZE (offsetof(TSVectorStat, data))

static Datum tsvector_update_trigger(PG_FUNCTION_ARGS, bool config_column);

/*
 * Check if datatype is the specified type or equivalent to it.
 *
 * Note: we could just do getBaseType() unconditionally, but since that's
 * a relatively expensive catalog lookup that most users won't need, we
 * try the straight comparison first.
 */
static bool is_expected_type(Oid typid, Oid expected_type)
{
    if (typid == expected_type)
        return true;
    typid = getBaseType(typid);
    if (typid == expected_type)
        return true;
    return false;
}

/* Check if datatype is TEXT or binary-equivalent to it */
static bool is_text_type(Oid typid)
{
    /* varchar(n) and char(n) are binary-compatible with text */
    if (typid == TEXTOID || typid == VARCHAROID || typid == BPCHAROID)
        return true;
    /* Allow domains over these types, too */
    typid = getBaseType(typid);
    if (typid == TEXTOID || typid == VARCHAROID || typid == BPCHAROID)
        return true;
    return false;
}

/*
 * Order: haspos, len, word, for all positions (pos, weight)
 */
static int silly_cmp_tsvector(const TSVector a, const TSVector b)
{
    if (VARSIZE(a) < VARSIZE(b))
        return -1;
    else if (VARSIZE(a) > VARSIZE(b))
        return 1;
    else if (a->size < b->size)
        return -1;
    else if (a->size > b->size)
        return 1;
    else {
        WordEntry* aptr = ARRPTR(a);
        WordEntry* bptr = ARRPTR(b);
        int i = 0;
        int res;

        for (i = 0; i < a->size; i++) {
            if (aptr->haspos != bptr->haspos) {
                return (aptr->haspos > bptr->haspos) ? -1 : 1;
            } else if ((res = tsCompareString(
                            STRPTR(a) + aptr->pos, aptr->len, STRPTR(b) + bptr->pos, bptr->len, false)) != 0) {
                return res;
            } else if (aptr->haspos) {
                WordEntryPos* ap = POSDATAPTR(a, aptr);
                WordEntryPos* bp = POSDATAPTR(b, bptr);
                int j;

                if (POSDATALEN(a, aptr) != POSDATALEN(b, bptr))
                    return (POSDATALEN(a, aptr) > POSDATALEN(b, bptr)) ? -1 : 1;

                for (j = 0; j < POSDATALEN(a, aptr); j++) {
                    if (WEP_GETPOS(*ap) != WEP_GETPOS(*bp)) {
                        return (WEP_GETPOS(*ap) > WEP_GETPOS(*bp)) ? -1 : 1;
                    } else if (WEP_GETWEIGHT(*ap) != WEP_GETWEIGHT(*bp)) {
                        return (WEP_GETWEIGHT(*ap) > WEP_GETWEIGHT(*bp)) ? -1 : 1;
                    }
                    ap++, bp++;
                }
            }

            aptr++;
            bptr++;
        }
    }

    return 0;
}

#define TSVECTORCMPFUNC(type, action, ret)  \
    Datum tsvector_##type(PG_FUNCTION_ARGS) \
    {                                       \
        TSVector a = PG_GETARG_TSVECTOR(0); \
        TSVector b = PG_GETARG_TSVECTOR(1); \
        int res = silly_cmp_tsvector(a, b); \
        PG_FREE_IF_COPY(a, 0);              \
        PG_FREE_IF_COPY(b, 1);              \
        PG_RETURN_##ret(res action 0);      \
    }                                       \
    /* keep compiler quiet - no extra ; */  \
    extern int no_such_variable

TSVECTORCMPFUNC(lt, <, BOOL);
TSVECTORCMPFUNC(le, <=, BOOL);
TSVECTORCMPFUNC(eq, ==, BOOL);
TSVECTORCMPFUNC(ge, >=, BOOL);
TSVECTORCMPFUNC(gt, >, BOOL);
TSVECTORCMPFUNC(ne, !=, BOOL);
TSVECTORCMPFUNC(cmp, +, INT32);

Datum tsvector_strip(PG_FUNCTION_ARGS)
{
    ts_check_feature_disable();
    TSVector in = PG_GETARG_TSVECTOR(0);
    TSVector out;
    int i, len = 0;
    WordEntry* arrin = ARRPTR(in);
    WordEntry* arrout = NULL;
    char* cur = NULL;
    errno_t rc;

    for (i = 0; i < in->size; i++)
        len += arrin[i].len;

    len = CALCDATASIZE(in->size, len);
    out = (TSVector)palloc0(len);
    SET_VARSIZE(out, len);
    out->size = in->size;
    arrout = ARRPTR(out);
    cur = STRPTR(out);
    for (i = 0; i < in->size; i++) {
        rc = memcpy_s(cur, len, STRPTR(in) + arrin[i].pos, arrin[i].len);
        securec_check(rc, "\0", "\0");

        arrout[i].haspos = 0;
        arrout[i].len = arrin[i].len;
        arrout[i].pos = cur - STRPTR(out);
        cur += arrout[i].len;
    }

    PG_FREE_IF_COPY(in, 0);
    PG_RETURN_POINTER(out);
}

Datum tsvector_length(PG_FUNCTION_ARGS)
{
    ts_check_feature_disable();
    TSVector in = PG_GETARG_TSVECTOR(0);
    int4 ret = in->size;

    PG_FREE_IF_COPY(in, 0);
    PG_RETURN_INT32(ret);
}

Datum tsvector_setweight(PG_FUNCTION_ARGS)
{
    ts_check_feature_disable();
    TSVector in = PG_GETARG_TSVECTOR(0);
    char cw = PG_GETARG_CHAR(1);
    TSVector out;
    int i, j;
    WordEntry* entry = NULL;
    WordEntryPos* p = NULL;
    unsigned int w = 0;

    switch (cw) {
        case 'A':
        case 'a':
            w = 3;
            break;
        case 'B':
        case 'b':
            w = 2;
            break;
        case 'C':
        case 'c':
            w = 1;
            break;
        case 'D':
        case 'd':
            w = 0;
            break;
        default:
            /* internal error */
            ereport(ERROR, (errcode(ERRCODE_UNRECOGNIZED_NODE_TYPE), errmsg("unrecognized weight: %d", cw)));
    }

    out = (TSVector)palloc(VARSIZE(in));
    errno_t rc = memcpy_s(out, VARSIZE(in), in, VARSIZE(in));
    securec_check(rc, "\0", "\0");
    entry = ARRPTR(out);
    i = out->size;
    while (i--) {
        if ((j = POSDATALEN(out, entry)) != 0) {
            p = POSDATAPTR(out, entry);
            while (j--) {
                WEP_SETWEIGHT(*p, w);
                p++;
            }
        }
        entry++;
    }

    PG_FREE_IF_COPY(in, 0);
    PG_RETURN_POINTER(out);
}

#define compareEntry(pa, a, pb, b) tsCompareString((pa) + (a)->pos, (a)->len, (pb) + (b)->pos, (b)->len, false)

/*
 * Add positions from src to dest after offsetting them by maxpos.
 * Return the number added (might be less than expected due to overflow)
 */
static int4 add_pos(TSVector src, WordEntry* srcptr, TSVector dest, WordEntry* destptr, int4 maxpos)
{
    uint16* clen = &_POSVECPTR(dest, destptr)->npos;
    int i;
    uint16 slen = POSDATALEN(src, srcptr), startlen;
    WordEntryPos *spos = POSDATAPTR(src, srcptr), *dpos = POSDATAPTR(dest, destptr);

    if (!destptr->haspos)
        *clen = 0;

    startlen = *clen;
    for (i = 0; i < slen && *clen < MAXNUMPOS && (*clen == 0 || WEP_GETPOS(dpos[*clen - 1]) != MAXENTRYPOS - 1); i++) {
        WEP_SETWEIGHT(dpos[*clen], WEP_GETWEIGHT(spos[i]));
        WEP_SETPOS(dpos[*clen], LIMITPOS(WEP_GETPOS(spos[i]) + maxpos));
        (*clen)++;
    }

    if (*clen != startlen)
        destptr->haspos = 1;
    return *clen - startlen;
}

Datum tsvector_concat(PG_FUNCTION_ARGS)
{
    TSVector in1 = PG_GETARG_TSVECTOR(0);
    TSVector in2 = PG_GETARG_TSVECTOR(1);
    TSVector out;
    WordEntry* ptr = NULL;
    WordEntry* ptr1 = NULL;
    WordEntry* ptr2 = NULL;
    WordEntryPos* p = NULL;
    int maxpos = 0, i, j, i1, i2, dataoff, output_bytes, output_size;
    char* data = NULL;
    char* data1 = NULL;
    char* data2 = NULL;
    char* data_init = NULL;
    errno_t ret = EOK;
    int data_total_length = 0;

    /* Get max position in in1; we'll need this to offset in2's positions */
    ptr = ARRPTR(in1);
    i = in1->size;
    while (i--) {
        if ((j = POSDATALEN(in1, ptr)) != 0) {
            p = POSDATAPTR(in1, ptr);
            while (j--) {
                if (WEP_GETPOS(*p) > maxpos)
                    maxpos = WEP_GETPOS(*p);
                p++;
            }
        }
        ptr++;
    }

    ptr1 = ARRPTR(in1);
    ptr2 = ARRPTR(in2);
    data1 = STRPTR(in1);
    data2 = STRPTR(in2);
    i1 = in1->size;
    i2 = in2->size;

    /*
     * Conservative estimate of space needed.  We might need all the data in
     * both inputs, and conceivably add a pad byte before position data for
     * each item where there was none before.
     */
    output_bytes = VARSIZE(in1) + VARSIZE(in2) + i1 + i2;

    out = (TSVector)palloc0(output_bytes);
    SET_VARSIZE(out, output_bytes);

    /*
     * We must make out->size valid so that STRPTR(out) is sensible.  We'll
     * collapse out any unused space at the end.
     */
    out->size = in1->size + in2->size;

    ptr = ARRPTR(out);
    data = STRPTR(out);
    data_init = data;
    data_total_length = output_bytes - CALCDATASIZE(out->size, 0);
    dataoff = 0;
    while (i1 && i2) {
        int cmp = compareEntry(data1, ptr1, data2, ptr2);

        Assert(data == data_init);
        if (cmp < 0) { /* in1 first */
            ptr->haspos = ptr1->haspos;
            ptr->len = ptr1->len;
            if (ptr1->len > 0) {
                ret = memcpy_s(data + dataoff, data_total_length - dataoff, data1 + ptr1->pos, ptr1->len);
                securec_check(ret, "\0", "\0");
            }
            ptr->pos = dataoff;
            dataoff += ptr1->len;
            if (ptr->haspos) {
                dataoff = SHORTALIGN(dataoff);
                ret = memcpy_s(data + dataoff,
                    data_total_length - dataoff,
                    _POSVECPTR(in1, ptr1),
                    POSDATALEN(in1, ptr1) * sizeof(WordEntryPos) + sizeof(uint16));
                securec_check(ret, "\0", "\0");
                dataoff += POSDATALEN(in1, ptr1) * sizeof(WordEntryPos) + sizeof(uint16);
            }

            ptr++;
            ptr1++;
            i1--;
        } else if (cmp > 0) { /* in2 first */
            ptr->haspos = ptr2->haspos;
            ptr->len = ptr2->len;
            if (ptr2->len > 0) {
                ret = memcpy_s(data + dataoff, data_total_length - dataoff, data2 + ptr2->pos, ptr2->len);
                securec_check(ret, "\0", "\0");
            }
            ptr->pos = dataoff;
            dataoff += ptr2->len;
            if (ptr->haspos) {
                int addlen = add_pos(in2, ptr2, out, ptr, maxpos);

                if (addlen == 0)
                    ptr->haspos = 0;
                else {
                    dataoff = SHORTALIGN(dataoff);
                    dataoff += addlen * sizeof(WordEntryPos) + sizeof(uint16);
                }
            }

            ptr++;
            ptr2++;
            i2--;
        } else {
            ptr->haspos = ptr1->haspos | ptr2->haspos;
            ptr->len = ptr1->len;
            if (ptr1->len > 0) {
                ret = memcpy_s(data + dataoff, data_total_length - dataoff, data1 + ptr1->pos, ptr1->len);
                securec_check(ret, "\0", "\0");
            }
            ptr->pos = dataoff;
            dataoff += ptr1->len;
            if (ptr->haspos) {
                if (ptr1->haspos) {
                    dataoff = SHORTALIGN(dataoff);
                    ret = memcpy_s(data + dataoff,
                        data_total_length - dataoff,
                        _POSVECPTR(in1, ptr1),
                        POSDATALEN(in1, ptr1) * sizeof(WordEntryPos) + sizeof(uint16));
                    securec_check(ret, "\0", "\0");
                    dataoff += POSDATALEN(in1, ptr1) * sizeof(WordEntryPos) + sizeof(uint16);
                    if (ptr2->haspos)
                        dataoff += add_pos(in2, ptr2, out, ptr, maxpos) * sizeof(WordEntryPos);
                } else /* must have ptr2->haspos */
                {
                    int addlen = add_pos(in2, ptr2, out, ptr, maxpos);

                    if (addlen == 0)
                        ptr->haspos = 0;
                    else {
                        dataoff = SHORTALIGN(dataoff);
                        dataoff += addlen * sizeof(WordEntryPos) + sizeof(uint16);
                    }
                }
            }

            ptr++;
            ptr1++;
            ptr2++;
            i1--;
            i2--;
        }
    }

    while (i1) {
        ptr->haspos = ptr1->haspos;
        ptr->len = ptr1->len;
        if (ptr1->len > 0) {
            ret = memcpy_s(data + dataoff, data_total_length - dataoff, data1 + ptr1->pos, ptr1->len);
            securec_check(ret, "\0", "\0");
        }
        ptr->pos = dataoff;
        dataoff += ptr1->len;
        if (ptr->haspos) {
            dataoff = SHORTALIGN(dataoff);
            ret = memcpy_s(data + dataoff,
                data_total_length - dataoff,
                _POSVECPTR(in1, ptr1),
                POSDATALEN(in1, ptr1) * sizeof(WordEntryPos) + sizeof(uint16));
            securec_check(ret, "\0", "\0");
            dataoff += POSDATALEN(in1, ptr1) * sizeof(WordEntryPos) + sizeof(uint16);
        }

        ptr++;
        ptr1++;
        i1--;
    }

    while (i2) {
        ptr->haspos = ptr2->haspos;
        ptr->len = ptr2->len;
        if (ptr2->len > 0) {
            ret = memcpy_s(data + dataoff, data_total_length - dataoff, data2 + ptr2->pos, ptr2->len);
            securec_check(ret, "\0", "\0");
        }
        ptr->pos = dataoff;
        dataoff += ptr2->len;
        if (ptr->haspos) {
            int addlen = add_pos(in2, ptr2, out, ptr, maxpos);

            if (addlen == 0)
                ptr->haspos = 0;
            else {
                dataoff = SHORTALIGN(dataoff);
                dataoff += addlen * sizeof(WordEntryPos) + sizeof(uint16);
            }
        }

        ptr++;
        ptr2++;
        i2--;
    }

    /*
     * Instead of checking each offset individually, we check for overflow of
     * pos fields once at the end.
     */
    if (dataoff > MAXSTRPOS)
        ereport(ERROR,
            (errcode(ERRCODE_PROGRAM_LIMIT_EXCEEDED),
                errmsg("string is too long for tsvector (%d bytes, max %d bytes)", dataoff, MAXSTRPOS)));

    /*
     * Adjust sizes (asserting that we didn't overrun the original estimates)
     * and collapse out any unused array entries.
     */
    output_size = ptr - ARRPTR(out);
    Assert(output_size <= out->size);
    out->size = output_size;
    if (data != STRPTR(out) && dataoff > 0) {
        ret = memmove_s(STRPTR(out), output_bytes - CALCDATASIZE(out->size, 0), data, dataoff);
        securec_check(ret, "\0", "\0");
    }

    output_bytes = CALCDATASIZE(out->size, dataoff);
    Assert((unsigned int)(output_bytes) <= VARSIZE(out));
    SET_VARSIZE(out, output_bytes);

    PG_FREE_IF_COPY(in1, 0);
    PG_FREE_IF_COPY(in2, 1);
    PG_RETURN_POINTER(out);
}

/*
 * Compare two strings by tsvector rules.
 *
 * if isPrefix = true then it returns zero value iff b has prefix a
 */
int4 tsCompareString(const char* a, int lena, const char* b, int lenb, bool prefix)
{
    int cmp;

    if (lena == 0) {
        if (prefix)
            cmp = 0; /* empty string is prefix of anything */
        else
            cmp = (lenb > 0) ? -1 : 0;
    } else if (lenb == 0) {
        cmp = (lena > 0) ? 1 : 0;
    } else {
        cmp = memcmp(a, b, Min(lena, lenb));

        if (prefix) {
            if (cmp == 0 && lena > lenb)
                cmp = 1; /* a is longer, so not a prefix of b */
        } else if (cmp == 0 && lena != lenb) {
            cmp = (lena < lenb) ? -1 : 1;
        }
    }

    return cmp;
}

/*
 * check weight info
 */
static bool checkclass_str(CHKVAL* chkval, WordEntry* val, QueryOperand* item)
{
    WordEntryPosVector* posvec = NULL;
    WordEntryPos* ptr = NULL;
    uint16 len;

    posvec = (WordEntryPosVector*)(chkval->values + SHORTALIGN(val->pos + val->len));

    len = posvec->npos;
    ptr = posvec->pos;

    while (len--) {
        if (item->weight & (1 << WEP_GETWEIGHT(*ptr)))
            return true;
        ptr++;
    }
    return false;
}

/*
 * is there value 'val' in array or not ?
 */
static bool checkcondition_str(const void* checkval, QueryOperand* val)
{
    CHKVAL* chkval = (CHKVAL*)checkval;
    WordEntry* StopLow = chkval->arrb;
    WordEntry* StopHigh = chkval->arre;
    WordEntry* StopMiddle = StopHigh;
    int difference = -1;
    bool res = false;

    /* Loop invariant: StopLow <= val < StopHigh */
    while (StopLow < StopHigh) {
        StopMiddle = StopLow + (StopHigh - StopLow) / 2;
        difference = tsCompareString(
            chkval->operand + val->distance, val->length, chkval->values + StopMiddle->pos, StopMiddle->len, false);

        if (difference == 0) {
            res = (val->weight && StopMiddle->haspos) ? checkclass_str(chkval, StopMiddle, val) : true;
            break;
        } else if (difference > 0)
            StopLow = StopMiddle + 1;
        else
            StopHigh = StopMiddle;
    }

    if (!res && val->prefix) {
        /*
         * there was a failed exact search, so we should scan further to find
         * a prefix match.
         */
        if (StopLow >= StopHigh)
            StopMiddle = StopHigh;

        while (res == false && StopMiddle < chkval->arre &&
               tsCompareString(chkval->operand + val->distance,
                   val->length,
                   chkval->values + StopMiddle->pos,
                   StopMiddle->len,
                   true) == 0) {
            res = (val->weight && StopMiddle->haspos) ? checkclass_str(chkval, StopMiddle, val) : true;

            StopMiddle++;
        }
    }

    return res;
}

/*
 * Evaluate tsquery boolean expression.
 *
 * chkcond is a callback function used to evaluate each VAL node in the query.
 * checkval can be used to pass information to the callback. TS_execute doesn't
 * do anything with it.
 * if calcnot is false, NOT expressions are always evaluated to be true. This
 * is used in ranking.
 */
bool TS_execute(
    QueryItem* curitem, void* checkval, bool calcnot, bool (*chkcond)(const void* checkval, QueryOperand* val))
{
    /* since this function recurses, it could be driven to stack overflow */
    check_stack_depth();

    if (curitem->type == QI_VAL)
        return chkcond(checkval, (QueryOperand*)curitem);

    switch (curitem->qoperator.oper) {
        case OP_NOT:
            if (calcnot)
                return !TS_execute(curitem + 1, checkval, calcnot, chkcond);
            else
                return true;

        case OP_AND:
            if (TS_execute(curitem + curitem->qoperator.left, checkval, calcnot, chkcond))
                return TS_execute(curitem + 1, checkval, calcnot, chkcond);
            else
                return false;

        case OP_OR:
            if (TS_execute(curitem + curitem->qoperator.left, checkval, calcnot, chkcond))
                return true;
            else
                return TS_execute(curitem + 1, checkval, calcnot, chkcond);

        default:
            ereport(ERROR,
                (errcode(ERRCODE_UNRECOGNIZED_NODE_TYPE),
                    errmsg("unrecognized operator: %d", curitem->qoperator.oper)));
    }

    /* not reachable, but keep compiler quiet */
    return false;
}

/*
 * Detect whether a tsquery boolean expression requires any positive matches
 * to values shown in the tsquery.
 *
 * This is needed to know whether a GIN index search requires full index scan.
 * For example, 'x & !y' requires a match of x, so it's sufficient to scan
 * entries for x; but 'x | !y' could match rows containing neither x nor y.
 */
bool tsquery_requires_match(QueryItem* curitem)
{
    /* since this function recurses, it could be driven to stack overflow */
    check_stack_depth();

    if (curitem->type == QI_VAL)
        return true;

    switch (curitem->qoperator.oper) {
        case OP_NOT:

            /*
             * Assume there are no required matches underneath a NOT.  For
             * some cases with nested NOTs, we could prove there's a required
             * match, but it seems unlikely to be worth the trouble.
             */
            return false;

        case OP_AND:
            /* If either side requires a match, we're good */
            if (tsquery_requires_match(curitem + curitem->qoperator.left))
                return true;
            else
                return tsquery_requires_match(curitem + 1);

        case OP_OR:
            /* Both sides must require a match */
            if (tsquery_requires_match(curitem + curitem->qoperator.left))
                return tsquery_requires_match(curitem + 1);
            else
                return false;

        default:
            ereport(ERROR,
                (errcode(ERRCODE_UNRECOGNIZED_NODE_TYPE),
                    errmsg("unrecognized operator: %d", curitem->qoperator.oper)));
    }

    /* not reachable, but keep compiler quiet */
    return false;
}

/*
 * boolean operations
 */
Datum ts_match_qv(PG_FUNCTION_ARGS)
{
    PG_RETURN_DATUM(DirectFunctionCall2(ts_match_vq, PG_GETARG_DATUM(1), PG_GETARG_DATUM(0)));
}

Datum ts_match_vq(PG_FUNCTION_ARGS)
{
    TSVector val = PG_GETARG_TSVECTOR(0);
    TSQuery query = PG_GETARG_TSQUERY(1);
    CHKVAL chkval;
    bool result = false;

    /* empty query matches nothing */
    if (!query->size) {
        PG_FREE_IF_COPY(val, 0);
        PG_FREE_IF_COPY(query, 1);
        PG_RETURN_BOOL(false);
    }

    chkval.arrb = ARRPTR(val);
    chkval.arre = chkval.arrb + val->size;
    chkval.values = STRPTR(val);
    chkval.operand = GETOPERAND(query);
    result = TS_execute(GETQUERY(query), &chkval, true, checkcondition_str);

    PG_FREE_IF_COPY(val, 0);
    PG_FREE_IF_COPY(query, 1);
    PG_RETURN_BOOL(result);
}

Datum ts_match_tt(PG_FUNCTION_ARGS)
{
    TSVector vector;
    TSQuery query;
    bool res = false;

    vector = DatumGetTSVector(DirectFunctionCall1(to_tsvector, PG_GETARG_DATUM(0)));
    query = DatumGetTSQuery(DirectFunctionCall1(plainto_tsquery, PG_GETARG_DATUM(1)));

    res = DatumGetBool(DirectFunctionCall2(ts_match_vq, TSVectorGetDatum(vector), TSQueryGetDatum(query)));

    pfree_ext(vector);
    pfree_ext(query);

    PG_RETURN_BOOL(res);
}

Datum ts_match_tq(PG_FUNCTION_ARGS)
{
    TSVector vector;
    TSQuery query = PG_GETARG_TSQUERY(1);
    bool res = false;

    vector = DatumGetTSVector(DirectFunctionCall1(to_tsvector, PG_GETARG_DATUM(0)));

    res = DatumGetBool(DirectFunctionCall2(ts_match_vq, TSVectorGetDatum(vector), TSQueryGetDatum(query)));

    pfree_ext(vector);
    PG_FREE_IF_COPY(query, 1);

    PG_RETURN_BOOL(res);
}

/*
 * ts_stat statistic function support
 */

/*
 * Returns the number of positions in value 'wptr' within tsvector 'txt',
 * that have a weight equal to one of the weights in 'weight' bitmask.
 */
static int check_weight(TSVector txt, WordEntry* wptr, int8 weight)
{
    int len = POSDATALEN(txt, wptr);
    int num = 0;
    WordEntryPos* ptr = POSDATAPTR(txt, wptr);

    while (len--) {
        if (weight & (1 << WEP_GETWEIGHT(*ptr)))
            num++;
        ptr++;
    }
    return num;
}

#define compareStatWord(a, e, t) tsCompareString((a)->lexeme, (a)->lenlexeme, STRPTR(t) + (e)->pos, (e)->len, false)

static void insertStatEntry(MemoryContext persistentContext, TSVectorStat* stat, TSVector txt, uint32 off)
{
    WordEntry* we = ARRPTR(txt) + off;
    StatEntry *node = stat->root, *pnode = NULL;
    int n, res = 0;
    uint32 depth = 1;

    if (stat->weight == 0)
        n = (we->haspos) ? POSDATALEN(txt, we) : 1;
    else
        n = (we->haspos) ? check_weight(txt, we, stat->weight) : 0;

    if (n == 0)
        return; /* nothing to insert */

    while (node != NULL) {
        res = compareStatWord(node, we, txt);

        if (res == 0) {
            break;
        } else {
            pnode = node;
            node = (res < 0) ? node->left : node->right;
        }
        depth++;
    }

    if (depth > stat->maxdepth)
        stat->maxdepth = depth;

    if (node == NULL) {
        node = (StatEntry*)MemoryContextAlloc(persistentContext, STATENTRYHDRSZ + we->len);
        node->left = node->right = NULL;
        node->ndoc = 1;
        node->nentry = n;
        node->lenlexeme = we->len;
        if (node->lenlexeme > 0) {
            errno_t ret = memcpy_s(node->lexeme, we->len, STRPTR(txt) + we->pos, node->lenlexeme);
            securec_check(ret, "\0", "\0");
        }

        if (pnode == NULL) {
            stat->root = node;
        } else {
            if (res < 0)
                pnode->left = node;
            else
                pnode->right = node;
        }

    } else {
        node->ndoc++;
        node->nentry += n;
    }
}

static void chooseNextStatEntry(
    MemoryContext persistentContext, TSVectorStat* stat, TSVector txt, uint32 low, uint32 high, uint32 offset)
{
    uint32 pos;
    uint32 middle = (low + high) >> 1;

    pos = (low + middle) >> 1;
    if (low != middle && pos >= offset && pos - offset < (uint32)(txt->size))
        insertStatEntry(persistentContext, stat, txt, pos - offset);
    pos = (high + middle + 1) >> 1;
    if (middle + 1 != high && pos >= offset && pos - offset < (uint32)(txt->size))
        insertStatEntry(persistentContext, stat, txt, pos - offset);

    if (low != middle)
        chooseNextStatEntry(persistentContext, stat, txt, low, middle, offset);
    if (high != middle + 1)
        chooseNextStatEntry(persistentContext, stat, txt, middle + 1, high, offset);
}

/*
 * This is written like a custom aggregate function, because the
 * original plan was to do just that. Unfortunately, an aggregate function
 * can't return a set, so that plan was abandoned. If that limitation is
 * lifted in the future, ts_stat could be a real aggregate function so that
 * you could use it like this:
 *
 *	 SELECT ts_stat(vector_column) FROM vector_table;
 *
 *	where vector_column is a tsvector-type column in vector_table.
 */

static TSVectorStat* ts_accum(MemoryContext persistentContext, TSVectorStat* stat, Datum data)
{
    TSVector txt = DatumGetTSVector(data);
    uint32 i, nbit = 0, offset;

    if (stat == NULL) { /* Init in first */
        stat = (TSVectorStat*)MemoryContextAllocZero(persistentContext, sizeof(TSVectorStat));
        stat->maxdepth = 1;
    }

    /* simple check of correctness */
    if (txt == NULL || txt->size == 0) {
        if (txt && txt != (TSVector)DatumGetPointer(data))
            pfree_ext(txt);
        return stat;
    }

    i = txt->size - 1;
    for (; i > 0; i >>= 1)
        nbit++;

    nbit = 1 << nbit;
    offset = (nbit - txt->size) / 2;

    insertStatEntry(persistentContext, stat, txt, (nbit >> 1) - offset);
    chooseNextStatEntry(persistentContext, stat, txt, 0, nbit, offset);

    return stat;
}

static void ts_setup_firstcall(FunctionCallInfo fcinfo, FuncCallContext* funcctx, TSVectorStat* stat)
{
    TupleDesc tupdesc;
    MemoryContext oldcontext;
    StatEntry* node = NULL;

    funcctx->user_fctx = (void*)stat;

    oldcontext = MemoryContextSwitchTo(funcctx->multi_call_memory_ctx);

    stat->stack = (StatEntry**)palloc0(sizeof(StatEntry*) * (stat->maxdepth + 1));
    stat->stackpos = 0;

    node = stat->root;
    /* find leftmost value */
    if (node == NULL)
        stat->stack[stat->stackpos] = NULL;
    else
        for (;;) {
            stat->stack[stat->stackpos] = node;
            if (node->left != NULL) {
                stat->stackpos++;
                node = node->left;
            } else
                break;
        }
    Assert(stat->stackpos <= stat->maxdepth);

    tupdesc = CreateTemplateTupleDesc(3, false);
    TupleDescInitEntry(tupdesc, (AttrNumber)1, "word", TEXTOID, -1, 0);
    TupleDescInitEntry(tupdesc, (AttrNumber)2, "ndoc", INT4OID, -1, 0);
    TupleDescInitEntry(tupdesc, (AttrNumber)3, "nentry", INT4OID, -1, 0);
    funcctx->tuple_desc = BlessTupleDesc(tupdesc);
    funcctx->attinmeta = TupleDescGetAttInMetadata(tupdesc);

    MemoryContextSwitchTo(oldcontext);
}

static StatEntry* walkStatEntryTree(TSVectorStat* stat)
{
    StatEntry* node = stat->stack[stat->stackpos];

    if (node == NULL)
        return NULL;

    if (node->ndoc != 0) {
        /* return entry itself: we already was at left sublink */
        return node;
    } else if (node->right && node->right != stat->stack[stat->stackpos + 1]) {
        /* go on right sublink */
        stat->stackpos++;
        node = node->right;

        /* find most-left value */
        for (;;) {
            stat->stack[stat->stackpos] = node;
            if (node->left != NULL) {
                stat->stackpos++;
                node = node->left;
            } else
                break;
        }
        Assert(stat->stackpos <= stat->maxdepth);
    } else {
        /* we already return all left subtree, itself and  right subtree */
        if (stat->stackpos == 0)
            return NULL;

        stat->stackpos--;
        return walkStatEntryTree(stat);
    }

    return node;
}

static Datum ts_process_call(FuncCallContext* funcctx)
{
    TSVectorStat* st = NULL;
    StatEntry* entry = NULL;
    errno_t rc;

    st = (TSVectorStat*)funcctx->user_fctx;
    entry = walkStatEntryTree(st);

    if (entry != NULL) {
        Datum result;
        char* values[3];
        char ndoc[16];
        char nentry[16];
        HeapTuple tuple;

        values[0] = (char*)palloc(entry->lenlexeme + 1);
        rc = memcpy_s(values[0], entry->lenlexeme, entry->lexeme, entry->lenlexeme);
        securec_check(rc, "\0", "\0");
        (values[0])[entry->lenlexeme] = '\0';
        rc = sprintf_s(ndoc, sizeof(ndoc), "%u", entry->ndoc);
        securec_check_ss(rc, "\0", "\0");
        rc = sprintf_s(nentry, sizeof(nentry), "%u", entry->nentry);
        securec_check_ss(rc, "\0", "\0");
        values[1] = ndoc;
        values[2] = nentry;
        tuple = BuildTupleFromCStrings(funcctx->attinmeta, values);
        result = HeapTupleGetDatum(tuple);
        pfree_ext(values[0]);

        /* mark entry as already visited */
        entry->ndoc = 0;

        return result;
    }

    return (Datum)0;
}

static TSVectorStat* ts_stat_sql(MemoryContext persistentContext, text* txt, text* ws)
{
    char* query = text_to_cstring(txt);
    int i;
    TSVectorStat* stat = NULL;
    bool isnull = false;
    Portal portal;
    SPIPlanPtr plan;

    if ((plan = SPI_prepare(query, 0, NULL)) == NULL)
        /* internal error */
        ereport(ERROR, (errcode(ERRCODE_SPI_PREPARE_FAILURE), errmsg("SPI_prepare(\"%s\") failed", query)));

    if ((portal = SPI_cursor_open(NULL, plan, NULL, NULL, true)) == NULL)
        /* internal error */
        ereport(ERROR, (errcode(ERRCODE_SPI_CURSOR_OPEN_FAILURE), errmsg("SPI_cursor_open(\"%s\") failed", query)));

    SPI_cursor_fetch(portal, true, 100);

    if (SPI_tuptable == NULL || SPI_tuptable->tupdesc->natts != 1 ||
        !is_expected_type(SPI_gettypeid(SPI_tuptable->tupdesc, 1), TSVECTOROID))
        ereport(
            ERROR, (errcode(ERRCODE_INVALID_PARAMETER_VALUE), errmsg("ts_stat query must return one tsvector column")));

    stat = (TSVectorStat*)MemoryContextAllocZero(persistentContext, sizeof(TSVectorStat));
    stat->maxdepth = 1;

    if (ws != NULL) {
        char* buf = NULL;

        buf = VARDATA(ws);
        while (buf - VARDATA(ws) < VARSIZE(ws) - VARHDRSZ) {
            if (pg_mblen(buf) == 1) {
                switch (*buf) {
                    case 'A':
                    case 'a':
                        stat->weight |= 1 << 3;
                        break;
                    case 'B':
                    case 'b':
                        stat->weight |= 1 << 2;
                        break;
                    case 'C':
                    case 'c':
                        stat->weight |= 1 << 1;
                        break;
                    case 'D':
                    case 'd':
                        stat->weight |= 1;
                        break;
                    default:
                        stat->weight |= 0;
                }
            }
            buf += pg_mblen(buf);
        }
    }

    while (SPI_processed > 0) {
        for (i = 0; i < (int)SPI_processed; i++) {
            Datum data = SPI_getbinval(SPI_tuptable->vals[i], SPI_tuptable->tupdesc, 1, &isnull);

            if (!isnull)
                stat = ts_accum(persistentContext, stat, data);
        }

        SPI_freetuptable(SPI_tuptable);
        SPI_tuptable = NULL;
        SPI_cursor_fetch(portal, true, 100);
    }

    SPI_freetuptable(SPI_tuptable);
    SPI_tuptable = NULL;
    SPI_cursor_close(portal);
    SPI_freeplan(plan);
    pfree_ext(query);

    return stat;
}

Datum ts_stat1(PG_FUNCTION_ARGS)
{
    ts_check_feature_disable();
    FuncCallContext* funcctx = NULL;
    Datum result;

    if (SRF_IS_FIRSTCALL()) {
        TSVectorStat* stat = NULL;
        text* txt = PG_GETARG_TEXT_P(0);
        int rc = 0;

        funcctx = SRF_FIRSTCALL_INIT();
        /*
         * Connect to SPI manager
         */
        SPI_STACK_LOG("connect", NULL, NULL);
        if ((rc = SPI_connect()) != SPI_OK_CONNECT)
            ereport(ERROR,
                (errcode(ERRCODE_SPI_CONNECTION_FAILURE),
                    errmsg("SPI_connect failed: %s", SPI_result_code_string(rc))));

        stat = ts_stat_sql(funcctx->multi_call_memory_ctx, txt, NULL);
        PG_FREE_IF_COPY(txt, 0);
        ts_setup_firstcall(fcinfo, funcctx, stat);
        SPI_STACK_LOG("finish", NULL, NULL);
        SPI_finish();
    }

    funcctx = SRF_PERCALL_SETUP();
    if ((result = ts_process_call(funcctx)) != (Datum)0)
        SRF_RETURN_NEXT(funcctx, result);
    SRF_RETURN_DONE(funcctx);
}

Datum ts_stat2(PG_FUNCTION_ARGS)
{
    ts_check_feature_disable();
    FuncCallContext* funcctx = NULL;
    Datum result;

    if (SRF_IS_FIRSTCALL()) {
        TSVectorStat* stat = NULL;
        text* txt = PG_GETARG_TEXT_P(0);
        text* ws = PG_GETARG_TEXT_P(1);
        int rc = 0;

        funcctx = SRF_FIRSTCALL_INIT();
        /*
         * Connect to SPI manager
         */
        SPI_STACK_LOG("connect", NULL, NULL);
        if ((rc = SPI_connect()) != SPI_OK_CONNECT)
            ereport(ERROR,
                (errcode(ERRCODE_SPI_CONNECTION_FAILURE),
                    errmsg("SPI_connect failed: %s", SPI_result_code_string(rc))));

        stat = ts_stat_sql(funcctx->multi_call_memory_ctx, txt, ws);
        PG_FREE_IF_COPY(txt, 0);
        PG_FREE_IF_COPY(ws, 1);
        ts_setup_firstcall(fcinfo, funcctx, stat);
        SPI_STACK_LOG("finish", NULL, NULL);
        SPI_finish();
    }

    funcctx = SRF_PERCALL_SETUP();
    if ((result = ts_process_call(funcctx)) != (Datum)0)
        SRF_RETURN_NEXT(funcctx, result);
    SRF_RETURN_DONE(funcctx);
}

/*
 * Triggers for automatic update of a tsvector column from text column(s)
 *
 * Trigger arguments are either
 *		name of tsvector col, name of tsconfig to use, name(s) of text col(s)
 *		name of tsvector col, name of regconfig col, name(s) of text col(s)
 * ie, tsconfig can either be specified by name, or indirectly as the
 * contents of a regconfig field in the row.  If the name is used, it must
 * be explicitly schema-qualified.
 */
Datum tsvector_update_trigger_byid(PG_FUNCTION_ARGS)
{
    return tsvector_update_trigger(fcinfo, false);
}

Datum tsvector_update_trigger_bycolumn(PG_FUNCTION_ARGS)
{
    return tsvector_update_trigger(fcinfo, true);
}

static Datum tsvector_update_trigger(PG_FUNCTION_ARGS, bool config_column)
{
    TriggerData* trigdata = NULL;
    Trigger* trigger = NULL;
    Relation rel;
    HeapTuple rettuple = NULL;
    int tsvector_attr_num, i;
    ParsedText prs;
    Datum datum;
    bool isnull = false;
    text* txt = NULL;
    Oid cfgId;

    /* Check call context */
    if (!CALLED_AS_TRIGGER(fcinfo)) /* internal error */
        ereport(ERROR,
            (errcode(ERRCODE_TRIGGERED_ACTION_EXCEPTION),
                errmsg("tsvector_update_trigger: not fired by trigger manager")));

    trigdata = (TriggerData*)fcinfo->context;
    if (!TRIGGER_FIRED_FOR_ROW(trigdata->tg_event))
        ereport(ERROR,
            (errcode(ERRCODE_TRIGGERED_ACTION_EXCEPTION), errmsg("tsvector_update_trigger: must be fired for row")));
    if (!TRIGGER_FIRED_BEFORE(trigdata->tg_event))
        ereport(ERROR,
            (errcode(ERRCODE_TRIGGERED_ACTION_EXCEPTION),
                errmsg("tsvector_update_trigger: must be fired BEFORE event")));

    if (TRIGGER_FIRED_BY_INSERT(trigdata->tg_event))
        rettuple = trigdata->tg_trigtuple;
    else if (TRIGGER_FIRED_BY_UPDATE(trigdata->tg_event))
        rettuple = trigdata->tg_newtuple;
    else
        ereport(ERROR,
            (errcode(ERRCODE_TRIGGERED_ACTION_EXCEPTION),
                errmsg("tsvector_update_trigger: must be fired for INSERT or UPDATE")));

    trigger = trigdata->tg_trigger;
    rel = trigdata->tg_relation;

    if (trigger->tgnargs < 3)
        ereport(ERROR,
            (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
                errmsg("tsvector_update_trigger: arguments must be tsvector_field, ts_config, text_field1, ...)")));

    /* Find the target tsvector column */
    tsvector_attr_num = SPI_fnumber(rel->rd_att, trigger->tgargs[0]);
    if (tsvector_attr_num == SPI_ERROR_NOATTRIBUTE)
        ereport(ERROR,
            (errcode(ERRCODE_UNDEFINED_COLUMN), errmsg("tsvector column \"%s\" does not exist", trigger->tgargs[0])));
    if (!is_expected_type(SPI_gettypeid(rel->rd_att, tsvector_attr_num), TSVECTOROID))
        ereport(ERROR,
            (errcode(ERRCODE_DATATYPE_MISMATCH), errmsg("column \"%s\" is not of tsvector type", trigger->tgargs[0])));

    /* Find the configuration to use */
    if (config_column) {
        int config_attr_num;

        config_attr_num = SPI_fnumber(rel->rd_att, trigger->tgargs[1]);
        if (config_attr_num == SPI_ERROR_NOATTRIBUTE)
            ereport(ERROR,
                (errcode(ERRCODE_UNDEFINED_COLUMN),
                    errmsg("configuration column \"%s\" does not exist", trigger->tgargs[1])));
        if (!is_expected_type(SPI_gettypeid(rel->rd_att, config_attr_num), REGCONFIGOID))
            ereport(ERROR,
                (errcode(ERRCODE_DATATYPE_MISMATCH),
                    errmsg("column \"%s\" is not of regconfig type", trigger->tgargs[1])));

        datum = SPI_getbinval(rettuple, rel->rd_att, config_attr_num, &isnull);
        if (isnull)
            ereport(ERROR,
                (errcode(ERRCODE_NULL_VALUE_NOT_ALLOWED),
                    errmsg("configuration column \"%s\" must not be null", trigger->tgargs[1])));
        cfgId = DatumGetObjectId(datum);
    } else {
        List* names = NIL;

        names = stringToQualifiedNameList(trigger->tgargs[1]);
        /* require a schema so that results are not search path dependent */
        if (list_length(names) < 2)
            ereport(ERROR,
                (errcode(ERRCODE_INVALID_PARAMETER_VALUE),
                    errmsg("text search configuration name \"%s\" must be schema-qualified", trigger->tgargs[1])));
        cfgId = get_ts_config_oid(names, false);
    }

    /* initialize parse state */
    prs.lenwords = 32;
    prs.curwords = 0;
    prs.pos = 0;
    prs.words = (ParsedWord*)palloc(sizeof(ParsedWord) * prs.lenwords);

    /* find all words in indexable column(s) */
    for (i = 2; i < trigger->tgnargs; i++) {
        int numattr;

        numattr = SPI_fnumber(rel->rd_att, trigger->tgargs[i]);
        if (numattr == SPI_ERROR_NOATTRIBUTE)
            ereport(
                ERROR, (errcode(ERRCODE_UNDEFINED_COLUMN), errmsg("column \"%s\" does not exist", trigger->tgargs[i])));
        if (!is_text_type(SPI_gettypeid(rel->rd_att, numattr)))
            ereport(ERROR,
                (errcode(ERRCODE_DATATYPE_MISMATCH),
                    errmsg("column \"%s\" is not of a character type", trigger->tgargs[i])));

        datum = SPI_getbinval(rettuple, rel->rd_att, numattr, &isnull);
        if (isnull)
            continue;

        txt = DatumGetTextP(datum);

        parsetext(cfgId, &prs, VARDATA(txt), VARSIZE(txt) - VARHDRSZ);

        if (txt != (text*)DatumGetPointer(datum))
            pfree_ext(txt);
    }

    /* make tsvector value */
    if (prs.curwords) {
        datum = PointerGetDatum(make_tsvector(&prs));
        rettuple = SPI_modifytuple(rel, rettuple, 1, &tsvector_attr_num, &datum, NULL);
        pfree(DatumGetPointer(datum));
    } else {
        TSVector out = (TSVector)palloc(CALCDATASIZE(0, 0));

        SET_VARSIZE(out, CALCDATASIZE(0, 0));
        out->size = 0;
        datum = PointerGetDatum(out);
        rettuple = SPI_modifytuple(rel, rettuple, 1, &tsvector_attr_num, &datum, NULL);
        pfree_ext(prs.words);
    }

    if (rettuple == NULL) /* internal error */
        ereport(ERROR,
            (errcode(ERRCODE_NOT_NULL_VIOLATION),
                errmsg("tsvector_update_trigger: %d returned by SPI_modifytuple", SPI_result)));

    return PointerGetDatum(rettuple);
}