2015-02-11 21:39:41 +01:00
|
|
|
/************* jsonudf C++ Program Source Code File (.CPP) *************/
|
|
|
|
/* PROGRAM NAME: jsonudf Version 1.0 */
|
|
|
|
/* (C) Copyright to the author Olivier BERTRAND 2015 */
|
|
|
|
/* This program are the JSON User Defined Functions . */
|
|
|
|
/***********************************************************************/
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Include relevant sections of the MariaDB header file. */
|
|
|
|
/***********************************************************************/
|
|
|
|
#include <my_global.h>
|
2015-02-22 17:53:02 +01:00
|
|
|
#include <mysqld.h>
|
2015-02-11 21:39:41 +01:00
|
|
|
#include <mysql.h>
|
2015-02-22 17:53:02 +01:00
|
|
|
#include <sql_error.h>
|
2015-02-11 21:39:41 +01:00
|
|
|
|
|
|
|
#include "global.h"
|
|
|
|
#include "plgdbsem.h"
|
|
|
|
#include "json.h"
|
|
|
|
|
2015-02-22 17:53:02 +01:00
|
|
|
#define MEMFIX 512
|
2015-05-05 11:37:21 +02:00
|
|
|
#define UDF_EXEC_ARGS \
|
|
|
|
UDF_INIT*, UDF_ARGS*, char*, unsigned long*, char*, char*
|
2015-02-22 17:53:02 +01:00
|
|
|
|
|
|
|
uint GetJsonGrpSize(void);
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
extern "C" {
|
|
|
|
DllExport my_bool Json_Value_init(UDF_INIT*, UDF_ARGS*, char*);
|
2015-05-05 11:37:21 +02:00
|
|
|
DllExport char *Json_Value(UDF_EXEC_ARGS);
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport void Json_Value_deinit(UDF_INIT*);
|
2015-05-05 11:37:21 +02:00
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport my_bool Json_Array_init(UDF_INIT*, UDF_ARGS*, char*);
|
2015-05-05 11:37:21 +02:00
|
|
|
DllExport char *Json_Array(UDF_EXEC_ARGS);
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport void Json_Array_deinit(UDF_INIT*);
|
2015-05-05 11:37:21 +02:00
|
|
|
|
2015-02-22 17:53:02 +01:00
|
|
|
DllExport my_bool Json_Array_Add_init(UDF_INIT*, UDF_ARGS*, char*);
|
2015-05-05 11:37:21 +02:00
|
|
|
DllExport char *Json_Array_Add(UDF_EXEC_ARGS);
|
2015-02-22 17:53:02 +01:00
|
|
|
DllExport void Json_Array_Add_deinit(UDF_INIT*);
|
2015-05-05 11:37:21 +02:00
|
|
|
|
|
|
|
DllExport my_bool Json_Array_Delete_init(UDF_INIT*, UDF_ARGS*, char*);
|
|
|
|
DllExport char *Json_Array_Delete(UDF_EXEC_ARGS);
|
|
|
|
DllExport void Json_Array_Delete_deinit(UDF_INIT*);
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport my_bool Json_Object_init(UDF_INIT*, UDF_ARGS*, char*);
|
2015-05-05 11:37:21 +02:00
|
|
|
DllExport char *Json_Object(UDF_EXEC_ARGS);
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport void Json_Object_deinit(UDF_INIT*);
|
2015-05-05 11:37:21 +02:00
|
|
|
|
2015-02-24 23:18:04 +01:00
|
|
|
DllExport my_bool Json_Object_Nonull_init(UDF_INIT*, UDF_ARGS*, char*);
|
2015-05-05 11:37:21 +02:00
|
|
|
DllExport char *Json_Object_Nonull(UDF_EXEC_ARGS);
|
2015-02-24 23:18:04 +01:00
|
|
|
DllExport void Json_Object_Nonull_deinit(UDF_INIT*);
|
2015-05-05 11:37:21 +02:00
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport my_bool Json_Array_Grp_init(UDF_INIT*, UDF_ARGS*, char*);
|
|
|
|
DllExport void Json_Array_Grp_add(UDF_INIT *, UDF_ARGS *, char *, char *);
|
2015-05-05 11:37:21 +02:00
|
|
|
DllExport char *Json_Array_Grp(UDF_EXEC_ARGS);
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport void Json_Array_Grp_clear(UDF_INIT *, char *, char *);
|
|
|
|
DllExport void Json_Array_Grp_deinit(UDF_INIT*);
|
2015-05-05 11:37:21 +02:00
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport my_bool Json_Object_Grp_init(UDF_INIT*, UDF_ARGS*, char*);
|
|
|
|
DllExport void Json_Object_Grp_add(UDF_INIT *, UDF_ARGS *, char *, char *);
|
2015-05-05 11:37:21 +02:00
|
|
|
DllExport char *Json_Object_Grp(UDF_EXEC_ARGS);
|
2015-02-11 21:39:41 +01:00
|
|
|
DllExport void Json_Object_Grp_clear(UDF_INIT *, char *, char *);
|
|
|
|
DllExport void Json_Object_Grp_deinit(UDF_INIT*);
|
|
|
|
} // extern "C"
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Allocate and initialise the memory area. */
|
|
|
|
/***********************************************************************/
|
2015-02-22 17:53:02 +01:00
|
|
|
static my_bool JsonInit(UDF_INIT *initid, char *message,
|
|
|
|
unsigned long reslen, unsigned long memlen)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
2015-02-19 00:59:02 +01:00
|
|
|
PGLOBAL g = PlugInit(NULL, memlen);
|
2015-02-11 21:39:41 +01:00
|
|
|
|
|
|
|
if (!g) {
|
|
|
|
strcpy(message, "Allocation error");
|
|
|
|
return true;
|
|
|
|
} else if (g->Sarea_Size == 0) {
|
|
|
|
strcpy(message, g->Message);
|
|
|
|
PlugExit(g);
|
|
|
|
return true;
|
|
|
|
} else
|
|
|
|
initid->ptr = (char*)g;
|
|
|
|
|
|
|
|
initid->maybe_null = false;
|
2015-02-19 00:59:02 +01:00
|
|
|
initid->max_length = reslen;
|
2015-02-11 21:39:41 +01:00
|
|
|
return false;
|
|
|
|
} // end of Json_Object_init
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Returns true if the argument is a JSON string. */
|
|
|
|
/***********************************************************************/
|
|
|
|
static my_bool IsJson(UDF_ARGS *args, int i)
|
|
|
|
{
|
|
|
|
return (args->arg_type[i] == STRING_RESULT &&
|
|
|
|
!strnicmp(args->attributes[i], "Json_", 5));
|
|
|
|
} // end of IsJson
|
|
|
|
|
2015-02-19 00:59:02 +01:00
|
|
|
/***********************************************************************/
|
|
|
|
/* Calculate the reslen and memlen needed by a function. */
|
|
|
|
/***********************************************************************/
|
|
|
|
static my_bool CalcLen(UDF_ARGS *args, my_bool obj,
|
|
|
|
unsigned long& reslen, unsigned long& memlen)
|
|
|
|
{
|
|
|
|
unsigned long i, k;
|
|
|
|
reslen = args->arg_count + 2;
|
|
|
|
|
|
|
|
// Calculate the result max length
|
|
|
|
for (i = 0; i < args->arg_count; i++) {
|
|
|
|
if (obj) {
|
|
|
|
if (!(k = args->attribute_lengths[i]))
|
|
|
|
k = strlen(args->attributes[i]);
|
|
|
|
|
|
|
|
reslen += (k + 3); // For quotes and :
|
|
|
|
} // endif obj
|
|
|
|
|
|
|
|
switch (args->arg_type[i]) {
|
|
|
|
case STRING_RESULT:
|
|
|
|
if (IsJson(args, i))
|
|
|
|
reslen += args->lengths[i];
|
|
|
|
else
|
|
|
|
reslen += (args->lengths[i] + 1) * 2; // Pessimistic !
|
|
|
|
|
|
|
|
break;
|
|
|
|
case INT_RESULT:
|
|
|
|
reslen += 20;
|
|
|
|
break;
|
|
|
|
case REAL_RESULT:
|
|
|
|
reslen += 31;
|
|
|
|
break;
|
|
|
|
case DECIMAL_RESULT:
|
|
|
|
reslen += (args->lengths[i] + 7); // 6 decimals
|
|
|
|
break;
|
|
|
|
case TIME_RESULT:
|
|
|
|
case ROW_RESULT:
|
|
|
|
case IMPOSSIBLE_RESULT:
|
|
|
|
default:
|
|
|
|
// What should we do here ?
|
|
|
|
break;
|
|
|
|
} // endswitch arg_type
|
|
|
|
|
|
|
|
} // endfor i
|
|
|
|
|
|
|
|
// Calculate the amount of memory needed
|
2015-02-22 17:53:02 +01:00
|
|
|
memlen = MEMFIX + sizeof(JOUTSTR) + reslen;
|
2015-02-19 00:59:02 +01:00
|
|
|
|
|
|
|
for (i = 0; i < args->arg_count; i++) {
|
|
|
|
memlen += (args->lengths[i] + sizeof(JVALUE));
|
|
|
|
|
|
|
|
if (obj) {
|
|
|
|
if (!(k = args->attribute_lengths[i]))
|
|
|
|
k = strlen(args->attributes[i]);
|
|
|
|
|
|
|
|
memlen += (k + sizeof(JOBJECT) + sizeof(JPAIR));
|
|
|
|
} else
|
|
|
|
memlen += sizeof(JARRAY);
|
|
|
|
|
|
|
|
switch (args->arg_type[i]) {
|
|
|
|
case STRING_RESULT:
|
|
|
|
if (IsJson(args, i))
|
|
|
|
memlen += args->lengths[i] * 5; // Estimate parse memory
|
|
|
|
|
|
|
|
memlen += sizeof(TYPVAL<PSZ>);
|
|
|
|
break;
|
|
|
|
case INT_RESULT:
|
|
|
|
memlen += sizeof(TYPVAL<int>);
|
|
|
|
break;
|
|
|
|
case REAL_RESULT:
|
|
|
|
case DECIMAL_RESULT:
|
|
|
|
memlen += sizeof(TYPVAL<double>);
|
|
|
|
break;
|
|
|
|
case TIME_RESULT:
|
|
|
|
case ROW_RESULT:
|
|
|
|
case IMPOSSIBLE_RESULT:
|
|
|
|
default:
|
|
|
|
// What should we do here ?
|
|
|
|
break;
|
|
|
|
} // endswitch arg_type
|
|
|
|
|
|
|
|
} // endfor i
|
|
|
|
|
|
|
|
return false;
|
|
|
|
} // end of CalcLen
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
/***********************************************************************/
|
|
|
|
/* Make a zero terminated string from the passed argument. */
|
|
|
|
/***********************************************************************/
|
|
|
|
static PSZ MakePSZ(PGLOBAL g, UDF_ARGS *args, int i)
|
|
|
|
{
|
|
|
|
if (args->args[i]) {
|
|
|
|
int n = args->lengths[i];
|
|
|
|
PSZ s = (PSZ)PlugSubAlloc(g, NULL, n + 1);
|
|
|
|
|
|
|
|
memcpy(s, args->args[i], n);
|
|
|
|
s[n] = 0;
|
|
|
|
return s;
|
|
|
|
} else
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
} // end of MakePSZ
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Make a valid key from the passed argument. */
|
|
|
|
/***********************************************************************/
|
|
|
|
static PSZ MakeKey(PGLOBAL g, UDF_ARGS *args, int i)
|
|
|
|
{
|
|
|
|
int n = args->attribute_lengths[i];
|
|
|
|
bool b; // true if attribute is zero terminated
|
|
|
|
PSZ p, s = args->attributes[i];
|
|
|
|
|
|
|
|
if (s && *s && (n || *s == '\'')) {
|
|
|
|
if ((b = (!n || !s[n])))
|
|
|
|
n = strlen(s);
|
|
|
|
|
|
|
|
if (n > 5 && IsJson(args, i)) {
|
|
|
|
s += 5;
|
|
|
|
n -= 5;
|
|
|
|
} else if (*s == '\'' && s[n-1] == '\'') {
|
|
|
|
s++;
|
|
|
|
n -= 2;
|
|
|
|
b = false;
|
|
|
|
} // endif *s
|
|
|
|
|
|
|
|
if (n < 1)
|
|
|
|
return "Key";
|
|
|
|
|
|
|
|
if (!b) {
|
|
|
|
p = (PSZ)PlugSubAlloc(g, NULL, n + 1);
|
|
|
|
memcpy(p, s, n);
|
|
|
|
p[n] = 0;
|
|
|
|
s = p;
|
|
|
|
} // endif b
|
|
|
|
|
|
|
|
} // endif s
|
|
|
|
|
|
|
|
return s;
|
|
|
|
} // end of MakeKey
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Make a JSON value from the passed argument. */
|
|
|
|
/***********************************************************************/
|
|
|
|
static PJVAL MakeValue(PGLOBAL g, UDF_ARGS *args, int i)
|
|
|
|
{
|
2015-03-01 19:29:56 +01:00
|
|
|
char *sap = (args->arg_count > (unsigned)i) ? args->args[i] : NULL;
|
2015-02-22 17:53:02 +01:00
|
|
|
PJSON jsp;
|
2015-02-11 21:39:41 +01:00
|
|
|
PJVAL jvp = new(g) JVALUE;
|
|
|
|
|
2015-02-19 00:59:02 +01:00
|
|
|
if (sap) switch (args->arg_type[i]) {
|
2015-02-11 21:39:41 +01:00
|
|
|
case STRING_RESULT:
|
2015-02-19 00:59:02 +01:00
|
|
|
if (args->lengths[i]) {
|
2015-02-22 17:53:02 +01:00
|
|
|
if (IsJson(args, i)) {
|
|
|
|
if (!(jsp = ParseJson(g, sap, args->lengths[i], 0)))
|
|
|
|
push_warning(current_thd, Sql_condition::WARN_LEVEL_WARN, 0,
|
|
|
|
g->Message);
|
|
|
|
|
|
|
|
if (jsp && jsp->GetType() == TYPE_JVAL)
|
|
|
|
jvp = (PJVAL)jsp;
|
|
|
|
else
|
|
|
|
jvp->SetValue(jsp);
|
|
|
|
|
|
|
|
} else
|
2015-02-19 00:59:02 +01:00
|
|
|
jvp->SetString(g, MakePSZ(g, args, i));
|
2015-02-11 21:39:41 +01:00
|
|
|
|
|
|
|
} // endif str
|
|
|
|
|
|
|
|
break;
|
|
|
|
case INT_RESULT:
|
2015-07-22 13:21:55 +02:00
|
|
|
jvp->SetInteger(g, (int)*(long long*)sap);
|
2015-02-11 21:39:41 +01:00
|
|
|
break;
|
|
|
|
case REAL_RESULT:
|
2015-02-19 00:59:02 +01:00
|
|
|
jvp->SetFloat(g, *(double*)sap);
|
2015-02-11 21:39:41 +01:00
|
|
|
break;
|
|
|
|
case DECIMAL_RESULT:
|
|
|
|
jvp->SetFloat(g, atof(MakePSZ(g, args, i)));
|
|
|
|
break;
|
|
|
|
case TIME_RESULT:
|
|
|
|
case ROW_RESULT:
|
|
|
|
case IMPOSSIBLE_RESULT:
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
} // endswitch arg_type
|
|
|
|
|
|
|
|
return jvp;
|
|
|
|
} // end of MakeValue
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Make a Json value containing the parameter. */
|
|
|
|
/***********************************************************************/
|
|
|
|
my_bool Json_Value_init(UDF_INIT *initid, UDF_ARGS *args, char *message)
|
|
|
|
{
|
2015-02-19 00:59:02 +01:00
|
|
|
unsigned long reslen, memlen;
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
if (args->arg_count > 1) {
|
|
|
|
strcpy(message, "Json_Value cannot accept more than 1 argument");
|
|
|
|
return true;
|
2015-02-19 00:59:02 +01:00
|
|
|
} else
|
|
|
|
CalcLen(args, false, reslen, memlen);
|
2015-02-11 21:39:41 +01:00
|
|
|
|
2015-02-19 00:59:02 +01:00
|
|
|
return JsonInit(initid, message, reslen, memlen);
|
2015-02-11 21:39:41 +01:00
|
|
|
} // end of Json_Value_init
|
|
|
|
|
|
|
|
char *Json_Value(UDF_INIT *initid, UDF_ARGS *args, char *result,
|
2015-05-09 17:30:20 +02:00
|
|
|
unsigned long *res_length, char *, char *)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
PJVAL jvp;
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
jvp = MakeValue(g, args, 0);
|
|
|
|
|
|
|
|
if (!(str = Serialize(g, jvp, NULL, 0)))
|
|
|
|
str = strcpy(result, g->Message);
|
|
|
|
|
|
|
|
*res_length = strlen(str);
|
|
|
|
return str;
|
|
|
|
} // end of Json_Value
|
|
|
|
|
|
|
|
void Json_Value_deinit(UDF_INIT* initid)
|
|
|
|
{
|
|
|
|
PlugExit((PGLOBAL)initid->ptr);
|
|
|
|
} // end of Json_Value_deinit
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Make a Json array containing all the parameters. */
|
|
|
|
/***********************************************************************/
|
|
|
|
my_bool Json_Array_init(UDF_INIT *initid, UDF_ARGS *args, char *message)
|
|
|
|
{
|
2015-02-19 00:59:02 +01:00
|
|
|
unsigned long reslen, memlen;
|
|
|
|
|
|
|
|
CalcLen(args, false, reslen, memlen);
|
|
|
|
return JsonInit(initid, message, reslen, memlen);
|
2015-02-11 21:39:41 +01:00
|
|
|
} // end of Json_Array_init
|
|
|
|
|
|
|
|
char *Json_Array(UDF_INIT *initid, UDF_ARGS *args, char *result,
|
2015-05-09 17:30:20 +02:00
|
|
|
unsigned long *res_length, char *, char *)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
uint i;
|
|
|
|
PJAR arp;
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
arp = new(g) JARRAY;
|
|
|
|
|
|
|
|
for (i = 0; i < args->arg_count; i++)
|
|
|
|
arp->AddValue(g, MakeValue(g, args, i));
|
|
|
|
|
|
|
|
arp->InitArray(g);
|
|
|
|
|
|
|
|
if (!(str = Serialize(g, arp, NULL, 0)))
|
|
|
|
str = strcpy(result, g->Message);
|
|
|
|
|
|
|
|
*res_length = strlen(str);
|
|
|
|
return str;
|
|
|
|
} // end of Json_Array
|
|
|
|
|
|
|
|
void Json_Array_deinit(UDF_INIT* initid)
|
|
|
|
{
|
|
|
|
PlugExit((PGLOBAL)initid->ptr);
|
|
|
|
} // end of Json_Array_deinit
|
|
|
|
|
2015-02-22 17:53:02 +01:00
|
|
|
/***********************************************************************/
|
|
|
|
/* Add values to a Json array. */
|
|
|
|
/***********************************************************************/
|
|
|
|
my_bool Json_Array_Add_init(UDF_INIT *initid, UDF_ARGS *args, char *message)
|
|
|
|
{
|
|
|
|
unsigned long reslen, memlen;
|
|
|
|
|
|
|
|
if (args->arg_count < 2) {
|
|
|
|
strcpy(message, "Json_Value_Add must have at least 2 arguments");
|
|
|
|
return true;
|
|
|
|
} else if (!IsJson(args, 0)) {
|
2015-03-01 19:20:40 +01:00
|
|
|
strcpy(message, "Json_Value_Add first argument must be a json item");
|
2015-02-22 17:53:02 +01:00
|
|
|
return true;
|
|
|
|
} else
|
|
|
|
CalcLen(args, false, reslen, memlen);
|
|
|
|
|
|
|
|
return JsonInit(initid, message, reslen, memlen);
|
|
|
|
} // end of Json_Array_Add_init
|
|
|
|
|
|
|
|
char *Json_Array_Add(UDF_INIT *initid, UDF_ARGS *args, char *result,
|
2015-05-09 17:30:20 +02:00
|
|
|
unsigned long *res_length, char *, char *)
|
2015-02-22 17:53:02 +01:00
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
PJVAL jvp;
|
|
|
|
PJAR arp;
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
jvp = MakeValue(g, args, 0);
|
|
|
|
|
|
|
|
if (jvp->GetValType() != TYPE_JAR) {
|
|
|
|
arp = new(g) JARRAY;
|
|
|
|
arp->AddValue(g, jvp);
|
|
|
|
} else
|
|
|
|
arp = jvp->GetArray();
|
|
|
|
|
|
|
|
for (uint i = 1; i < args->arg_count; i++)
|
|
|
|
arp->AddValue(g, MakeValue(g, args, i));
|
|
|
|
|
|
|
|
arp->InitArray(g);
|
|
|
|
|
|
|
|
if (!(str = Serialize(g, arp, NULL, 0)))
|
|
|
|
str = strcpy(result, g->Message);
|
|
|
|
|
|
|
|
*res_length = strlen(str);
|
|
|
|
return str;
|
|
|
|
} // end of Json_Array_Add
|
|
|
|
|
|
|
|
void Json_Array_Add_deinit(UDF_INIT* initid)
|
|
|
|
{
|
|
|
|
PlugExit((PGLOBAL)initid->ptr);
|
|
|
|
} // end of Json_Array_Add_deinit
|
|
|
|
|
2015-05-05 11:37:21 +02:00
|
|
|
/***********************************************************************/
|
2015-05-17 15:22:42 +02:00
|
|
|
/* Delete a value from a Json array. */
|
2015-05-05 11:37:21 +02:00
|
|
|
/***********************************************************************/
|
|
|
|
my_bool Json_Array_Delete_init(UDF_INIT *initid, UDF_ARGS *args, char *message)
|
|
|
|
{
|
|
|
|
unsigned long reslen, memlen;
|
|
|
|
|
|
|
|
if (args->arg_count != 2) {
|
|
|
|
strcpy(message, "Json_Value_Delete must have 2 arguments");
|
|
|
|
return true;
|
|
|
|
} else if (!IsJson(args, 0)) {
|
|
|
|
strcpy(message, "Json_Value_Delete first argument must be a json item");
|
|
|
|
return true;
|
|
|
|
} else
|
|
|
|
CalcLen(args, false, reslen, memlen);
|
|
|
|
|
|
|
|
return JsonInit(initid, message, reslen, memlen);
|
|
|
|
} // end of Json_Array_Delete_init
|
|
|
|
|
|
|
|
char *Json_Array_Delete(UDF_INIT *initid, UDF_ARGS *args, char *result,
|
2015-05-09 17:30:20 +02:00
|
|
|
unsigned long *res_length, char *, char *)
|
2015-05-05 11:37:21 +02:00
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
int n;
|
|
|
|
PJVAL jvp;
|
|
|
|
PJAR arp;
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
jvp = MakeValue(g, args, 0);
|
|
|
|
|
|
|
|
if (jvp->GetValType() != TYPE_JAR) {
|
|
|
|
push_warning(current_thd, Sql_condition::WARN_LEVEL_WARN, 0,
|
|
|
|
"First argument is not an array");
|
|
|
|
str = args->args[0];
|
|
|
|
} else if (args->arg_type[1] != INT_RESULT) {
|
|
|
|
push_warning(current_thd, Sql_condition::WARN_LEVEL_WARN, 0,
|
|
|
|
"Second argument is not an integer");
|
|
|
|
str = args->args[0];
|
|
|
|
} else {
|
|
|
|
n = *(int*)args->args[1];
|
|
|
|
arp = jvp->GetArray();
|
2015-05-17 15:22:42 +02:00
|
|
|
arp->DeleteValue(n);
|
2015-05-05 11:37:21 +02:00
|
|
|
arp->InitArray(g);
|
|
|
|
|
|
|
|
if (!(str = Serialize(g, arp, NULL, 0))) {
|
|
|
|
str = strcpy(result, g->Message);
|
|
|
|
push_warning(current_thd, Sql_condition::WARN_LEVEL_WARN, 0, str);
|
|
|
|
} // endif str
|
|
|
|
|
|
|
|
} // endif's
|
|
|
|
|
|
|
|
*res_length = strlen(str);
|
|
|
|
return str;
|
|
|
|
} // end of Json_Array_Delete
|
|
|
|
|
|
|
|
void Json_Array_Delete_deinit(UDF_INIT* initid)
|
|
|
|
{
|
|
|
|
PlugExit((PGLOBAL)initid->ptr);
|
|
|
|
} // end of Json_Array_Delete_deinit
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
/***********************************************************************/
|
|
|
|
/* Make a Json Oject containing all the parameters. */
|
|
|
|
/***********************************************************************/
|
|
|
|
my_bool Json_Object_init(UDF_INIT *initid, UDF_ARGS *args, char *message)
|
|
|
|
{
|
2015-02-19 00:59:02 +01:00
|
|
|
unsigned long reslen, memlen;
|
|
|
|
|
|
|
|
CalcLen(args, true, reslen, memlen);
|
|
|
|
return JsonInit(initid, message, reslen, memlen);
|
2015-02-11 21:39:41 +01:00
|
|
|
} // end of Json_Object_init
|
|
|
|
|
|
|
|
char *Json_Object(UDF_INIT *initid, UDF_ARGS *args, char *result,
|
2015-05-09 17:30:20 +02:00
|
|
|
unsigned long *res_length, char *, char *)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
uint i;
|
|
|
|
PJOB objp;
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
objp = new(g) JOBJECT;
|
|
|
|
|
|
|
|
for (i = 0; i < args->arg_count; i++)
|
|
|
|
objp->SetValue(g, MakeValue(g, args, i), MakeKey(g, args, i));
|
|
|
|
|
|
|
|
if (!(str = Serialize(g, objp, NULL, 0)))
|
|
|
|
str = strcpy(result, g->Message);
|
|
|
|
|
|
|
|
*res_length = strlen(str);
|
|
|
|
return str;
|
|
|
|
} // end of Json_Object
|
|
|
|
|
|
|
|
void Json_Object_deinit(UDF_INIT* initid)
|
|
|
|
{
|
|
|
|
PlugExit((PGLOBAL)initid->ptr);
|
|
|
|
} // end of Json_Object_deinit
|
|
|
|
|
|
|
|
/***********************************************************************/
|
2015-02-24 23:18:04 +01:00
|
|
|
/* Make a Json Oject containing all not null parameters. */
|
|
|
|
/***********************************************************************/
|
|
|
|
my_bool Json_Object_Nonull_init(UDF_INIT *initid, UDF_ARGS *args,
|
|
|
|
char *message)
|
|
|
|
{
|
|
|
|
unsigned long reslen, memlen;
|
|
|
|
|
|
|
|
CalcLen(args, true, reslen, memlen);
|
|
|
|
return JsonInit(initid, message, reslen, memlen);
|
|
|
|
} // end of Json_Object_Nonull_init
|
|
|
|
|
|
|
|
char *Json_Object_Nonull(UDF_INIT *initid, UDF_ARGS *args, char *result,
|
2015-05-09 17:30:20 +02:00
|
|
|
unsigned long *res_length, char *, char *)
|
2015-02-24 23:18:04 +01:00
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
uint i;
|
|
|
|
PJOB objp;
|
|
|
|
PJVAL jvp;
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
objp = new(g) JOBJECT;
|
|
|
|
|
|
|
|
for (i = 0; i < args->arg_count; i++)
|
|
|
|
if (!(jvp = MakeValue(g, args, i))->IsNull())
|
|
|
|
objp->SetValue(g, jvp, MakeKey(g, args, i));
|
|
|
|
|
|
|
|
if (!(str = Serialize(g, objp, NULL, 0)))
|
|
|
|
str = strcpy(result, g->Message);
|
|
|
|
|
|
|
|
*res_length = strlen(str);
|
|
|
|
return str;
|
|
|
|
} // end of Json_Object_Nonull
|
|
|
|
|
|
|
|
void Json_Object_Nonull_deinit(UDF_INIT* initid)
|
|
|
|
{
|
|
|
|
PlugExit((PGLOBAL)initid->ptr);
|
|
|
|
} // end of Json_Object_nonull_deinit
|
|
|
|
|
|
|
|
/***********************************************************************/
|
2015-02-11 21:39:41 +01:00
|
|
|
/* Make a Json array from values comming from rows. */
|
|
|
|
/***********************************************************************/
|
|
|
|
my_bool Json_Array_Grp_init(UDF_INIT *initid, UDF_ARGS *args, char *message)
|
|
|
|
{
|
2015-02-22 17:53:02 +01:00
|
|
|
unsigned long reslen, memlen, n = GetJsonGrpSize();
|
2015-02-19 00:59:02 +01:00
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
if (args->arg_count != 1) {
|
|
|
|
strcpy(message, "Json_Array_Grp can only accept 1 argument");
|
|
|
|
return true;
|
2015-02-19 00:59:02 +01:00
|
|
|
} else
|
|
|
|
CalcLen(args, false, reslen, memlen);
|
|
|
|
|
|
|
|
reslen *= n;
|
2015-02-22 17:53:02 +01:00
|
|
|
memlen += ((memlen - MEMFIX) * (n - 1));
|
2015-02-19 00:59:02 +01:00
|
|
|
|
|
|
|
if (JsonInit(initid, message, reslen, memlen))
|
2015-02-11 21:39:41 +01:00
|
|
|
return true;
|
|
|
|
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
g->Activityp = (PACTIVITY)new(g) JARRAY;
|
2015-02-22 17:53:02 +01:00
|
|
|
g->N = (int)n;
|
2015-02-11 21:39:41 +01:00
|
|
|
return false;
|
|
|
|
} // end of Json_Array_Grp_init
|
|
|
|
|
2015-05-09 17:30:20 +02:00
|
|
|
void Json_Array_Grp_add(UDF_INIT *initid, UDF_ARGS *args, char*, char*)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
PJAR arp = (PJAR)g->Activityp;
|
|
|
|
|
2015-02-22 17:53:02 +01:00
|
|
|
if (g->N-- > 0)
|
|
|
|
arp->AddValue(g, MakeValue(g, args, 0));
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
} // end of Json_Array_Grp_add
|
|
|
|
|
2015-05-09 17:30:20 +02:00
|
|
|
char *Json_Array_Grp(UDF_INIT *initid, UDF_ARGS *, char *result,
|
|
|
|
unsigned long *res_length, char *, char *)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
PJAR arp = (PJAR)g->Activityp;
|
|
|
|
|
2015-02-22 17:53:02 +01:00
|
|
|
if (g->N < 0)
|
|
|
|
push_warning(current_thd, Sql_condition::WARN_LEVEL_WARN, 0,
|
|
|
|
"Result truncated to json_grp_size values");
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
arp->InitArray(g);
|
|
|
|
|
|
|
|
if (!(str = Serialize(g, arp, NULL, 0)))
|
|
|
|
str = strcpy(result, g->Message);
|
|
|
|
|
|
|
|
*res_length = strlen(str);
|
|
|
|
return str;
|
|
|
|
} // end of Json_Array_Grp
|
|
|
|
|
2015-05-09 17:30:20 +02:00
|
|
|
void Json_Array_Grp_clear(UDF_INIT *initid, char*, char*)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
g->Activityp = (PACTIVITY)new(g) JARRAY;
|
2015-02-22 17:53:02 +01:00
|
|
|
g->N = GetJsonGrpSize();
|
2015-02-11 21:39:41 +01:00
|
|
|
} // end of Json_Array_Grp_clear
|
|
|
|
|
|
|
|
void Json_Array_Grp_deinit(UDF_INIT* initid)
|
|
|
|
{
|
|
|
|
PlugExit((PGLOBAL)initid->ptr);
|
|
|
|
} // end of Json_Array_Grp_deinit
|
|
|
|
|
|
|
|
/***********************************************************************/
|
|
|
|
/* Make a Json object from values comming from rows. */
|
|
|
|
/***********************************************************************/
|
|
|
|
my_bool Json_Object_Grp_init(UDF_INIT *initid, UDF_ARGS *args, char *message)
|
|
|
|
{
|
2015-02-22 17:53:02 +01:00
|
|
|
unsigned long reslen, memlen, n = GetJsonGrpSize();
|
2015-02-19 00:59:02 +01:00
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
if (args->arg_count != 2) {
|
2015-03-01 19:20:40 +01:00
|
|
|
strcpy(message, "Json_Array_Grp can only accept 2 arguments");
|
2015-02-11 21:39:41 +01:00
|
|
|
return true;
|
2015-02-19 00:59:02 +01:00
|
|
|
} else
|
|
|
|
CalcLen(args, true, reslen, memlen);
|
|
|
|
|
|
|
|
reslen *= n;
|
2015-02-22 17:53:02 +01:00
|
|
|
memlen += ((memlen - MEMFIX) * (n - 1));
|
2015-02-19 00:59:02 +01:00
|
|
|
|
|
|
|
if (JsonInit(initid, message, reslen, memlen))
|
2015-02-11 21:39:41 +01:00
|
|
|
return true;
|
|
|
|
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
g->Activityp = (PACTIVITY)new(g) JOBJECT;
|
2015-02-22 17:53:02 +01:00
|
|
|
g->N = (int)n;
|
2015-02-11 21:39:41 +01:00
|
|
|
return false;
|
|
|
|
} // end of Json_Object_Grp_init
|
|
|
|
|
2015-05-09 17:30:20 +02:00
|
|
|
void Json_Object_Grp_add(UDF_INIT *initid, UDF_ARGS *args, char*, char*)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
PJOB objp = (PJOB)g->Activityp;
|
|
|
|
|
2015-02-22 17:53:02 +01:00
|
|
|
if (g->N-- > 0)
|
|
|
|
objp->SetValue(g, MakeValue(g, args, 0), MakePSZ(g, args, 1));
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
} // end of Json_Object_Grp_add
|
|
|
|
|
2015-05-09 17:30:20 +02:00
|
|
|
char *Json_Object_Grp(UDF_INIT *initid, UDF_ARGS *, char *result,
|
|
|
|
unsigned long *res_length, char *, char *)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
char *str;
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
PJOB objp = (PJOB)g->Activityp;
|
|
|
|
|
2015-02-22 17:53:02 +01:00
|
|
|
if (g->N < 0)
|
|
|
|
push_warning(current_thd, Sql_condition::WARN_LEVEL_WARN, 0,
|
|
|
|
"Result truncated to json_grp_size values");
|
|
|
|
|
2015-02-11 21:39:41 +01:00
|
|
|
if (!(str = Serialize(g, objp, NULL, 0)))
|
|
|
|
str = strcpy(result, g->Message);
|
|
|
|
|
|
|
|
*res_length = strlen(str);
|
|
|
|
return str;
|
|
|
|
} // end of Json_Object_Grp
|
|
|
|
|
2015-05-09 17:30:20 +02:00
|
|
|
void Json_Object_Grp_clear(UDF_INIT *initid, char*, char*)
|
2015-02-11 21:39:41 +01:00
|
|
|
{
|
|
|
|
PGLOBAL g = (PGLOBAL)initid->ptr;
|
|
|
|
|
|
|
|
PlugSubSet(g, g->Sarea, g->Sarea_Size);
|
|
|
|
g->Activityp = (PACTIVITY)new(g) JOBJECT;
|
2015-02-22 17:53:02 +01:00
|
|
|
g->N = GetJsonGrpSize();
|
2015-02-11 21:39:41 +01:00
|
|
|
} // end of Json_Object_Grp_clear
|
|
|
|
|
|
|
|
void Json_Object_Grp_deinit(UDF_INIT* initid)
|
|
|
|
{
|
|
|
|
PlugExit((PGLOBAL)initid->ptr);
|
|
|
|
} // end of Json_Object_Grp_deinit
|
|
|
|
|
|
|
|
|