25#include "utils/fmgrprotos.h"
74 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
75 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
76 errdetail(
"Initial element must be an array."));
82 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
83 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
90 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
91 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
92 errdetail(
"Value of \"%s\" must be an array of attribute numbers.",
99 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
100 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
101 errdetail(
"Attribute lists can only contain attribute numbers."));
107 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
108 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
109 errdetail(
"Value of \"%s\" must be an integer.",
115 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
116 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
141 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
142 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
147 if (!
parse->found_attributes)
150 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
151 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
152 errdetail(
"Item must contain \"%s\" key.",
157 if (!
parse->found_ndistinct)
160 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
161 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
162 errdetail(
"Item must contain \"%s\" key.",
175 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
176 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
177 errdetail(
"The \"%s\" key must contain an array of at least %d and no more than %d attributes.",
188 for (
int i = 0;
i < natts;
i++)
196 parse->ndistinct = 0;
197 parse->found_attributes =
false;
198 parse->found_ndistinct =
false;
217 switch (
parse->state)
229 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
230 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
231 errdetail(
"Array found in unexpected place."));
249 switch (
parse->state)
263 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
264 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
265 errdetail(
"The \"%s\" key must be a non-empty array.",
278 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
279 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
280 errdetail(
"Item array cannot be empty."));
290 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
291 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
292 errdetail(
"Array found in unexpected place."));
313 if (
parse->found_attributes)
316 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
317 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
318 errdetail(
"Multiple \"%s\" keys are not allowed.",
322 parse->found_attributes =
true;
329 if (
parse->found_ndistinct)
332 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
333 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
334 errdetail(
"Multiple \"%s\" keys are not allowed.",
338 parse->found_ndistinct =
true;
344 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
345 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
346 errdetail(
"Only allowed keys are \"%s\" and \"%s\".",
363 switch (
parse->state)
370 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
371 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
372 errdetail(
"Attribute number array cannot be null."));
380 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
381 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
382 errdetail(
"Item list elements cannot be null."));
388 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
389 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
415 return ((
cur > prev) || (
cur < 0));
433 switch (
parse->state)
441 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
442 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
454 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
455 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
468 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
469 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
470 errdetail(
"Invalid \"%s\" element: %d cannot follow %d.",
495 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
496 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
503 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
504 errmsg(
"malformed pg_ndistinct: \"%s\"",
parse->str),
524 if (
a->nattributes !=
b->nattributes)
527 for (
int i = 0;
i <
a->nattributes;
i++)
529 if (
a->attributes[
i] !=
b->attributes[
i])
600 int item_most_attrs = 0;
601 int item_most_attrs_idx = 0;
603 switch (
parse->state)
614 "cannot have empty item list after parsing success.");
620 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
621 errmsg(
"malformed pg_ndistinct: \"%s\"",
str),
628 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
629 errmsg(
"malformed pg_ndistinct: \"%s\"",
str),
649 for (
int j = 0;
j <
i;
j++)
656 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
657 errmsg(
"malformed pg_ndistinct: \"%s\"",
str),
658 errdetail(
"Duplicated \"%s\" array found: [%s]",
681 item_most_attrs_idx =
i;
694 if (
i == item_most_attrs_idx)
698 &ndistinct->
items[item_most_attrs_idx]))
706 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
707 errmsg(
"malformed pg_ndistinct: \"%s\"",
str),
708 errdetail(
"\"%s\" array: [%s] must be a subset of array: [%s]",
710 item_list, refitem_list));
754 sem_action.
semstate = (
void *) &parse_state;
785 errcode(ERRCODE_INVALID_TEXT_REPRESENTATION),
786 errmsg(
"malformed pg_ndistinct: \"%s\"",
str),
817 elog(
ERROR,
"invalid zero-length attribute array in MVNDistinct");
842 (
errcode(ERRCODE_FEATURE_NOT_SUPPORTED),
843 errmsg(
"cannot accept a value of type %s",
"pg_ndistinct")));
Datum byteasend(PG_FUNCTION_ARGS)
int errdetail(const char *fmt,...)
int errcode(int sqlerrcode)
int errmsg(const char *fmt,...)
#define errsave(context,...)
#define ereport(elevel,...)
#define PG_GETARG_BYTEA_PP(n)
#define PG_RETURN_BYTEA_P(x)
#define PG_RETURN_CSTRING(x)
#define PG_GETARG_CSTRING(n)
Assert(PointerIsAligned(start, uint64))
JsonParseErrorType pg_parse_json(JsonLexContext *lex, const JsonSemAction *sem)
JsonLexContext * makeJsonLexContextCstringLen(JsonLexContext *lex, const char *json, size_t len, int encoding, bool need_escapes)
void freeJsonLexContext(JsonLexContext *lex)
List * lappend(List *list, void *datum)
List * lappend_int(List *list, int datum)
void list_free(List *list)
void list_free_deep(List *list)
void pfree(void *pointer)
void * palloc0(Size size)
#define SOFT_ERROR_OCCURRED(escontext)
MVNDistinct * statext_ndistinct_deserialize(bytea *data)
bytea * statext_ndistinct_serialize(MVNDistinct *ndistinct)
int32 pg_strtoint32_safe(const char *s, Node *escontext)
int16 pg_strtoint16_safe(const char *s, Node *escontext)
static int list_length(const List *l)
static void * list_nth(const List *list, int n)
static int list_nth_int(const List *list, int n)
Datum pg_ndistinct_out(PG_FUNCTION_ARGS)
static char * item_attnum_list(const MVNDistinctItem *item)
static JsonParseErrorType ndistinct_array_element_start(void *state, bool isnull)
static JsonParseErrorType ndistinct_array_end(void *state)
static JsonParseErrorType ndistinct_object_start(void *state)
Datum pg_ndistinct_in(PG_FUNCTION_ARGS)
static JsonParseErrorType ndistinct_array_start(void *state)
Datum pg_ndistinct_recv(PG_FUNCTION_ARGS)
static JsonParseErrorType ndistinct_scalar(void *state, char *token, JsonTokenType tokentype)
static bytea * build_mvndistinct(NDistinctParseState *parse, char *str)
static JsonParseErrorType ndistinct_object_end(void *state)
static bool item_attributes_eq(const MVNDistinctItem *a, const MVNDistinctItem *b)
@ NDIST_EXPECT_ATTNUM_LIST
static JsonParseErrorType ndistinct_object_field_start(void *state, char *fname, bool isnull)
Datum pg_ndistinct_send(PG_FUNCTION_ARGS)
static bool item_has_attnum(const MVNDistinctItem *item, AttrNumber attnum)
static bool item_is_attnum_subset(const MVNDistinctItem *item, const MVNDistinctItem *refitem)
static bool valid_subsequent_attnum(AttrNumber prev, AttrNumber cur)
static struct subre * parse(struct vars *v, int stopper, int type, struct state *init, struct state *final)
#define STATS_NDISTINCT_MAGIC
#define STATS_NDISTINCT_TYPE_BASIC
#define STATS_MAX_DIMENSIONS
void appendStringInfo(StringInfo str, const char *fmt,...)
void appendStringInfoString(StringInfo str, const char *s)
void appendStringInfoChar(StringInfo str, char ch)
void initStringInfo(StringInfo str)
json_struct_action array_end
json_struct_action object_start
json_ofield_action object_field_start
json_aelem_action array_element_start
json_scalar_action scalar
json_aelem_action array_element_end
json_struct_action array_start
json_struct_action object_end
json_ofield_action object_field_end
MVNDistinctItem items[FLEXIBLE_ARRAY_MEMBER]
NDistinctSemanticState state