2022-07-28 10:47:33 +03:00
|
|
|
/* Copyright (c) 2016, 2022, MariaDB Corporation.
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
This program is free software; you can redistribute it and/or modify
|
|
|
|
it under the terms of the GNU General Public License as published by
|
|
|
|
the Free Software Foundation; version 2 of the License.
|
|
|
|
|
|
|
|
This program is distributed in the hope that it will be useful,
|
|
|
|
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
|
|
|
GNU General Public License for more details.
|
|
|
|
|
|
|
|
You should have received a copy of the GNU General Public License
|
|
|
|
along with this program; if not, write to the Free Software
|
|
|
|
Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA */
|
|
|
|
|
|
|
|
|
2017-06-18 06:42:16 +03:00
|
|
|
#include "mariadb.h"
|
2016-10-19 14:10:03 +04:00
|
|
|
#include "sql_priv.h"
|
|
|
|
#include "sql_class.h"
|
|
|
|
#include "item.h"
|
2022-06-21 14:58:34 +05:30
|
|
|
#include "sql_parse.h" // For check_stack_overrun
|
2022-10-28 13:03:13 +05:30
|
|
|
#include "json_schema_helper.h"
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2023-06-17 18:37:22 +02:00
|
|
|
static bool get_current_value(json_engine_t *, const uchar *&, size_t &);
|
|
|
|
static int check_overlaps(json_engine_t *, json_engine_t *, bool);
|
|
|
|
static int json_find_overlap_with_object(json_engine_t *, json_engine_t *, bool);
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
/*
|
|
|
|
Compare ASCII string against the string with the specified
|
|
|
|
character set.
|
2021-03-11 09:50:29 +08:00
|
|
|
Only compares the equality, case insensitive.
|
2016-10-19 14:10:03 +04:00
|
|
|
*/
|
|
|
|
static bool eq_ascii_string(const CHARSET_INFO *cs,
|
|
|
|
const char *ascii,
|
|
|
|
const char *s, uint32 s_len)
|
|
|
|
{
|
|
|
|
const char *s_end= s + s_len;
|
|
|
|
|
|
|
|
while (*ascii && s < s_end)
|
|
|
|
{
|
|
|
|
my_wc_t wc;
|
|
|
|
int wc_len;
|
|
|
|
|
2020-01-26 20:27:13 +04:00
|
|
|
wc_len= cs->mb_wc(&wc, (uchar *) s, (uchar *) s_end);
|
2016-10-19 14:10:03 +04:00
|
|
|
if (wc_len <= 0 || (wc | 0x20) != (my_wc_t) *ascii)
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
ascii++;
|
|
|
|
s+= wc_len;
|
|
|
|
}
|
|
|
|
|
|
|
|
return *ascii == 0 && s >= s_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-28 10:38:02 +00:00
|
|
|
static bool append_simple(String *s, const char *a, size_t a_len)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
if (!s->realloc_with_extra_if_needed(s->length() + a_len))
|
|
|
|
{
|
|
|
|
s->q_append(a, a_len);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-09-28 10:38:02 +00:00
|
|
|
static inline bool append_simple(String *s, const uchar *a, size_t a_len)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
return append_simple(s, (const char *) a, a_len);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Appends JSON string to the String object taking charsets in
|
|
|
|
consideration.
|
2017-11-16 19:59:27 +04:00
|
|
|
*/
|
2016-11-15 17:04:31 +04:00
|
|
|
static int st_append_json(String *s,
|
|
|
|
CHARSET_INFO *json_cs, const uchar *js, uint js_len)
|
|
|
|
{
|
|
|
|
int str_len= js_len * s->charset()->mbmaxlen;
|
|
|
|
|
|
|
|
if (!s->reserve(str_len, 1024) &&
|
|
|
|
(str_len= json_unescape(json_cs, js, js + js_len,
|
|
|
|
s->charset(), (uchar *) s->end(), (uchar *) s->end() + str_len)) > 0)
|
|
|
|
{
|
|
|
|
s->length(s->length() + str_len);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2017-11-16 19:59:27 +04:00
|
|
|
return str_len;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
/*
|
|
|
|
Appends arbitrary String to the JSON string taking charsets in
|
|
|
|
consideration.
|
|
|
|
*/
|
|
|
|
static int st_append_escaped(String *s, const String *a)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
In the worst case one character from the 'a' string
|
|
|
|
turns into '\uXXXX\uXXXX' which is 12.
|
|
|
|
*/
|
|
|
|
int str_len= a->length() * 12 * s->charset()->mbmaxlen /
|
|
|
|
a->charset()->mbminlen;
|
|
|
|
if (!s->reserve(str_len, 1024) &&
|
|
|
|
(str_len=
|
|
|
|
json_escape(a->charset(), (uchar *) a->ptr(), (uchar *)a->end(),
|
|
|
|
s->charset(),
|
|
|
|
(uchar *) s->end(), (uchar *)s->end() + str_len)) > 0)
|
|
|
|
{
|
|
|
|
s->length(s->length() + str_len);
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return a->length();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-14 17:51:03 +04:00
|
|
|
static const int TAB_SIZE_LIMIT= 8;
|
|
|
|
static const char tab_arr[TAB_SIZE_LIMIT+1]= " ";
|
|
|
|
|
|
|
|
static int append_tab(String *js, int depth, int tab_size)
|
|
|
|
{
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (js->append('\n'))
|
2017-02-14 17:51:03 +04:00
|
|
|
return 1;
|
|
|
|
for (int i=0; i<depth; i++)
|
|
|
|
{
|
|
|
|
if (js->append(tab_arr, tab_size))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
2022-06-21 14:58:34 +05:30
|
|
|
int json_path_parts_compare(
|
|
|
|
const json_path_step_t *a, const json_path_step_t *a_end,
|
|
|
|
const json_path_step_t *b, const json_path_step_t *b_end,
|
2022-07-28 10:47:33 +03:00
|
|
|
enum json_value_types vt, const int *array_sizes)
|
2022-06-21 14:58:34 +05:30
|
|
|
{
|
|
|
|
int res, res2;
|
2022-07-28 10:47:33 +03:00
|
|
|
const json_path_step_t *temp_b= b;
|
2022-07-25 14:34:44 +05:30
|
|
|
|
2022-06-21 14:58:34 +05:30
|
|
|
DBUG_EXECUTE_IF("json_check_min_stack_requirement",
|
2022-07-29 15:41:43 +05:30
|
|
|
{
|
|
|
|
long arbitrary_var;
|
|
|
|
long stack_used_up= (available_stack_size(current_thd->thread_stack, &arbitrary_var));
|
|
|
|
ALLOCATE_MEM_ON_STACK(my_thread_stack_size-stack_used_up-STACK_MIN_SIZE);
|
|
|
|
});
|
2022-07-23 19:56:08 +05:30
|
|
|
if (check_stack_overrun(current_thd, STACK_MIN_SIZE , NULL))
|
2022-06-21 14:58:34 +05:30
|
|
|
return 1;
|
2022-07-25 14:34:44 +05:30
|
|
|
|
2022-06-21 14:58:34 +05:30
|
|
|
while (a <= a_end)
|
|
|
|
{
|
|
|
|
if (b > b_end)
|
|
|
|
{
|
|
|
|
while (vt != JSON_VALUE_ARRAY &&
|
|
|
|
(a->type & JSON_PATH_ARRAY_WILD) == JSON_PATH_ARRAY &&
|
|
|
|
a->n_item == 0)
|
|
|
|
{
|
|
|
|
if (++a > a_end)
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
return -2;
|
|
|
|
}
|
|
|
|
|
|
|
|
DBUG_ASSERT((b->type & (JSON_PATH_WILD | JSON_PATH_DOUBLE_WILD)) == 0);
|
|
|
|
|
|
|
|
if (a->type & JSON_PATH_ARRAY)
|
|
|
|
{
|
|
|
|
if (b->type & JSON_PATH_ARRAY)
|
|
|
|
{
|
2022-07-28 10:47:33 +03:00
|
|
|
int res= 0, corrected_n_item_a= 0;
|
|
|
|
if (array_sizes)
|
|
|
|
corrected_n_item_a= a->n_item < 0 ?
|
|
|
|
array_sizes[b-temp_b] + a->n_item : a->n_item;
|
|
|
|
if (a->type & JSON_PATH_ARRAY_RANGE)
|
|
|
|
{
|
|
|
|
int corrected_n_item_end_a= 0;
|
|
|
|
if (array_sizes)
|
|
|
|
corrected_n_item_end_a= a->n_item_end < 0 ?
|
|
|
|
array_sizes[b-temp_b] + a->n_item_end :
|
|
|
|
a->n_item_end;
|
|
|
|
res= b->n_item >= corrected_n_item_a &&
|
|
|
|
b->n_item <= corrected_n_item_end_a;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
res= corrected_n_item_a == b->n_item;
|
|
|
|
|
|
|
|
if ((a->type & JSON_PATH_WILD) || res)
|
2022-06-21 14:58:34 +05:30
|
|
|
goto step_fits;
|
|
|
|
goto step_failed;
|
|
|
|
}
|
|
|
|
if ((a->type & JSON_PATH_WILD) == 0 && a->n_item == 0)
|
|
|
|
goto step_fits_autowrap;
|
|
|
|
goto step_failed;
|
|
|
|
}
|
|
|
|
else /* JSON_PATH_KEY */
|
|
|
|
{
|
|
|
|
if (!(b->type & JSON_PATH_KEY))
|
|
|
|
goto step_failed;
|
|
|
|
|
|
|
|
if (!(a->type & JSON_PATH_WILD) &&
|
|
|
|
(a->key_end - a->key != b->key_end - b->key ||
|
|
|
|
memcmp(a->key, b->key, a->key_end - a->key) != 0))
|
|
|
|
goto step_failed;
|
|
|
|
|
|
|
|
goto step_fits;
|
|
|
|
}
|
|
|
|
step_failed:
|
|
|
|
if (!(a->type & JSON_PATH_DOUBLE_WILD))
|
|
|
|
return -1;
|
|
|
|
b++;
|
|
|
|
continue;
|
|
|
|
|
|
|
|
step_fits:
|
|
|
|
b++;
|
|
|
|
if (!(a->type & JSON_PATH_DOUBLE_WILD))
|
|
|
|
{
|
|
|
|
a++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Double wild handling needs recursions. */
|
2022-07-28 10:47:33 +03:00
|
|
|
res= json_path_parts_compare(a+1, a_end, b, b_end, vt,
|
|
|
|
array_sizes ? array_sizes + (b - temp_b) :
|
|
|
|
NULL);
|
2022-06-21 14:58:34 +05:30
|
|
|
if (res == 0)
|
|
|
|
return 0;
|
|
|
|
|
2022-07-28 10:47:33 +03:00
|
|
|
res2= json_path_parts_compare(a, a_end, b, b_end, vt,
|
|
|
|
array_sizes ? array_sizes + (b - temp_b) :
|
|
|
|
NULL);
|
2022-06-21 14:58:34 +05:30
|
|
|
|
|
|
|
return (res2 >= 0) ? res2 : res;
|
|
|
|
|
|
|
|
step_fits_autowrap:
|
|
|
|
if (!(a->type & JSON_PATH_DOUBLE_WILD))
|
|
|
|
{
|
|
|
|
a++;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Double wild handling needs recursions. */
|
2022-07-28 10:47:33 +03:00
|
|
|
res= json_path_parts_compare(a+1, a_end, b+1, b_end, vt,
|
|
|
|
array_sizes ? array_sizes + (b - temp_b) :
|
|
|
|
NULL);
|
2022-06-21 14:58:34 +05:30
|
|
|
if (res == 0)
|
|
|
|
return 0;
|
|
|
|
|
2022-07-28 10:47:33 +03:00
|
|
|
res2= json_path_parts_compare(a, a_end, b+1, b_end, vt,
|
|
|
|
array_sizes ? array_sizes + (b - temp_b) :
|
|
|
|
NULL);
|
2022-06-21 14:58:34 +05:30
|
|
|
|
|
|
|
return (res2 >= 0) ? res2 : res;
|
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
return b <= b_end;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int json_path_compare(const json_path_t *a, const json_path_t *b,
|
2022-07-28 10:47:33 +03:00
|
|
|
enum json_value_types vt, const int *array_size)
|
2022-06-21 14:58:34 +05:30
|
|
|
{
|
|
|
|
return json_path_parts_compare(a->steps+1, a->last_step,
|
2022-07-28 10:47:33 +03:00
|
|
|
b->steps+1, b->last_step, vt, array_size);
|
2022-06-21 14:58:34 +05:30
|
|
|
}
|
|
|
|
|
2017-02-14 17:51:03 +04:00
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
static int json_nice(json_engine_t *je, String *nice_js,
|
2017-02-14 17:51:03 +04:00
|
|
|
Item_func_json_format::formats mode, int tab_size=4)
|
2017-02-06 06:47:48 +04:00
|
|
|
{
|
|
|
|
int depth= 0;
|
2021-07-01 20:02:12 +02:00
|
|
|
static const char *comma= ", ", *colon= "\": ";
|
2017-02-06 06:47:48 +04:00
|
|
|
uint comma_len, colon_len;
|
|
|
|
int first_value= 1;
|
2023-01-04 18:44:03 +00:00
|
|
|
int value_size = 0;
|
|
|
|
int curr_state= -1;
|
|
|
|
int64_t value_len= 0;
|
|
|
|
String curr_str{};
|
2017-02-06 06:47:48 +04:00
|
|
|
|
2021-07-01 20:02:12 +02:00
|
|
|
nice_js->length(0);
|
|
|
|
nice_js->set_charset(je->s.cs);
|
2021-07-02 19:11:42 +02:00
|
|
|
nice_js->alloc(je->s.str_end - je->s.c_str + 32);
|
|
|
|
|
2017-02-14 17:51:03 +04:00
|
|
|
DBUG_ASSERT(mode != Item_func_json_format::DETAILED ||
|
|
|
|
(tab_size >= 0 && tab_size <= TAB_SIZE_LIMIT));
|
2017-02-06 06:47:48 +04:00
|
|
|
|
|
|
|
if (mode == Item_func_json_format::LOOSE)
|
|
|
|
{
|
|
|
|
comma_len= 2;
|
2017-02-14 17:51:03 +04:00
|
|
|
colon_len= 3;
|
|
|
|
}
|
|
|
|
else if (mode == Item_func_json_format::DETAILED)
|
|
|
|
{
|
|
|
|
comma_len= 1;
|
2017-02-06 06:47:48 +04:00
|
|
|
colon_len= 3;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
comma_len= 1;
|
|
|
|
colon_len= 2;
|
|
|
|
}
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
2023-01-04 18:44:03 +00:00
|
|
|
curr_state= je->state;
|
2017-02-06 06:47:48 +04:00
|
|
|
switch (je->state)
|
|
|
|
{
|
|
|
|
case JST_KEY:
|
|
|
|
{
|
|
|
|
const uchar *key_start= je->s.c_str;
|
|
|
|
const uchar *key_end;
|
|
|
|
|
2017-02-09 17:38:53 +04:00
|
|
|
do
|
|
|
|
{
|
2017-02-06 06:47:48 +04:00
|
|
|
key_end= je->s.c_str;
|
2017-02-09 17:38:53 +04:00
|
|
|
} while (json_read_keyname_chr(je) == 0);
|
2017-02-06 06:47:48 +04:00
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je->s.error))
|
2017-02-06 06:47:48 +04:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (!first_value)
|
|
|
|
nice_js->append(comma, comma_len);
|
|
|
|
|
2017-02-14 17:51:03 +04:00
|
|
|
if (mode == Item_func_json_format::DETAILED &&
|
|
|
|
append_tab(nice_js, depth, tab_size))
|
|
|
|
goto error;
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
nice_js->append('"');
|
2017-02-06 06:47:48 +04:00
|
|
|
append_simple(nice_js, key_start, key_end - key_start);
|
|
|
|
nice_js->append(colon, colon_len);
|
|
|
|
}
|
|
|
|
/* now we have key value to handle, so no 'break'. */
|
|
|
|
DBUG_ASSERT(je->state == JST_VALUE);
|
|
|
|
goto handle_value;
|
|
|
|
|
|
|
|
case JST_VALUE:
|
|
|
|
if (!first_value)
|
|
|
|
nice_js->append(comma, comma_len);
|
|
|
|
|
2017-02-14 17:51:03 +04:00
|
|
|
if (mode == Item_func_json_format::DETAILED &&
|
|
|
|
depth > 0 &&
|
|
|
|
append_tab(nice_js, depth, tab_size))
|
|
|
|
goto error;
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
handle_value:
|
|
|
|
if (json_read_value(je))
|
|
|
|
goto error;
|
|
|
|
if (json_value_scalar(je))
|
|
|
|
{
|
|
|
|
if (append_simple(nice_js, je->value_begin,
|
|
|
|
je->value_end - je->value_begin))
|
|
|
|
goto error;
|
2023-01-04 18:44:03 +00:00
|
|
|
|
|
|
|
curr_str.copy((const char *)je->value_begin,
|
|
|
|
je->value_end - je->value_begin, je->s.cs);
|
|
|
|
value_len= je->value_end - je->value_begin;
|
2017-02-06 06:47:48 +04:00
|
|
|
first_value= 0;
|
2023-01-04 18:44:03 +00:00
|
|
|
if (value_size != -1)
|
|
|
|
value_size++;
|
2017-02-06 06:47:48 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2017-02-14 17:51:03 +04:00
|
|
|
if (mode == Item_func_json_format::DETAILED &&
|
2023-01-04 18:44:03 +00:00
|
|
|
depth > 0 && !(curr_state != JST_KEY) &&
|
2017-02-14 17:51:03 +04:00
|
|
|
append_tab(nice_js, depth, tab_size))
|
|
|
|
goto error;
|
2017-02-06 06:47:48 +04:00
|
|
|
nice_js->append((je->value_type == JSON_VALUE_OBJECT) ? "{" : "[", 1);
|
|
|
|
first_value= 1;
|
2023-01-04 18:44:03 +00:00
|
|
|
value_size= (je->value_type == JSON_VALUE_OBJECT) ? -1: 0;
|
2017-02-06 06:47:48 +04:00
|
|
|
depth++;
|
|
|
|
}
|
|
|
|
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JST_OBJ_END:
|
|
|
|
case JST_ARRAY_END:
|
|
|
|
depth--;
|
2023-01-04 18:44:03 +00:00
|
|
|
if (mode == Item_func_json_format::DETAILED && (value_size > 1 || value_size == -1) &&
|
2017-02-14 17:51:03 +04:00
|
|
|
append_tab(nice_js, depth, tab_size))
|
|
|
|
goto error;
|
2023-01-04 18:44:03 +00:00
|
|
|
|
|
|
|
if (mode == Item_func_json_format::DETAILED &&
|
|
|
|
value_size == 1 && je->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
for (auto i = 0; i < value_len; i++)
|
|
|
|
{
|
|
|
|
nice_js->chop();
|
|
|
|
}
|
|
|
|
for (auto i = 0; i < (depth + 1) * tab_size + 1; i++)
|
|
|
|
{
|
|
|
|
nice_js->chop();
|
|
|
|
}
|
|
|
|
nice_js->append(curr_str);
|
|
|
|
}
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
nice_js->append((je->state == JST_OBJ_END) ? "}": "]", 1);
|
|
|
|
first_value= 0;
|
2023-01-04 18:44:03 +00:00
|
|
|
value_size= -1;
|
2017-02-06 06:47:48 +04:00
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
};
|
|
|
|
} while (json_scan_next(je) == 0);
|
|
|
|
|
2021-07-02 19:13:26 +02:00
|
|
|
return je->s.error || *je->killed_ptr;
|
2017-02-06 06:47:48 +04:00
|
|
|
|
|
|
|
error:
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
#define report_json_error(js, je, n_param) \
|
2021-03-17 09:03:45 +04:00
|
|
|
report_json_error_ex(js->ptr(), je, func_name(), n_param, \
|
2016-12-13 12:39:48 +04:00
|
|
|
Sql_condition::WARN_LEVEL_WARN)
|
|
|
|
|
2021-03-17 09:03:45 +04:00
|
|
|
void report_json_error_ex(const char *js, json_engine_t *je,
|
2017-01-24 02:29:04 +04:00
|
|
|
const char *fname, int n_param,
|
|
|
|
Sql_condition::enum_warning_level lv)
|
2016-12-13 12:39:48 +04:00
|
|
|
{
|
|
|
|
THD *thd= current_thd;
|
2021-03-17 09:03:45 +04:00
|
|
|
int position= (int)((const char *) je->s.c_str - js);
|
2016-12-13 12:39:48 +04:00
|
|
|
uint code;
|
|
|
|
|
|
|
|
n_param++;
|
|
|
|
|
|
|
|
switch (je->s.error)
|
|
|
|
{
|
|
|
|
case JE_BAD_CHR:
|
|
|
|
code= ER_JSON_BAD_CHR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JE_NOT_JSON_CHR:
|
|
|
|
code= ER_JSON_NOT_JSON_CHR;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JE_EOS:
|
|
|
|
code= ER_JSON_EOS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JE_SYN:
|
|
|
|
case JE_STRING_CONST:
|
|
|
|
code= ER_JSON_SYNTAX;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JE_ESCAPING:
|
|
|
|
code= ER_JSON_ESCAPING;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JE_DEPTH:
|
|
|
|
code= ER_JSON_DEPTH;
|
2021-03-17 09:03:45 +04:00
|
|
|
if (lv == Sql_condition::WARN_LEVEL_ERROR)
|
|
|
|
my_error(code, MYF(0), JSON_DEPTH_LIMIT, n_param, fname, position);
|
|
|
|
else
|
|
|
|
push_warning_printf(thd, lv, code, ER_THD(thd, code), JSON_DEPTH_LIMIT,
|
|
|
|
n_param, fname, position);
|
2016-12-13 12:39:48 +04:00
|
|
|
return;
|
|
|
|
|
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2021-03-17 09:03:45 +04:00
|
|
|
if (lv == Sql_condition::WARN_LEVEL_ERROR)
|
|
|
|
my_error(code, MYF(0), n_param, fname, position);
|
|
|
|
else
|
|
|
|
push_warning_printf(thd, lv, code, ER_THD(thd, code),
|
|
|
|
n_param, fname, position);
|
2016-12-13 12:39:48 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
#define NO_WILDCARD_ALLOWED 1
|
|
|
|
#define SHOULD_END_WITH_ARRAY 2
|
2017-01-24 17:34:44 +04:00
|
|
|
#define TRIVIAL_PATH_NOT_ALLOWED 3
|
2016-12-13 12:39:48 +04:00
|
|
|
|
|
|
|
#define report_path_error(js, je, n_param) \
|
2021-03-17 09:03:45 +04:00
|
|
|
report_path_error_ex(js->ptr(), je, func_name(), n_param,\
|
2016-12-13 12:39:48 +04:00
|
|
|
Sql_condition::WARN_LEVEL_WARN)
|
|
|
|
|
2021-03-17 09:03:45 +04:00
|
|
|
void report_path_error_ex(const char *ps, json_path_t *p,
|
|
|
|
const char *fname, int n_param,
|
|
|
|
Sql_condition::enum_warning_level lv)
|
2016-12-13 12:39:48 +04:00
|
|
|
{
|
|
|
|
THD *thd= current_thd;
|
2021-03-17 09:03:45 +04:00
|
|
|
int position= (int)((const char *) p->s.c_str - ps + 1);
|
2016-12-13 12:39:48 +04:00
|
|
|
uint code;
|
|
|
|
|
|
|
|
n_param++;
|
|
|
|
|
|
|
|
switch (p->s.error)
|
|
|
|
{
|
|
|
|
case JE_BAD_CHR:
|
|
|
|
case JE_NOT_JSON_CHR:
|
|
|
|
case JE_SYN:
|
|
|
|
code= ER_JSON_PATH_SYNTAX;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JE_EOS:
|
|
|
|
code= ER_JSON_PATH_EOS;
|
|
|
|
break;
|
|
|
|
|
|
|
|
case JE_DEPTH:
|
|
|
|
code= ER_JSON_PATH_DEPTH;
|
2021-03-17 09:03:45 +04:00
|
|
|
if (lv == Sql_condition::WARN_LEVEL_ERROR)
|
|
|
|
my_error(code, MYF(0), JSON_DEPTH_LIMIT, n_param, fname, position);
|
|
|
|
else
|
|
|
|
push_warning_printf(thd, lv, code, ER_THD(thd, code),
|
|
|
|
JSON_DEPTH_LIMIT, n_param, fname, position);
|
2016-12-13 12:39:48 +04:00
|
|
|
return;
|
|
|
|
|
|
|
|
case NO_WILDCARD_ALLOWED:
|
|
|
|
code= ER_JSON_PATH_NO_WILDCARD;
|
|
|
|
break;
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
case TRIVIAL_PATH_NOT_ALLOWED:
|
|
|
|
code= ER_JSON_PATH_EMPTY;
|
|
|
|
break;
|
|
|
|
|
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
default:
|
|
|
|
return;
|
|
|
|
}
|
2021-03-17 09:03:45 +04:00
|
|
|
if (lv == Sql_condition::WARN_LEVEL_ERROR)
|
|
|
|
my_error(code, MYF(0), n_param, fname, position);
|
|
|
|
else
|
|
|
|
push_warning_printf(thd, lv, code, ER_THD(thd, code),
|
|
|
|
n_param, fname, position);
|
2016-12-13 12:39:48 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Checks if the path has '.*' '[*]' or '**' constructions
|
|
|
|
and sets the NO_WILDCARD_ALLOWED error if the case.
|
|
|
|
*/
|
|
|
|
static int path_setup_nwc(json_path_t *p, CHARSET_INFO *i_cs,
|
|
|
|
const uchar *str, const uchar *end)
|
|
|
|
{
|
|
|
|
if (!json_path_setup(p, i_cs, str, end))
|
|
|
|
{
|
2022-03-05 01:03:49 +05:30
|
|
|
if ((p->types_used & (JSON_PATH_WILD | JSON_PATH_DOUBLE_WILD |
|
|
|
|
JSON_PATH_ARRAY_RANGE)) == 0)
|
2016-12-13 12:39:48 +04:00
|
|
|
return 0;
|
|
|
|
p->s.error= NO_WILDCARD_ALLOWED;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
longlong Item_func_json_valid::val_int()
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_value);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
if ((null_value= args[0]->null_value))
|
2016-10-19 14:10:03 +04:00
|
|
|
return 0;
|
|
|
|
|
2018-11-28 18:16:46 +01:00
|
|
|
return json_valid(js->ptr(), js->length(), js->charset());
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_equals::fix_length_and_dec(THD *thd)
|
2021-07-02 08:20:37 +02:00
|
|
|
{
|
2022-01-26 03:02:45 +05:30
|
|
|
if (Item_bool_func::fix_length_and_dec(thd))
|
2021-07-02 08:20:37 +02:00
|
|
|
return TRUE;
|
|
|
|
set_maybe_null();
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
longlong Item_func_json_equals::val_int()
|
|
|
|
{
|
|
|
|
longlong result= 0;
|
|
|
|
|
|
|
|
String a_tmp, b_tmp;
|
|
|
|
|
|
|
|
String *a= args[0]->val_json(&a_tmp);
|
|
|
|
String *b= args[1]->val_json(&b_tmp);
|
|
|
|
|
|
|
|
DYNAMIC_STRING a_res;
|
|
|
|
if (init_dynamic_string(&a_res, NULL, 0, 0))
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
DYNAMIC_STRING b_res;
|
|
|
|
if (init_dynamic_string(&b_res, NULL, 0, 0))
|
|
|
|
{
|
|
|
|
dynstr_free(&a_res);
|
|
|
|
null_value= 1;
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value || args[1]->null_value))
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
2021-08-06 11:27:52 +03:00
|
|
|
if (json_normalize(&a_res, a->ptr(), a->length(), a->charset()))
|
2021-07-02 08:20:37 +02:00
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
2021-08-06 11:27:52 +03:00
|
|
|
if (json_normalize(&b_res, b->ptr(), b->length(), b->charset()))
|
2021-07-02 08:20:37 +02:00
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
result= strcmp(a_res.str, b_res.str) ? 0 : 1;
|
|
|
|
|
|
|
|
end:
|
|
|
|
dynstr_free(&b_res);
|
|
|
|
dynstr_free(&a_res);
|
|
|
|
return result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_exists::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2022-01-26 03:02:45 +05:30
|
|
|
if (Item_bool_func::fix_length_and_dec(thd))
|
2018-05-08 15:26:26 +02:00
|
|
|
return TRUE;
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2016-10-19 14:10:03 +04:00
|
|
|
path.set_constant_flag(args[1]->const_item());
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
longlong Item_func_json_exists::val_int()
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
if (!path.parsed)
|
|
|
|
{
|
|
|
|
String *s_p= args[1]->val_str(&tmp_path);
|
|
|
|
if (s_p &&
|
|
|
|
json_path_setup(&path.p, s_p->charset(), (const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
goto err_return;
|
|
|
|
path.parsed= path.constant;
|
|
|
|
}
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value || args[1]->null_value))
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
null_value= 0;
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
|
|
|
path.cur_step= path.p.steps;
|
|
|
|
if (json_find_path(&je, &path.p, &path.cur_step, array_counters))
|
|
|
|
{
|
|
|
|
if (je.s.error)
|
|
|
|
goto err_return;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
err_return:
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_value::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
max_length= args[0]->max_length;
|
2019-06-28 12:35:02 +04:00
|
|
|
set_constant_flag(args[1]->const_item());
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2019-06-28 12:35:02 +04:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_query::fix_length_and_dec(THD *thd)
|
2019-06-28 12:35:02 +04:00
|
|
|
{
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
max_length= args[0]->max_length;
|
|
|
|
set_constant_flag(args[1]->const_item());
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
bool Json_path_extractor::extract(String *str, Item *item_js, Item *item_jp,
|
|
|
|
CHARSET_INFO *cs)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2019-06-28 12:35:02 +04:00
|
|
|
String *js= item_js->val_json(&tmp_js);
|
2016-10-19 14:10:03 +04:00
|
|
|
int error= 0;
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
if (!parsed)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2019-06-28 12:35:02 +04:00
|
|
|
String *s_p= item_jp->val_str(&tmp_path);
|
2016-10-19 14:10:03 +04:00
|
|
|
if (s_p &&
|
2019-06-28 12:35:02 +04:00
|
|
|
json_path_setup(&p, s_p->charset(), (const uchar *) s_p->ptr(),
|
2016-10-19 14:10:03 +04:00
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
2019-06-28 12:35:02 +04:00
|
|
|
return true;
|
|
|
|
parsed= constant;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
if (item_js->null_value || item_jp->null_value)
|
|
|
|
return true;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
Json_engine_scan je(*js);
|
2017-11-16 19:59:27 +04:00
|
|
|
str->length(0);
|
2019-06-28 12:35:02 +04:00
|
|
|
str->set_charset(cs);
|
2017-11-16 19:59:27 +04:00
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
cur_step= p.steps;
|
2016-10-19 14:10:03 +04:00
|
|
|
continue_search:
|
2019-06-28 12:35:02 +04:00
|
|
|
if (json_find_path(&je, &p, &cur_step, array_counters))
|
|
|
|
return true;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
if (json_read_value(&je))
|
2019-06-28 12:35:02 +04:00
|
|
|
return true;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2022-08-01 19:39:09 +05:30
|
|
|
if (je.value_type == JSON_VALUE_NULL)
|
2022-08-22 13:38:40 +03:00
|
|
|
return true;
|
2022-08-01 19:39:09 +05:30
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(check_and_get_value(&je, str, &error)))
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
if (error)
|
2019-06-28 12:35:02 +04:00
|
|
|
return true;
|
2016-10-19 14:10:03 +04:00
|
|
|
goto continue_search;
|
|
|
|
}
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
return false;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
bool Json_engine_scan::check_and_get_value_scalar(String *res, int *error)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2018-06-17 17:15:21 +04:00
|
|
|
CHARSET_INFO *json_cs;
|
|
|
|
const uchar *js;
|
|
|
|
uint js_len;
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
if (!json_value_scalar(this))
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
/* We only look for scalar values! */
|
2019-06-28 12:35:02 +04:00
|
|
|
if (json_skip_level(this) || json_scan_next(this))
|
2016-10-19 14:10:03 +04:00
|
|
|
*error= 1;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
if (value_type == JSON_VALUE_TRUE ||
|
|
|
|
value_type == JSON_VALUE_FALSE)
|
2018-06-17 17:15:21 +04:00
|
|
|
{
|
|
|
|
json_cs= &my_charset_utf8mb4_bin;
|
2019-06-28 12:35:02 +04:00
|
|
|
js= (const uchar *) ((value_type == JSON_VALUE_TRUE) ? "1" : "0");
|
2018-06-17 17:15:21 +04:00
|
|
|
js_len= 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2019-06-28 12:35:02 +04:00
|
|
|
json_cs= s.cs;
|
|
|
|
js= value;
|
|
|
|
js_len= value_len;
|
2018-06-17 17:15:21 +04:00
|
|
|
}
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
return st_append_json(res, json_cs, js, js_len);
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-05-25 15:45:43 +05:30
|
|
|
bool Json_engine_scan::check_and_get_value_complex(String *res, int *error,
|
|
|
|
json_value_types
|
|
|
|
cur_value_type)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2019-06-28 12:35:02 +04:00
|
|
|
if (json_value_scalar(this))
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
/* We skip scalar values. */
|
2019-06-28 12:35:02 +04:00
|
|
|
if (json_scan_next(this))
|
2016-10-19 14:10:03 +04:00
|
|
|
*error= 1;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
const uchar *tmp_value= value;
|
|
|
|
if (json_skip_level(this))
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
*error= 1;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2023-05-25 15:45:43 +05:30
|
|
|
if (cur_value_type != JSON_VALUE_UNINITIALIZED &&
|
|
|
|
value_type != cur_value_type)
|
|
|
|
{
|
|
|
|
*error= 1;
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
2019-06-28 12:35:02 +04:00
|
|
|
res->set((const char *) value, (uint32)(s.c_str - tmp_value), s.cs);
|
2016-10-19 14:10:03 +04:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_quote::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2016-12-04 14:49:06 +04:00
|
|
|
collation.set(&my_charset_utf8mb4_bin);
|
2016-10-19 14:10:03 +04:00
|
|
|
/*
|
|
|
|
Odd but realistic worst case is when all characters
|
|
|
|
of the argument turn into '\uXXXX\uXXXX', which is 12.
|
|
|
|
*/
|
2018-07-24 18:15:15 +04:00
|
|
|
fix_char_length_ulonglong((ulonglong) args[0]->max_char_length() * 12 + 2);
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_quote::val_str(String *str)
|
|
|
|
{
|
|
|
|
String *s= args[0]->val_str(&tmp_s);
|
|
|
|
|
2016-12-04 14:49:06 +04:00
|
|
|
if ((null_value= (args[0]->null_value ||
|
|
|
|
args[0]->result_type() != STRING_RESULT)))
|
2016-10-19 14:10:03 +04:00
|
|
|
return NULL;
|
|
|
|
|
|
|
|
str->length(0);
|
2016-12-04 14:49:06 +04:00
|
|
|
str->set_charset(&my_charset_utf8mb4_bin);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('"') ||
|
2016-12-04 14:49:06 +04:00
|
|
|
st_append_escaped(str, s) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append('"'))
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
/* Report an error. */
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_unquote::fix_length_and_dec(THD *thd)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2019-06-28 09:05:12 +04:00
|
|
|
collation.set(&my_charset_utf8mb3_general_ci,
|
2017-10-06 09:28:33 +04:00
|
|
|
DERIVATION_COERCIBLE, MY_REPERTOIRE_ASCII);
|
2016-11-15 17:04:31 +04:00
|
|
|
max_length= args[0]->max_length;
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-08 10:35:26 +04:00
|
|
|
String *Item_func_json_unquote::read_json(json_engine_t *je)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_s);
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
2017-08-08 10:35:26 +04:00
|
|
|
return 0;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2017-08-08 10:35:26 +04:00
|
|
|
json_scan_start(je, js->charset(),(const uchar *) js->ptr(),
|
2016-11-15 17:04:31 +04:00
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
2017-08-08 10:35:26 +04:00
|
|
|
if (json_read_value(je))
|
2016-11-15 17:04:31 +04:00
|
|
|
goto error;
|
|
|
|
|
2017-08-08 10:35:26 +04:00
|
|
|
return js;
|
|
|
|
|
|
|
|
error:
|
|
|
|
if (je->value_type == JSON_VALUE_STRING)
|
|
|
|
report_json_error(js, je, 0);
|
|
|
|
return js;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_unquote::val_str(String *str)
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
|
|
|
int c_len;
|
|
|
|
String *js;
|
|
|
|
|
|
|
|
if (!(js= read_json(&je)))
|
|
|
|
return NULL;
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error) || je.value_type != JSON_VALUE_STRING)
|
2016-11-15 17:04:31 +04:00
|
|
|
return js;
|
|
|
|
|
|
|
|
str->length(0);
|
2019-06-28 09:05:12 +04:00
|
|
|
str->set_charset(&my_charset_utf8mb3_general_ci);
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (str->realloc_with_extra_if_needed(je.value_len) ||
|
|
|
|
(c_len= json_unescape(js->charset(),
|
|
|
|
je.value, je.value + je.value_len,
|
2019-06-28 09:05:12 +04:00
|
|
|
&my_charset_utf8mb3_general_ci,
|
2016-11-15 17:04:31 +04:00
|
|
|
(uchar *) str->ptr(), (uchar *) (str->ptr() + je.value_len))) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
str->length(c_len);
|
|
|
|
return str;
|
|
|
|
|
|
|
|
error:
|
2017-08-08 10:35:26 +04:00
|
|
|
report_json_error(js, &je, 0);
|
2016-12-06 00:34:25 +04:00
|
|
|
return js;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
static int alloc_tmp_paths(THD *thd, uint n_paths,
|
2017-08-13 01:08:30 +02:00
|
|
|
json_path_with_flags **paths, String **tmp_paths)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
if (n_paths > 0)
|
|
|
|
{
|
2017-08-13 01:08:30 +02:00
|
|
|
if (*tmp_paths == 0)
|
|
|
|
{
|
|
|
|
MEM_ROOT *root= thd->stmt_arena->mem_root;
|
2017-09-13 00:36:09 +04:00
|
|
|
|
2017-08-13 01:08:30 +02:00
|
|
|
*paths= (json_path_with_flags *) alloc_root(root,
|
|
|
|
sizeof(json_path_with_flags) * n_paths);
|
2019-03-15 15:03:26 +01:00
|
|
|
|
|
|
|
*tmp_paths= new (root) String[n_paths];
|
2017-08-13 01:08:30 +02:00
|
|
|
if (*paths == 0 || *tmp_paths == 0)
|
|
|
|
return 1;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-09-13 00:36:09 +04:00
|
|
|
for (uint c_path=0; c_path < n_paths; c_path++)
|
2019-06-28 09:05:12 +04:00
|
|
|
(*tmp_paths)[c_path].set_charset(&my_charset_utf8mb3_general_ci);
|
2017-08-13 01:08:30 +02:00
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* n_paths == 0 */
|
|
|
|
*paths= 0;
|
|
|
|
*tmp_paths= 0;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static void mark_constant_paths(json_path_with_flags *p,
|
|
|
|
Item** args, uint n_args)
|
|
|
|
{
|
|
|
|
uint n;
|
|
|
|
for (n= 0; n < n_args; n++)
|
|
|
|
p[n].set_constant_flag(args[n]->const_item());
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_json_str_multipath::fix_fields(THD *thd, Item **ref)
|
|
|
|
{
|
|
|
|
return alloc_tmp_paths(thd, get_n_paths(), &paths, &tmp_paths) ||
|
|
|
|
Item_str_func::fix_fields(thd, ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Item_json_str_multipath::cleanup()
|
|
|
|
{
|
|
|
|
if (tmp_paths)
|
|
|
|
{
|
|
|
|
for (uint i= get_n_paths(); i>0; i--)
|
|
|
|
tmp_paths[i-1].free();
|
|
|
|
}
|
|
|
|
Item_str_func::cleanup();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_extract::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
max_length= args[0]->max_length * (arg_count - 1);
|
|
|
|
|
|
|
|
mark_constant_paths(paths, args+1, arg_count-1);
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
static bool path_exact(const json_path_with_flags *paths_list, int n_paths,
|
2021-11-22 22:59:30 +05:30
|
|
|
const json_path_t *p, json_value_types vt,
|
|
|
|
const int *array_size_counter)
|
2017-01-24 17:34:44 +04:00
|
|
|
{
|
|
|
|
for (; n_paths > 0; n_paths--, paths_list++)
|
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
if (json_path_compare(&paths_list->p, p, vt, array_size_counter) == 0)
|
2017-01-24 17:34:44 +04:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static bool path_ok(const json_path_with_flags *paths_list, int n_paths,
|
2021-11-22 22:59:30 +05:30
|
|
|
const json_path_t *p, json_value_types vt,
|
|
|
|
const int *array_size_counter)
|
2017-01-24 17:34:44 +04:00
|
|
|
{
|
|
|
|
for (; n_paths > 0; n_paths--, paths_list++)
|
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
if (json_path_compare(&paths_list->p, p, vt, array_size_counter) >= 0)
|
2017-01-24 17:34:44 +04:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
String *Item_func_json_extract::read_json(String *str,
|
|
|
|
json_value_types *type,
|
|
|
|
char **out_val, int *value_len)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2017-01-24 17:34:44 +04:00
|
|
|
json_engine_t je, sav_je;
|
|
|
|
json_path_t p;
|
2016-10-19 14:10:03 +04:00
|
|
|
const uchar *value;
|
2017-01-24 17:34:44 +04:00
|
|
|
int not_first_value= 0;
|
2017-09-28 10:38:02 +00:00
|
|
|
uint n_arg;
|
|
|
|
size_t v_len;
|
2017-01-24 17:34:44 +04:00
|
|
|
int possible_multiple_values;
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_size_counter[JSON_DEPTH_LIMIT];
|
|
|
|
uint has_negative_path= 0;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (n_arg=1; n_arg < arg_count; n_arg++)
|
|
|
|
{
|
|
|
|
json_path_with_flags *c_path= paths + n_arg - 1;
|
|
|
|
if (!c_path->parsed)
|
|
|
|
{
|
2022-12-29 21:53:12 +05:30
|
|
|
c_path->p.types_used= JSON_PATH_KEY_NULL;
|
2016-10-19 14:10:03 +04:00
|
|
|
String *s_p= args[n_arg]->val_str(tmp_paths + (n_arg-1));
|
2022-04-19 21:43:31 +05:30
|
|
|
if (s_p)
|
2017-01-24 17:34:44 +04:00
|
|
|
{
|
2022-04-19 21:43:31 +05:30
|
|
|
if (json_path_setup(&c_path->p,s_p->charset(),(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
|
|
|
goto return_null;
|
|
|
|
}
|
|
|
|
c_path->parsed= c_path->constant;
|
|
|
|
has_negative_path|= c_path->p.types_used & JSON_PATH_NEGATIVE_INDEX;
|
2017-01-24 17:34:44 +04:00
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
2017-01-25 00:13:15 +04:00
|
|
|
|
|
|
|
if (args[n_arg]->null_value)
|
|
|
|
goto return_null;
|
2017-01-24 17:34:44 +04:00
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
possible_multiple_values= arg_count > 2 ||
|
2022-03-05 01:03:49 +05:30
|
|
|
(paths[0].p.types_used & (JSON_PATH_WILD | JSON_PATH_DOUBLE_WILD |
|
|
|
|
JSON_PATH_ARRAY_RANGE));
|
2016-12-03 11:22:42 +04:00
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
*type= possible_multiple_values ? JSON_VALUE_ARRAY : JSON_VALUE_NULL;
|
|
|
|
|
|
|
|
if (str)
|
|
|
|
{
|
|
|
|
str->set_charset(js->charset());
|
|
|
|
str->length(0);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (possible_multiple_values && str->append('['))
|
2017-12-25 08:10:48 +04:00
|
|
|
goto error;
|
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
json_get_path_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length(), &p);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
while (json_get_path_next(&je, &p) == 0)
|
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
if (has_negative_path && je.value_type == JSON_VALUE_ARRAY &&
|
|
|
|
json_skip_array_and_count(&je,
|
|
|
|
array_size_counter + (p.last_step - p.steps)))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (!path_exact(paths, arg_count-1, &p, je.value_type, array_size_counter))
|
2017-01-24 17:34:44 +04:00
|
|
|
continue;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
value= je.value_begin;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
if (*type == JSON_VALUE_NULL)
|
|
|
|
{
|
|
|
|
*type= je.value_type;
|
|
|
|
*out_val= (char *) je.value;
|
|
|
|
*value_len= je.value_len;
|
|
|
|
}
|
|
|
|
if (!str)
|
|
|
|
{
|
|
|
|
/* If str is NULL, we only care about the first found value. */
|
|
|
|
goto return_ok;
|
|
|
|
}
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
if (json_value_scalar(&je))
|
|
|
|
v_len= je.value_end - value;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (possible_multiple_values)
|
|
|
|
sav_je= je;
|
|
|
|
if (json_skip_level(&je))
|
|
|
|
goto error;
|
|
|
|
v_len= je.s.c_str - value;
|
|
|
|
if (possible_multiple_values)
|
|
|
|
je= sav_je;
|
|
|
|
}
|
2016-12-05 09:34:28 +04:00
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
if ((not_first_value && str->append(", ", 2)) ||
|
|
|
|
str->append((const char *) value, v_len))
|
|
|
|
goto error; /* Out of memory. */
|
2016-12-08 11:25:21 +04:00
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
not_first_value= 1;
|
|
|
|
|
|
|
|
if (!possible_multiple_values)
|
2017-09-12 11:20:30 +04:00
|
|
|
{
|
|
|
|
/* Loop to the end of the JSON just to make sure it's valid. */
|
2022-12-29 22:46:53 +05:30
|
|
|
while (json_scan_next(&je) == 0) {}
|
2017-01-24 17:34:44 +04:00
|
|
|
break;
|
2017-09-12 11:20:30 +04:00
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error))
|
2016-12-08 11:25:21 +04:00
|
|
|
goto error;
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
if (!not_first_value)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
/* Nothing was found. */
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (possible_multiple_values && str->append(']'))
|
2016-10-19 14:10:03 +04:00
|
|
|
goto error; /* Out of memory. */
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
js= str;
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
if (json_nice(&je, &tmp_js, Item_func_json_format::LOOSE))
|
|
|
|
goto error;
|
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
return_ok:
|
2017-02-06 06:47:48 +04:00
|
|
|
return &tmp_js;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
error:
|
2016-12-13 12:39:48 +04:00
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
return_null:
|
2016-10-19 14:10:03 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
String *Item_func_json_extract::val_str(String *str)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2017-08-11 00:50:29 +04:00
|
|
|
json_value_types type;
|
|
|
|
char *value;
|
|
|
|
int value_len;
|
|
|
|
return read_json(str, &type, &value, &value_len);
|
|
|
|
}
|
2016-12-03 11:22:42 +04:00
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
longlong Item_func_json_extract::val_int()
|
|
|
|
{
|
|
|
|
json_value_types type;
|
|
|
|
char *value;
|
|
|
|
int value_len;
|
2017-09-12 15:33:30 +04:00
|
|
|
longlong i= 0;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
if (read_json(NULL, &type, &value, &value_len) != NULL)
|
|
|
|
{
|
|
|
|
switch (type)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2017-08-11 00:50:29 +04:00
|
|
|
case JSON_VALUE_NUMBER:
|
|
|
|
case JSON_VALUE_STRING:
|
|
|
|
{
|
|
|
|
char *end;
|
|
|
|
int err;
|
2020-01-26 20:27:13 +04:00
|
|
|
i= collation.collation->strntoll(value, value_len, 10, &end, &err);
|
2017-08-11 00:50:29 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case JSON_VALUE_TRUE:
|
|
|
|
i= 1;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
i= 0;
|
|
|
|
break;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
return i;
|
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
double Item_func_json_extract::val_real()
|
|
|
|
{
|
|
|
|
json_value_types type;
|
|
|
|
char *value;
|
|
|
|
int value_len;
|
|
|
|
double d= 0.0;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
if (read_json(NULL, &type, &value, &value_len) != NULL)
|
|
|
|
{
|
|
|
|
switch (type)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2017-08-11 00:50:29 +04:00
|
|
|
case JSON_VALUE_STRING:
|
|
|
|
case JSON_VALUE_NUMBER:
|
|
|
|
{
|
|
|
|
char *end;
|
|
|
|
int err;
|
2020-01-26 20:27:13 +04:00
|
|
|
d= collation.collation->strntod(value, value_len, &end, &err);
|
2017-08-11 00:50:29 +04:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
case JSON_VALUE_TRUE:
|
|
|
|
d= 1.0;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
};
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
return d;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-22 15:43:53 +04:00
|
|
|
my_decimal *Item_func_json_extract::val_decimal(my_decimal *to)
|
|
|
|
{
|
|
|
|
json_value_types type;
|
|
|
|
char *value;
|
|
|
|
int value_len;
|
|
|
|
|
|
|
|
if (read_json(NULL, &type, &value, &value_len) != NULL)
|
|
|
|
{
|
|
|
|
switch (type)
|
|
|
|
{
|
|
|
|
case JSON_VALUE_STRING:
|
|
|
|
case JSON_VALUE_NUMBER:
|
|
|
|
{
|
|
|
|
my_decimal *res= decimal_from_string_with_check(to, collation.collation,
|
|
|
|
value,
|
|
|
|
value + value_len);
|
|
|
|
null_value= res == NULL;
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
case JSON_VALUE_TRUE:
|
|
|
|
int2my_decimal(E_DEC_FATAL_ERROR, 1, false/*unsigned_flag*/, to);
|
|
|
|
return to;
|
|
|
|
case JSON_VALUE_OBJECT:
|
|
|
|
case JSON_VALUE_ARRAY:
|
|
|
|
case JSON_VALUE_FALSE:
|
2021-05-23 14:16:01 +00:00
|
|
|
case JSON_VALUE_UNINITIALIZED:
|
2020-06-22 15:43:53 +04:00
|
|
|
case JSON_VALUE_NULL:
|
2022-07-28 16:17:03 +02:00
|
|
|
int2my_decimal(E_DEC_FATAL_ERROR, 0, false/*unsigned_flag*/, to);
|
|
|
|
return to;
|
2020-06-22 15:43:53 +04:00
|
|
|
};
|
|
|
|
}
|
2022-07-28 16:17:03 +02:00
|
|
|
DBUG_ASSERT(null_value);
|
|
|
|
return 0;
|
2020-06-22 15:43:53 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_contains::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
a2_constant= args[1]->const_item();
|
|
|
|
a2_parsed= FALSE;
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2016-12-11 01:12:33 +04:00
|
|
|
if (arg_count > 2)
|
|
|
|
path.set_constant_flag(args[2]->const_item());
|
2022-01-26 03:02:45 +05:30
|
|
|
return Item_bool_func::fix_length_and_dec(thd);
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-11 01:12:33 +04:00
|
|
|
static int find_key_in_object(json_engine_t *j, json_string_t *key)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2016-12-11 01:12:33 +04:00
|
|
|
const uchar *c_str= key->c_str;
|
|
|
|
|
|
|
|
while (json_scan_next(j) == 0 && j->state != JST_OBJ_END)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2016-12-11 01:12:33 +04:00
|
|
|
DBUG_ASSERT(j->state == JST_KEY);
|
|
|
|
if (json_key_matches(j, key))
|
|
|
|
return TRUE;
|
|
|
|
if (json_skip_key(j))
|
|
|
|
return FALSE;
|
|
|
|
key->c_str= c_str;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
2016-12-11 01:12:33 +04:00
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int check_contains(json_engine_t *js, json_engine_t *value)
|
|
|
|
{
|
|
|
|
json_engine_t loc_js;
|
|
|
|
bool set_js;
|
2022-07-23 19:56:08 +05:30
|
|
|
DBUG_EXECUTE_IF("json_check_min_stack_requirement",
|
2022-07-29 16:17:09 +05:30
|
|
|
{
|
|
|
|
long arbitrary_var;
|
|
|
|
long stack_used_up= (available_stack_size(current_thd->thread_stack, &arbitrary_var));
|
|
|
|
ALLOCATE_MEM_ON_STACK(my_thread_stack_size-stack_used_up-STACK_MIN_SIZE);
|
|
|
|
});
|
2022-07-23 19:56:08 +05:30
|
|
|
if (check_stack_overrun(current_thd, STACK_MIN_SIZE , NULL))
|
|
|
|
return 1;
|
2016-12-11 01:12:33 +04:00
|
|
|
|
|
|
|
switch (js->value_type)
|
|
|
|
{
|
|
|
|
case JSON_VALUE_OBJECT:
|
|
|
|
{
|
|
|
|
json_string_t key_name;
|
|
|
|
|
|
|
|
if (value->value_type != JSON_VALUE_OBJECT)
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
loc_js= *js;
|
|
|
|
set_js= FALSE;
|
|
|
|
json_string_set_cs(&key_name, value->s.cs);
|
|
|
|
while (json_scan_next(value) == 0 && value->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
const uchar *k_start, *k_end;
|
|
|
|
|
|
|
|
DBUG_ASSERT(value->state == JST_KEY);
|
|
|
|
k_start= value->s.c_str;
|
2017-02-09 17:38:53 +04:00
|
|
|
do
|
|
|
|
{
|
2016-12-11 01:12:33 +04:00
|
|
|
k_end= value->s.c_str;
|
2017-02-09 17:38:53 +04:00
|
|
|
} while (json_read_keyname_chr(value) == 0);
|
2016-12-11 01:12:33 +04:00
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(value->s.error) || json_read_value(value))
|
2016-12-11 01:12:33 +04:00
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (set_js)
|
|
|
|
*js= loc_js;
|
|
|
|
else
|
|
|
|
set_js= TRUE;
|
|
|
|
|
|
|
|
json_string_set_str(&key_name, k_start, k_end);
|
|
|
|
if (!find_key_in_object(js, &key_name) ||
|
|
|
|
json_read_value(js) ||
|
|
|
|
!check_contains(js, value))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return value->state == JST_OBJ_END && !json_skip_level(js);
|
|
|
|
}
|
|
|
|
case JSON_VALUE_ARRAY:
|
|
|
|
if (value->value_type != JSON_VALUE_ARRAY)
|
|
|
|
{
|
2017-10-05 23:23:39 +04:00
|
|
|
loc_js= *value;
|
|
|
|
set_js= FALSE;
|
2016-12-11 01:12:33 +04:00
|
|
|
while (json_scan_next(js) == 0 && js->state != JST_ARRAY_END)
|
|
|
|
{
|
2017-01-24 17:34:44 +04:00
|
|
|
int c_level, v_scalar;
|
2016-12-11 01:12:33 +04:00
|
|
|
DBUG_ASSERT(js->state == JST_VALUE);
|
|
|
|
if (json_read_value(js))
|
|
|
|
return FALSE;
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
if (!(v_scalar= json_value_scalar(js)))
|
|
|
|
c_level= json_get_level(js);
|
|
|
|
|
2017-10-05 23:23:39 +04:00
|
|
|
if (set_js)
|
|
|
|
*value= loc_js;
|
|
|
|
else
|
|
|
|
set_js= TRUE;
|
|
|
|
|
2016-12-11 01:12:33 +04:00
|
|
|
if (check_contains(js, value))
|
|
|
|
{
|
|
|
|
if (json_skip_level(js))
|
|
|
|
return FALSE;
|
|
|
|
return TRUE;
|
|
|
|
}
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(value->s.error) || unlikely(js->s.error) ||
|
2017-01-24 17:34:44 +04:00
|
|
|
(!v_scalar && json_skip_to_level(js, c_level)))
|
2016-12-11 01:12:33 +04:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
/* else */
|
|
|
|
loc_js= *js;
|
|
|
|
set_js= FALSE;
|
|
|
|
while (json_scan_next(value) == 0 && value->state != JST_ARRAY_END)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(value->state == JST_VALUE);
|
|
|
|
if (json_read_value(value))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
if (set_js)
|
|
|
|
*js= loc_js;
|
|
|
|
else
|
|
|
|
set_js= TRUE;
|
|
|
|
if (!check_contains(js, value))
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
return value->state == JST_ARRAY_END;
|
|
|
|
|
|
|
|
case JSON_VALUE_STRING:
|
|
|
|
if (value->value_type != JSON_VALUE_STRING)
|
|
|
|
return FALSE;
|
|
|
|
/*
|
2021-03-11 09:50:29 +08:00
|
|
|
TODO: make proper json-json comparison here that takes excipient
|
2016-12-11 01:12:33 +04:00
|
|
|
into account.
|
|
|
|
*/
|
|
|
|
return value->value_len == js->value_len &&
|
|
|
|
memcmp(value->value, js->value, value->value_len) == 0;
|
|
|
|
case JSON_VALUE_NUMBER:
|
|
|
|
if (value->value_type == JSON_VALUE_NUMBER)
|
|
|
|
{
|
|
|
|
double d_j, d_v;
|
|
|
|
char *end;
|
|
|
|
int err;
|
|
|
|
|
2020-01-26 20:27:13 +04:00
|
|
|
d_j= js->s.cs->strntod((char *) js->value, js->value_len, &end, &err);;
|
|
|
|
d_v= value->s.cs->strntod((char *) value->value, value->value_len, &end, &err);;
|
2016-12-11 01:12:33 +04:00
|
|
|
|
|
|
|
return (fabs(d_j - d_v) < 1e-12);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
We have these not mentioned in the 'switch' above:
|
|
|
|
|
|
|
|
case JSON_VALUE_TRUE:
|
|
|
|
case JSON_VALUE_FALSE:
|
|
|
|
case JSON_VALUE_NULL:
|
|
|
|
*/
|
|
|
|
return value->value_type == js->value_type;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
longlong Item_func_json_contains::val_int()
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-12-11 01:12:33 +04:00
|
|
|
json_engine_t je, ve;
|
|
|
|
int result;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!a2_parsed)
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
val= args[1]->val_json(&tmp_val);
|
2016-10-19 14:10:03 +04:00
|
|
|
a2_parsed= a2_constant;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (val == 0)
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
2016-12-11 01:12:33 +04:00
|
|
|
if (arg_count>2) /* Path specified. */
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-12-11 01:12:33 +04:00
|
|
|
if (!path.parsed)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2016-12-11 01:12:33 +04:00
|
|
|
String *s_p= args[2]->val_str(&tmp_path);
|
2016-10-19 14:10:03 +04:00
|
|
|
if (s_p &&
|
2016-12-13 12:39:48 +04:00
|
|
|
path_setup_nwc(&path.p,s_p->charset(),(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->end()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &path.p, 2);
|
|
|
|
goto return_null;
|
|
|
|
}
|
2016-12-11 01:12:33 +04:00
|
|
|
path.parsed= path.constant;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
2016-12-11 01:12:33 +04:00
|
|
|
if (args[2]->null_value)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null;
|
2016-12-03 11:22:42 +04:00
|
|
|
|
2016-12-11 01:12:33 +04:00
|
|
|
path.cur_step= path.p.steps;
|
|
|
|
if (json_find_path(&je, &path.p, &path.cur_step, array_counters))
|
2016-12-13 12:39:48 +04:00
|
|
|
{
|
|
|
|
if (je.s.error)
|
|
|
|
{
|
|
|
|
ve.s.error= 0;
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
2016-12-11 01:12:33 +04:00
|
|
|
json_scan_start(&ve, val->charset(),(const uchar *) val->ptr(),
|
|
|
|
(const uchar *) val->end());
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2016-12-11 01:12:33 +04:00
|
|
|
if (json_read_value(&je) || json_read_value(&ve))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto error;
|
2016-12-11 01:12:33 +04:00
|
|
|
|
|
|
|
result= check_contains(&je, &ve);
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error || ve.s.error))
|
2016-12-11 01:12:33 +04:00
|
|
|
goto error;
|
|
|
|
|
|
|
|
return result;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
error:
|
2016-12-13 12:39:48 +04:00
|
|
|
if (je.s.error)
|
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
if (ve.s.error)
|
|
|
|
report_json_error(val, &ve, 1);
|
|
|
|
return_null:
|
2016-10-19 14:10:03 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_func_json_contains_path::fix_fields(THD *thd, Item **ref)
|
|
|
|
{
|
|
|
|
return alloc_tmp_paths(thd, arg_count-2, &paths, &tmp_paths) ||
|
2017-01-24 17:34:44 +04:00
|
|
|
(p_found= (bool *) alloc_root(thd->mem_root,
|
|
|
|
(arg_count-2)*sizeof(bool))) == NULL ||
|
2016-10-19 14:10:03 +04:00
|
|
|
Item_int_func::fix_fields(thd, ref);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_contains_path::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
ooa_constant= args[1]->const_item();
|
|
|
|
ooa_parsed= FALSE;
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2016-10-19 14:10:03 +04:00
|
|
|
mark_constant_paths(paths, args+2, arg_count-2);
|
2022-01-26 03:02:45 +05:30
|
|
|
return Item_bool_func::fix_length_and_dec(thd);
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Item_func_json_contains_path::cleanup()
|
|
|
|
{
|
|
|
|
if (tmp_paths)
|
|
|
|
{
|
|
|
|
for (uint i= arg_count-2; i>0; i--)
|
|
|
|
tmp_paths[i-1].free();
|
|
|
|
tmp_paths= 0;
|
|
|
|
}
|
|
|
|
Item_int_func::cleanup();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
static int parse_one_or_all(const Item_func *f, Item *ooa_arg,
|
2016-11-15 17:04:31 +04:00
|
|
|
bool *ooa_parsed, bool ooa_constant, bool *mode_one)
|
|
|
|
{
|
|
|
|
if (!*ooa_parsed)
|
|
|
|
{
|
|
|
|
char buff[20];
|
|
|
|
String *res, tmp(buff, sizeof(buff), &my_charset_bin);
|
2016-12-03 11:02:28 +04:00
|
|
|
if ((res= ooa_arg->val_str(&tmp)) == NULL)
|
|
|
|
return TRUE;
|
2016-12-13 12:39:48 +04:00
|
|
|
|
2016-11-15 17:04:31 +04:00
|
|
|
*mode_one=eq_ascii_string(res->charset(), "one",
|
|
|
|
res->ptr(), res->length());
|
|
|
|
if (!*mode_one)
|
|
|
|
{
|
|
|
|
if (!eq_ascii_string(res->charset(), "all", res->ptr(), res->length()))
|
2016-12-13 12:39:48 +04:00
|
|
|
{
|
|
|
|
THD *thd= current_thd;
|
|
|
|
push_warning_printf(thd, Sql_condition::WARN_LEVEL_WARN,
|
|
|
|
ER_JSON_ONE_OR_ALL, ER_THD(thd, ER_JSON_ONE_OR_ALL),
|
|
|
|
f->func_name());
|
|
|
|
*mode_one= TRUE;
|
2016-11-15 17:04:31 +04:00
|
|
|
return TRUE;
|
2016-12-13 12:39:48 +04:00
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
*ooa_parsed= ooa_constant;
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
#ifdef DUMMY
|
2016-10-19 14:10:03 +04:00
|
|
|
longlong Item_func_json_contains_path::val_int()
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-10-19 14:10:03 +04:00
|
|
|
json_engine_t je;
|
|
|
|
uint n_arg;
|
|
|
|
longlong result;
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
if (parse_one_or_all(this, args[1], &ooa_parsed, ooa_constant, &mode_one))
|
|
|
|
goto return_null;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
result= !mode_one;
|
|
|
|
for (n_arg=2; n_arg < arg_count; n_arg++)
|
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-10-19 14:10:03 +04:00
|
|
|
json_path_with_flags *c_path= paths + n_arg - 2;
|
|
|
|
if (!c_path->parsed)
|
|
|
|
{
|
2022-04-19 21:43:31 +05:30
|
|
|
String *s_p= args[n_arg]->val_str(tmp_paths + (n_arg-2));
|
|
|
|
if (s_p)
|
2016-12-13 12:39:48 +04:00
|
|
|
{
|
2022-04-19 21:43:31 +05:30
|
|
|
if (json_path_setup(&c_path->p,s_p->charset(),(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
c_path->parsed= c_path->constant;
|
|
|
|
has_negative_path|= c_path->p.types_used & JSON_PATH_NEGATIVE_INDEX;
|
2016-12-13 12:39:48 +04:00
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
2016-12-03 11:22:42 +04:00
|
|
|
if (args[n_arg]->null_value)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null;
|
2016-12-03 11:22:42 +04:00
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
|
|
|
c_path->cur_step= c_path->p.steps;
|
|
|
|
if (json_find_path(&je, &c_path->p, &c_path->cur_step, array_counters))
|
|
|
|
{
|
|
|
|
/* Path wasn't found. */
|
|
|
|
if (je.s.error)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
if (!mode_one)
|
|
|
|
{
|
|
|
|
result= 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (mode_one)
|
|
|
|
{
|
|
|
|
result= 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
js_error:
|
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
return_null:
|
2016-10-19 14:10:03 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
2017-01-24 17:34:44 +04:00
|
|
|
#endif /*DUMMY*/
|
|
|
|
|
|
|
|
longlong Item_func_json_contains_path::val_int()
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2017-01-24 17:34:44 +04:00
|
|
|
json_engine_t je;
|
|
|
|
uint n_arg;
|
|
|
|
longlong result;
|
|
|
|
json_path_t p;
|
|
|
|
int n_found;
|
2017-05-07 18:26:10 +03:00
|
|
|
LINT_INIT(n_found);
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_sizes[JSON_DEPTH_LIMIT];
|
|
|
|
uint has_negative_path= 0;
|
2017-01-24 17:34:44 +04:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (parse_one_or_all(this, args[1], &ooa_parsed, ooa_constant, &mode_one))
|
|
|
|
goto null_return;;
|
|
|
|
|
|
|
|
for (n_arg=2; n_arg < arg_count; n_arg++)
|
|
|
|
{
|
|
|
|
json_path_with_flags *c_path= paths + n_arg - 2;
|
2022-04-19 21:43:31 +05:30
|
|
|
c_path->p.types_used= JSON_PATH_KEY_NULL;
|
2017-01-24 17:34:44 +04:00
|
|
|
if (!c_path->parsed)
|
|
|
|
{
|
|
|
|
String *s_p= args[n_arg]->val_str(tmp_paths + (n_arg-2));
|
2022-04-19 21:43:31 +05:30
|
|
|
if (s_p)
|
2017-01-24 17:34:44 +04:00
|
|
|
{
|
2022-04-19 21:43:31 +05:30
|
|
|
if (json_path_setup(&c_path->p,s_p->charset(),(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
c_path->parsed= c_path->constant;
|
|
|
|
has_negative_path|= c_path->p.types_used & JSON_PATH_NEGATIVE_INDEX;
|
2017-01-24 17:34:44 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
if (args[n_arg]->null_value)
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_get_path_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length(), &p);
|
|
|
|
|
|
|
|
|
|
|
|
if (!mode_one)
|
|
|
|
{
|
|
|
|
bzero(p_found, (arg_count-2) * sizeof(bool));
|
|
|
|
n_found= arg_count - 2;
|
|
|
|
}
|
2017-10-30 14:59:43 +04:00
|
|
|
else
|
2021-03-11 09:50:29 +08:00
|
|
|
n_found= 0; /* Just to prevent 'uninitialized value' warnings */
|
2017-01-24 17:34:44 +04:00
|
|
|
|
|
|
|
result= 0;
|
|
|
|
while (json_get_path_next(&je, &p) == 0)
|
|
|
|
{
|
|
|
|
int n_path= arg_count - 2;
|
2021-11-22 22:59:30 +05:30
|
|
|
if (has_negative_path && je.value_type == JSON_VALUE_ARRAY &&
|
|
|
|
json_skip_array_and_count(&je, array_sizes + (p.last_step - p.steps)))
|
|
|
|
{
|
|
|
|
result= 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
json_path_with_flags *c_path= paths;
|
|
|
|
for (; n_path > 0; n_path--, c_path++)
|
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
if (json_path_compare(&c_path->p, &p, je.value_type, array_sizes) >= 0)
|
2017-01-24 17:34:44 +04:00
|
|
|
{
|
|
|
|
if (mode_one)
|
|
|
|
{
|
|
|
|
result= 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
/* mode_all */
|
|
|
|
if (p_found[n_path-1])
|
|
|
|
continue; /* already found */
|
|
|
|
if (--n_found == 0)
|
|
|
|
{
|
|
|
|
result= 1;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
p_found[n_path-1]= TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (likely(je.s.error == 0))
|
2017-01-24 17:34:44 +04:00
|
|
|
return result;
|
|
|
|
|
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
null_return:
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
|
2022-01-10 18:05:55 +04:00
|
|
|
/*
|
|
|
|
This reproduces behavior according to the former
|
|
|
|
Item_func_conv_charset::is_json_type() which returned args[0]->is_json_type().
|
|
|
|
JSON functions with multiple string input with different character sets
|
|
|
|
wrap some arguments into Item_func_conv_charset. So the former
|
|
|
|
Item_func_conv_charset::is_json_type() took the JSON propery from args[0],
|
|
|
|
i.e. from the original argument before the conversion.
|
|
|
|
This is probably not always correct because an *explicit*
|
|
|
|
`CONVERT(arg USING charset)` is actually a general purpose string
|
|
|
|
expression, not a JSON expression.
|
|
|
|
*/
|
2022-02-03 17:01:31 +01:00
|
|
|
bool is_json_type(const Item *item)
|
2022-01-10 18:05:55 +04:00
|
|
|
{
|
|
|
|
for ( ; ; )
|
|
|
|
{
|
|
|
|
if (Type_handler_json_common::is_json_type_handler(item->type_handler()))
|
|
|
|
return true;
|
|
|
|
const Item_func_conv_charset *func;
|
2023-01-16 12:08:05 +01:00
|
|
|
if (!(func= dynamic_cast<const Item_func_conv_charset*>(item->real_item())))
|
2022-01-10 18:05:55 +04:00
|
|
|
return false;
|
|
|
|
item= func->arguments()[0];
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
static int append_json_value(String *str, Item *item, String *tmp_val)
|
|
|
|
{
|
2018-05-31 18:52:32 +04:00
|
|
|
if (item->type_handler()->is_bool_type())
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
longlong v_int= item->val_int();
|
|
|
|
const char *t_f;
|
|
|
|
int t_f_len;
|
|
|
|
|
|
|
|
if (item->null_value)
|
|
|
|
goto append_null;
|
|
|
|
|
|
|
|
if (v_int)
|
|
|
|
{
|
|
|
|
t_f= "true";
|
|
|
|
t_f_len= 4;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
t_f= "false";
|
|
|
|
t_f_len= 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
return str->append(t_f, t_f_len);
|
|
|
|
}
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *sv= item->val_json(tmp_val);
|
2016-10-19 14:10:03 +04:00
|
|
|
if (item->null_value)
|
|
|
|
goto append_null;
|
2022-01-10 18:05:55 +04:00
|
|
|
if (is_json_type(item))
|
2016-11-15 17:04:31 +04:00
|
|
|
return str->append(sv->ptr(), sv->length());
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
if (item->result_type() == STRING_RESULT)
|
|
|
|
{
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return str->append('"') ||
|
2016-10-19 14:10:03 +04:00
|
|
|
st_append_escaped(str, sv) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append('"');
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
return st_append_escaped(str, sv);
|
|
|
|
}
|
|
|
|
|
|
|
|
append_null:
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return str->append(STRING_WITH_LEN("null"));
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-04 13:53:14 +04:00
|
|
|
static int append_json_value_from_field(String *str,
|
|
|
|
Item *i, Field *f, const uchar *key, size_t offset, String *tmp_val)
|
|
|
|
{
|
|
|
|
if (i->type_handler()->is_bool_type())
|
|
|
|
{
|
|
|
|
longlong v_int= f->val_int(key + offset);
|
|
|
|
const char *t_f;
|
|
|
|
int t_f_len;
|
|
|
|
|
2020-06-09 13:15:14 +05:30
|
|
|
if (f->is_null_in_record(key))
|
2020-06-04 13:53:14 +04:00
|
|
|
goto append_null;
|
|
|
|
|
|
|
|
if (v_int)
|
|
|
|
{
|
|
|
|
t_f= "true";
|
|
|
|
t_f_len= 4;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
t_f= "false";
|
|
|
|
t_f_len= 5;
|
|
|
|
}
|
|
|
|
|
|
|
|
return str->append(t_f, t_f_len);
|
|
|
|
}
|
|
|
|
{
|
|
|
|
String *sv= f->val_str(tmp_val, key + offset);
|
2020-06-09 13:15:14 +05:30
|
|
|
if (f->is_null_in_record(key))
|
2020-06-04 13:53:14 +04:00
|
|
|
goto append_null;
|
2022-01-10 18:05:55 +04:00
|
|
|
if (is_json_type(i))
|
2020-06-04 13:53:14 +04:00
|
|
|
return str->append(sv->ptr(), sv->length());
|
|
|
|
|
|
|
|
if (i->result_type() == STRING_RESULT)
|
|
|
|
{
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return str->append('"') ||
|
2020-06-04 13:53:14 +04:00
|
|
|
st_append_escaped(str, sv) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append('"');
|
2020-06-04 13:53:14 +04:00
|
|
|
}
|
|
|
|
return st_append_escaped(str, sv);
|
|
|
|
}
|
|
|
|
|
|
|
|
append_null:
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return str->append(STRING_WITH_LEN("null"));
|
2020-06-04 13:53:14 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
static int append_json_keyname(String *str, Item *item, String *tmp_val)
|
|
|
|
{
|
|
|
|
String *sv= item->val_str(tmp_val);
|
|
|
|
if (item->null_value)
|
|
|
|
goto append_null;
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return str->append('"') ||
|
2016-10-19 14:10:03 +04:00
|
|
|
st_append_escaped(str, sv) ||
|
|
|
|
str->append("\": ", 3);
|
|
|
|
|
|
|
|
append_null:
|
|
|
|
return str->append("\"\": ", 4);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_array::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2016-12-04 14:22:01 +04:00
|
|
|
ulonglong char_length= 2;
|
2016-10-19 14:10:03 +04:00
|
|
|
uint n_arg;
|
|
|
|
|
2017-08-08 13:49:29 +04:00
|
|
|
result_limit= 0;
|
|
|
|
|
2016-12-03 11:45:24 +04:00
|
|
|
if (arg_count == 0)
|
|
|
|
{
|
2019-03-12 01:09:55 +04:00
|
|
|
THD* thd= current_thd;
|
|
|
|
collation.set(thd->variables.collation_connection,
|
2017-10-06 09:28:33 +04:00
|
|
|
DERIVATION_COERCIBLE, MY_REPERTOIRE_ASCII);
|
2019-03-12 01:09:55 +04:00
|
|
|
tmp_val.set_charset(thd->variables.collation_connection);
|
2016-12-03 11:45:24 +04:00
|
|
|
max_length= 2;
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-12-03 11:45:24 +04:00
|
|
|
}
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
if (agg_arg_charsets_for_string_result(collation, args, arg_count))
|
2018-05-08 15:26:26 +02:00
|
|
|
return TRUE;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
for (n_arg=0 ; n_arg < arg_count ; n_arg++)
|
2022-08-13 12:49:48 +08:00
|
|
|
char_length+= static_cast<ulonglong>(args[n_arg]->max_char_length()) + 4;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
fix_char_length_ulonglong(char_length);
|
|
|
|
tmp_val.set_charset(collation.collation);
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_array::val_str(String *str)
|
|
|
|
{
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2016-10-19 14:10:03 +04:00
|
|
|
uint n_arg;
|
|
|
|
|
|
|
|
str->length(0);
|
2017-09-13 15:17:28 +04:00
|
|
|
str->set_charset(collation.collation);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('[') ||
|
2016-12-13 12:39:48 +04:00
|
|
|
((arg_count > 0) && append_json_value(str, args[0], &tmp_val)))
|
2016-10-19 14:10:03 +04:00
|
|
|
goto err_return;
|
|
|
|
|
|
|
|
for (n_arg=1; n_arg < arg_count; n_arg++)
|
|
|
|
{
|
|
|
|
if (str->append(", ", 2) ||
|
|
|
|
append_json_value(str, args[n_arg], &tmp_val))
|
|
|
|
goto err_return;
|
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append(']'))
|
2016-10-19 14:10:03 +04:00
|
|
|
goto err_return;
|
|
|
|
|
2017-03-14 17:31:14 +04:00
|
|
|
if (result_limit == 0)
|
|
|
|
result_limit= current_thd->variables.max_allowed_packet;
|
|
|
|
|
|
|
|
if (str->length() <= result_limit)
|
|
|
|
return str;
|
|
|
|
|
|
|
|
push_warning_printf(current_thd, Sql_condition::WARN_LEVEL_WARN,
|
|
|
|
ER_WARN_ALLOWED_PACKET_OVERFLOWED,
|
|
|
|
ER_THD(current_thd, ER_WARN_ALLOWED_PACKET_OVERFLOWED),
|
|
|
|
func_name(), result_limit);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
err_return:
|
|
|
|
/*TODO: Launch out of memory error. */
|
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_array_append::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
|
|
|
uint n_arg;
|
|
|
|
ulonglong char_length;
|
|
|
|
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
char_length= args[0]->max_char_length();
|
|
|
|
|
|
|
|
for (n_arg= 1; n_arg < arg_count; n_arg+= 2)
|
|
|
|
{
|
2016-11-16 12:47:46 +04:00
|
|
|
paths[n_arg/2].set_constant_flag(args[n_arg]->const_item());
|
2022-08-13 12:49:48 +08:00
|
|
|
char_length+=
|
|
|
|
static_cast<ulonglong>(args[n_arg+1]->max_char_length()) + 4;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
fix_char_length_ulonglong(char_length);
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_array_append::val_str(String *str)
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2017-09-28 10:38:02 +00:00
|
|
|
uint n_arg, n_path;
|
|
|
|
size_t str_rest_len;
|
2016-10-19 14:10:03 +04:00
|
|
|
const uchar *ar_end;
|
2021-07-02 19:13:26 +02:00
|
|
|
THD *thd= current_thd;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (n_arg=1, n_path=0; n_arg < arg_count; n_arg+=2, n_path++)
|
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-10-19 14:10:03 +04:00
|
|
|
json_path_with_flags *c_path= paths + n_path;
|
|
|
|
if (!c_path->parsed)
|
|
|
|
{
|
|
|
|
String *s_p= args[n_arg]->val_str(tmp_paths+n_path);
|
|
|
|
if (s_p &&
|
2016-12-13 12:39:48 +04:00
|
|
|
path_setup_nwc(&c_path->p,s_p->charset(),(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
|
|
|
goto return_null;
|
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
c_path->parsed= c_path->constant;
|
|
|
|
}
|
|
|
|
if (args[n_arg]->null_value)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
c_path->cur_step= c_path->p.steps;
|
|
|
|
|
|
|
|
if (json_find_path(&je, &c_path->p, &c_path->cur_step, array_counters))
|
|
|
|
{
|
|
|
|
if (je.s.error)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
|
|
|
|
|
|
|
goto return_null;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (json_read_value(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
str->length(0);
|
|
|
|
str->set_charset(js->charset());
|
|
|
|
if (str->reserve(js->length() + 8, 1024))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null; /* Out of memory. */
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2016-12-05 08:43:15 +04:00
|
|
|
if (je.value_type == JSON_VALUE_ARRAY)
|
|
|
|
{
|
2018-09-11 14:37:45 +04:00
|
|
|
int n_items;
|
|
|
|
if (json_skip_level_and_count(&je, &n_items))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-12-05 08:43:15 +04:00
|
|
|
|
|
|
|
ar_end= je.s.c_str - je.sav_c_len;
|
|
|
|
str_rest_len= js->length() - (ar_end - (const uchar *) js->ptr());
|
|
|
|
str->q_append(js->ptr(), ar_end-(const uchar *) js->ptr());
|
2018-09-11 14:37:45 +04:00
|
|
|
if (n_items)
|
|
|
|
str->append(", ", 2);
|
2016-12-05 08:43:15 +04:00
|
|
|
if (append_json_value(str, args[n_arg+1], &tmp_val))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null; /* Out of memory. */
|
2016-12-05 08:43:15 +04:00
|
|
|
|
|
|
|
if (str->reserve(str_rest_len, 1024))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null; /* Out of memory. */
|
2016-12-05 08:43:15 +04:00
|
|
|
str->q_append((const char *) ar_end, str_rest_len);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const uchar *c_from, *c_to;
|
|
|
|
|
|
|
|
/* Wrap as an array. */
|
|
|
|
str->q_append(js->ptr(), (const char *) je.value_begin - js->ptr());
|
|
|
|
c_from= je.value_begin;
|
|
|
|
|
|
|
|
if (je.value_type == JSON_VALUE_OBJECT)
|
|
|
|
{
|
|
|
|
if (json_skip_level(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-12-05 08:43:15 +04:00
|
|
|
c_to= je.s.c_str;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
c_to= je.value_end;
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('[') ||
|
2016-12-05 08:43:15 +04:00
|
|
|
str->append((const char *) c_from, c_to - c_from) ||
|
|
|
|
str->append(", ", 2) ||
|
|
|
|
append_json_value(str, args[n_arg+1], &tmp_val) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append(']') ||
|
2016-12-05 08:43:15 +04:00
|
|
|
str->append((const char *) je.s.c_str,
|
|
|
|
js->end() - (const char *) je.s.c_str))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null; /* Out of memory. */
|
2016-12-05 08:43:15 +04:00
|
|
|
}
|
|
|
|
{
|
|
|
|
/* Swap str and js. */
|
|
|
|
if (str == &tmp_js)
|
|
|
|
{
|
|
|
|
str= js;
|
|
|
|
js= &tmp_js;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
js= str;
|
|
|
|
str= &tmp_js;
|
|
|
|
}
|
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2017-02-06 06:47:48 +04:00
|
|
|
if (json_nice(&je, str, Item_func_json_format::LOOSE))
|
|
|
|
goto js_error;
|
|
|
|
|
|
|
|
return str;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
js_error:
|
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
|
|
|
|
return_null:
|
2021-07-02 19:13:26 +02:00
|
|
|
thd->check_killed(); // to get the error message right
|
2016-10-19 14:10:03 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-11-15 17:04:31 +04:00
|
|
|
String *Item_func_json_array_insert::val_str(String *str)
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-11-15 17:04:31 +04:00
|
|
|
uint n_arg, n_path;
|
2021-07-02 19:13:26 +02:00
|
|
|
THD *thd= current_thd;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
for (n_arg=1, n_path=0; n_arg < arg_count; n_arg+=2, n_path++)
|
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-11-15 17:04:31 +04:00
|
|
|
json_path_with_flags *c_path= paths + n_path;
|
|
|
|
const char *item_pos;
|
2021-11-22 22:59:30 +05:30
|
|
|
int n_item, corrected_n_item;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (!c_path->parsed)
|
|
|
|
{
|
|
|
|
String *s_p= args[n_arg]->val_str(tmp_paths+n_path);
|
|
|
|
if (s_p &&
|
2016-12-13 12:39:48 +04:00
|
|
|
(path_setup_nwc(&c_path->p,s_p->charset(),(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()) ||
|
2016-12-05 08:03:11 +04:00
|
|
|
c_path->p.last_step - 1 < c_path->p.steps ||
|
2016-11-15 17:04:31 +04:00
|
|
|
c_path->p.last_step->type != JSON_PATH_ARRAY))
|
2016-12-13 12:39:48 +04:00
|
|
|
{
|
|
|
|
if (c_path->p.s.error == 0)
|
|
|
|
c_path->p.s.error= SHOULD_END_WITH_ARRAY;
|
|
|
|
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
|
|
|
|
|
|
|
goto return_null;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
c_path->parsed= c_path->constant;
|
|
|
|
c_path->p.last_step--;
|
|
|
|
}
|
|
|
|
if (args[n_arg]->null_value)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
c_path->cur_step= c_path->p.steps;
|
|
|
|
|
|
|
|
if (json_find_path(&je, &c_path->p, &c_path->cur_step, array_counters))
|
|
|
|
{
|
|
|
|
if (je.s.error)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
/* Can't find the array to insert. */
|
2016-12-13 12:39:48 +04:00
|
|
|
continue;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (json_read_value(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (je.value_type != JSON_VALUE_ARRAY)
|
|
|
|
{
|
|
|
|
/* Must be an array. */
|
2016-12-13 12:39:48 +04:00
|
|
|
continue;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
item_pos= 0;
|
|
|
|
n_item= 0;
|
2021-11-22 22:59:30 +05:30
|
|
|
corrected_n_item= c_path->p.last_step[1].n_item;
|
|
|
|
if (corrected_n_item < 0)
|
|
|
|
{
|
|
|
|
int array_size;
|
|
|
|
if (json_skip_array_and_count(&je, &array_size))
|
|
|
|
goto js_error;
|
|
|
|
corrected_n_item+= array_size + 1;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2016-12-16 13:51:35 +04:00
|
|
|
while (json_scan_next(&je) == 0 && je.state != JST_ARRAY_END)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2016-12-16 13:51:35 +04:00
|
|
|
DBUG_ASSERT(je.state == JST_VALUE);
|
2021-11-22 22:59:30 +05:30
|
|
|
|
|
|
|
if (n_item == corrected_n_item)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2016-12-16 13:51:35 +04:00
|
|
|
item_pos= (const char *) je.s.c_str;
|
2016-11-15 17:04:31 +04:00
|
|
|
break;
|
|
|
|
}
|
2016-12-16 13:51:35 +04:00
|
|
|
n_item++;
|
|
|
|
|
|
|
|
if (json_read_value(&je) ||
|
|
|
|
(!json_value_scalar(&je) && json_skip_level(&je)))
|
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
2021-07-02 19:13:26 +02:00
|
|
|
if (unlikely(je.s.error || *je.killed_ptr))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
|
|
|
|
2016-11-15 17:04:31 +04:00
|
|
|
str->length(0);
|
|
|
|
str->set_charset(js->charset());
|
2016-12-16 13:51:35 +04:00
|
|
|
if (item_pos)
|
|
|
|
{
|
|
|
|
if (append_simple(str, js->ptr(), item_pos - js->ptr()) ||
|
|
|
|
(n_item > 0 && str->append(" ", 1)) ||
|
|
|
|
append_json_value(str, args[n_arg+1], &tmp_val) ||
|
|
|
|
str->append(",", 1) ||
|
|
|
|
(n_item == 0 && str->append(" ", 1)) ||
|
|
|
|
append_simple(str, item_pos, js->end() - item_pos))
|
|
|
|
goto return_null; /* Out of memory. */
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Insert position wasn't found - append to the array. */
|
|
|
|
DBUG_ASSERT(je.state == JST_ARRAY_END);
|
2016-11-15 17:04:31 +04:00
|
|
|
item_pos= (const char *) (je.s.c_str - je.sav_c_len);
|
2016-12-16 13:51:35 +04:00
|
|
|
if (append_simple(str, js->ptr(), item_pos - js->ptr()) ||
|
|
|
|
(n_item > 0 && str->append(", ", 2)) ||
|
|
|
|
append_json_value(str, args[n_arg+1], &tmp_val) ||
|
|
|
|
append_simple(str, item_pos, js->end() - item_pos))
|
|
|
|
goto return_null; /* Out of memory. */
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
{
|
2016-12-05 08:43:15 +04:00
|
|
|
/* Swap str and js. */
|
|
|
|
if (str == &tmp_js)
|
|
|
|
{
|
|
|
|
str= js;
|
|
|
|
js= &tmp_js;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
js= str;
|
|
|
|
str= &tmp_js;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2017-02-06 06:47:48 +04:00
|
|
|
if (json_nice(&je, str, Item_func_json_format::LOOSE))
|
|
|
|
goto js_error;
|
|
|
|
|
|
|
|
return str;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
js_error:
|
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
return_null:
|
2021-07-02 19:13:26 +02:00
|
|
|
thd->check_killed(); // to get the error message right
|
2016-11-15 17:04:31 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
String *Item_func_json_object::val_str(String *str)
|
|
|
|
{
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2016-10-19 14:10:03 +04:00
|
|
|
uint n_arg;
|
|
|
|
|
|
|
|
str->length(0);
|
2017-09-13 15:17:28 +04:00
|
|
|
str->set_charset(collation.collation);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('{') ||
|
2016-10-19 14:10:03 +04:00
|
|
|
(arg_count > 0 &&
|
|
|
|
(append_json_keyname(str, args[0], &tmp_val) ||
|
|
|
|
append_json_value(str, args[1], &tmp_val))))
|
|
|
|
goto err_return;
|
|
|
|
|
|
|
|
for (n_arg=2; n_arg < arg_count; n_arg+=2)
|
|
|
|
{
|
|
|
|
if (str->append(", ", 2) ||
|
|
|
|
append_json_keyname(str, args[n_arg], &tmp_val) ||
|
|
|
|
append_json_value(str, args[n_arg+1], &tmp_val))
|
|
|
|
goto err_return;
|
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('}'))
|
2016-10-19 14:10:03 +04:00
|
|
|
goto err_return;
|
|
|
|
|
2017-03-14 17:31:14 +04:00
|
|
|
if (result_limit == 0)
|
|
|
|
result_limit= current_thd->variables.max_allowed_packet;
|
|
|
|
|
|
|
|
if (str->length() <= result_limit)
|
|
|
|
return str;
|
|
|
|
|
|
|
|
push_warning_printf(current_thd, Sql_condition::WARN_LEVEL_WARN,
|
|
|
|
ER_WARN_ALLOWED_PACKET_OVERFLOWED,
|
|
|
|
ER_THD(current_thd, ER_WARN_ALLOWED_PACKET_OVERFLOWED),
|
|
|
|
func_name(), result_limit);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
err_return:
|
|
|
|
/*TODO: Launch out of memory error. */
|
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-10 01:05:27 +04:00
|
|
|
static int do_merge(String *str, json_engine_t *je1, json_engine_t *je2)
|
|
|
|
{
|
2022-06-21 14:58:34 +05:30
|
|
|
|
|
|
|
DBUG_EXECUTE_IF("json_check_min_stack_requirement",
|
2022-07-29 16:17:09 +05:30
|
|
|
{
|
|
|
|
long arbitrary_var;
|
|
|
|
long stack_used_up= (available_stack_size(current_thd->thread_stack, &arbitrary_var));
|
|
|
|
ALLOCATE_MEM_ON_STACK(my_thread_stack_size-stack_used_up-STACK_MIN_SIZE);
|
|
|
|
});
|
2022-07-23 19:56:08 +05:30
|
|
|
if (check_stack_overrun(current_thd, STACK_MIN_SIZE , NULL))
|
|
|
|
return 1;
|
2022-06-21 14:58:34 +05:30
|
|
|
|
2017-02-10 01:05:27 +04:00
|
|
|
if (json_read_value(je1) || json_read_value(je2))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (je1->value_type == JSON_VALUE_OBJECT &&
|
|
|
|
je2->value_type == JSON_VALUE_OBJECT)
|
|
|
|
{
|
|
|
|
json_engine_t sav_je1= *je1;
|
|
|
|
json_engine_t sav_je2= *je2;
|
|
|
|
|
|
|
|
int first_key= 1;
|
|
|
|
json_string_t key_name;
|
|
|
|
|
|
|
|
json_string_set_cs(&key_name, je1->s.cs);
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('{'))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 3;
|
|
|
|
while (json_scan_next(je1) == 0 &&
|
|
|
|
je1->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
const uchar *key_start, *key_end;
|
|
|
|
/* Loop through the Json_1 keys and compare with the Json_2 keys. */
|
|
|
|
DBUG_ASSERT(je1->state == JST_KEY);
|
|
|
|
key_start= je1->s.c_str;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
key_end= je1->s.c_str;
|
|
|
|
} while (json_read_keyname_chr(je1) == 0);
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je1->s.error))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (first_key)
|
|
|
|
first_key= 0;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (str->append(", ", 2))
|
|
|
|
return 3;
|
|
|
|
*je2= sav_je2;
|
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('"') ||
|
2017-02-10 01:05:27 +04:00
|
|
|
append_simple(str, key_start, key_end - key_start) ||
|
|
|
|
str->append("\":", 2))
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
while (json_scan_next(je2) == 0 &&
|
|
|
|
je2->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
int ires;
|
|
|
|
DBUG_ASSERT(je2->state == JST_KEY);
|
|
|
|
json_string_set_str(&key_name, key_start, key_end);
|
|
|
|
if (!json_key_matches(je2, &key_name))
|
|
|
|
{
|
|
|
|
if (je2->s.error || json_skip_key(je2))
|
|
|
|
return 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Json_2 has same key as Json_1. Merge them. */
|
|
|
|
if ((ires= do_merge(str, je1, je2)))
|
|
|
|
return ires;
|
|
|
|
goto merged_j1;
|
|
|
|
}
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je2->s.error))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 2;
|
|
|
|
|
|
|
|
key_start= je1->s.c_str;
|
|
|
|
/* Just append the Json_1 key value. */
|
|
|
|
if (json_skip_key(je1))
|
|
|
|
return 1;
|
|
|
|
if (append_simple(str, key_start, je1->s.c_str - key_start))
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
merged_j1:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
*je2= sav_je2;
|
|
|
|
/*
|
|
|
|
Now loop through the Json_2 keys.
|
|
|
|
Skip if there is same key in Json_1
|
|
|
|
*/
|
|
|
|
while (json_scan_next(je2) == 0 &&
|
|
|
|
je2->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
const uchar *key_start, *key_end;
|
|
|
|
DBUG_ASSERT(je2->state == JST_KEY);
|
|
|
|
key_start= je2->s.c_str;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
key_end= je2->s.c_str;
|
|
|
|
} while (json_read_keyname_chr(je2) == 0);
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je2->s.error))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
*je1= sav_je1;
|
|
|
|
while (json_scan_next(je1) == 0 &&
|
|
|
|
je1->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(je1->state == JST_KEY);
|
|
|
|
json_string_set_str(&key_name, key_start, key_end);
|
|
|
|
if (!json_key_matches(je1, &key_name))
|
|
|
|
{
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je1->s.error || json_skip_key(je1)))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 2;
|
|
|
|
continue;
|
|
|
|
}
|
2021-07-01 20:02:12 +02:00
|
|
|
if (json_skip_key(je2) || json_skip_level(je1))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 1;
|
|
|
|
goto continue_j2;
|
|
|
|
}
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je1->s.error))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 2;
|
|
|
|
|
|
|
|
if (first_key)
|
|
|
|
first_key= 0;
|
|
|
|
else if (str->append(", ", 2))
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
if (json_skip_key(je2))
|
|
|
|
return 1;
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('"') ||
|
2017-02-10 01:05:27 +04:00
|
|
|
append_simple(str, key_start, je2->s.c_str - key_start))
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
continue_j2:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('}'))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
const uchar *end1, *beg1, *end2, *beg2;
|
2018-09-13 13:42:09 +04:00
|
|
|
int n_items1=1, n_items2= 1;
|
2017-02-10 01:05:27 +04:00
|
|
|
|
|
|
|
beg1= je1->value_begin;
|
|
|
|
|
|
|
|
/* Merge as a single array. */
|
|
|
|
if (je1->value_type == JSON_VALUE_ARRAY)
|
|
|
|
{
|
2018-09-13 13:42:09 +04:00
|
|
|
if (json_skip_level_and_count(je1, &n_items1))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 1;
|
2017-09-12 15:21:53 +04:00
|
|
|
|
2017-02-10 01:05:27 +04:00
|
|
|
end1= je1->s.c_str - je1->sav_c_len;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('['))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 3;
|
|
|
|
if (je1->value_type == JSON_VALUE_OBJECT)
|
|
|
|
{
|
|
|
|
if (json_skip_level(je1))
|
|
|
|
return 1;
|
|
|
|
end1= je1->s.c_str;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
end1= je1->value_end;
|
|
|
|
}
|
|
|
|
|
2018-09-13 13:42:09 +04:00
|
|
|
if (str->append((const char*) beg1, end1 - beg1))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 3;
|
|
|
|
|
|
|
|
if (json_value_scalar(je2))
|
|
|
|
{
|
|
|
|
beg2= je2->value_begin;
|
|
|
|
end2= je2->value_end;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (je2->value_type == JSON_VALUE_OBJECT)
|
2018-09-13 13:42:09 +04:00
|
|
|
{
|
2017-02-10 01:05:27 +04:00
|
|
|
beg2= je2->value_begin;
|
2018-09-13 13:42:09 +04:00
|
|
|
if (json_skip_level(je2))
|
|
|
|
return 2;
|
|
|
|
}
|
2017-02-10 01:05:27 +04:00
|
|
|
else
|
2018-09-13 13:42:09 +04:00
|
|
|
{
|
2017-02-10 01:05:27 +04:00
|
|
|
beg2= je2->s.c_str;
|
2018-09-13 13:42:09 +04:00
|
|
|
if (json_skip_level_and_count(je2, &n_items2))
|
|
|
|
return 2;
|
|
|
|
}
|
2017-02-10 01:05:27 +04:00
|
|
|
end2= je2->s.c_str;
|
|
|
|
}
|
|
|
|
|
2018-09-13 13:42:09 +04:00
|
|
|
if ((n_items1 && n_items2 && str->append(", ", 2)) ||
|
|
|
|
str->append((const char*) beg2, end2 - beg2))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 3;
|
|
|
|
|
|
|
|
if (je2->value_type != JSON_VALUE_ARRAY &&
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append(']'))
|
2017-02-10 01:05:27 +04:00
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
String *Item_func_json_merge::val_str(String *str)
|
|
|
|
{
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2016-12-06 01:32:13 +04:00
|
|
|
json_engine_t je1, je2;
|
2017-10-30 14:59:43 +04:00
|
|
|
String *js1= args[0]->val_json(&tmp_js1), *js2=NULL;
|
2016-10-19 14:10:03 +04:00
|
|
|
uint n_arg;
|
2021-07-02 19:13:26 +02:00
|
|
|
THD *thd= current_thd;
|
2017-05-07 18:26:10 +03:00
|
|
|
LINT_INIT(js2);
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2016-12-06 01:32:13 +04:00
|
|
|
if (args[0]->null_value)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto null_return;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
for (n_arg=1; n_arg < arg_count; n_arg++)
|
|
|
|
{
|
2017-02-10 01:05:27 +04:00
|
|
|
str->set_charset(js1->charset());
|
|
|
|
str->length(0);
|
|
|
|
|
2017-02-14 11:11:47 +01:00
|
|
|
js2= args[n_arg]->val_json(&tmp_js2);
|
2016-12-06 01:32:13 +04:00
|
|
|
if (args[n_arg]->null_value)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto null_return;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2016-12-06 01:32:13 +04:00
|
|
|
json_scan_start(&je1, js1->charset(),(const uchar *) js1->ptr(),
|
|
|
|
(const uchar *) js1->ptr() + js1->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je1.killed_ptr= (uchar*)&thd->killed;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2016-12-06 01:32:13 +04:00
|
|
|
json_scan_start(&je2, js2->charset(),(const uchar *) js2->ptr(),
|
|
|
|
(const uchar *) js2->ptr() + js2->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je2.killed_ptr= (uchar*)&thd->killed;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-02-10 01:05:27 +04:00
|
|
|
if (do_merge(str, &je1, &je2))
|
2016-12-06 01:32:13 +04:00
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
{
|
|
|
|
/* Swap str and js1. */
|
|
|
|
if (str == &tmp_js1)
|
|
|
|
{
|
|
|
|
str= js1;
|
|
|
|
js1= &tmp_js1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
js1= str;
|
|
|
|
str= &tmp_js1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
json_scan_start(&je1, js1->charset(),(const uchar *) js1->ptr(),
|
|
|
|
(const uchar *) js1->ptr() + js1->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je1.killed_ptr= (uchar*)&thd->killed;
|
2017-02-06 06:47:48 +04:00
|
|
|
if (json_nice(&je1, str, Item_func_json_format::LOOSE))
|
|
|
|
goto error_return;
|
|
|
|
|
2016-12-06 01:32:13 +04:00
|
|
|
null_value= 0;
|
2017-02-06 06:47:48 +04:00
|
|
|
return str;
|
2016-12-06 01:32:13 +04:00
|
|
|
|
2019-05-17 11:53:58 +04:00
|
|
|
error_return:
|
|
|
|
if (je1.s.error)
|
|
|
|
report_json_error(js1, &je1, 0);
|
|
|
|
if (je2.s.error)
|
|
|
|
report_json_error(js2, &je2, n_arg);
|
2021-07-02 19:13:26 +02:00
|
|
|
thd->check_killed(); // to get the error message right
|
2019-05-17 11:53:58 +04:00
|
|
|
null_return:
|
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int copy_value_patch(String *str, json_engine_t *je)
|
|
|
|
{
|
|
|
|
int first_key= 1;
|
|
|
|
|
|
|
|
if (je->value_type != JSON_VALUE_OBJECT)
|
|
|
|
{
|
|
|
|
const uchar *beg, *end;
|
|
|
|
|
|
|
|
beg= je->value_begin;
|
|
|
|
|
|
|
|
if (!json_value_scalar(je))
|
|
|
|
{
|
|
|
|
if (json_skip_level(je))
|
|
|
|
return 1;
|
|
|
|
end= je->s.c_str;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
end= je->value_end;
|
|
|
|
|
|
|
|
if (append_simple(str, beg, end-beg))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
/* JSON_VALUE_OBJECT */
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('{'))
|
2019-05-17 11:53:58 +04:00
|
|
|
return 1;
|
|
|
|
while (json_scan_next(je) == 0 && je->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
const uchar *key_start;
|
|
|
|
/* Loop through the Json_1 keys and compare with the Json_2 keys. */
|
|
|
|
DBUG_ASSERT(je->state == JST_KEY);
|
|
|
|
key_start= je->s.c_str;
|
|
|
|
|
|
|
|
if (json_read_value(je))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (je->value_type == JSON_VALUE_NULL)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (!first_key)
|
|
|
|
{
|
|
|
|
if (str->append(", ", 2))
|
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
first_key= 0;
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('"') ||
|
2019-05-17 11:53:58 +04:00
|
|
|
append_simple(str, key_start, je->value_begin - key_start) ||
|
|
|
|
copy_value_patch(str, je))
|
|
|
|
return 1;
|
|
|
|
}
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('}'))
|
2019-05-17 11:53:58 +04:00
|
|
|
return 1;
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int do_merge_patch(String *str, json_engine_t *je1, json_engine_t *je2,
|
|
|
|
bool *empty_result)
|
|
|
|
{
|
2022-06-21 14:58:34 +05:30
|
|
|
DBUG_EXECUTE_IF("json_check_min_stack_requirement",
|
2022-07-29 16:17:09 +05:30
|
|
|
{
|
|
|
|
long arbitrary_var;
|
|
|
|
long stack_used_up= (available_stack_size(current_thd->thread_stack, &arbitrary_var));
|
|
|
|
ALLOCATE_MEM_ON_STACK(my_thread_stack_size-stack_used_up-STACK_MIN_SIZE);
|
|
|
|
});
|
2022-07-23 19:56:08 +05:30
|
|
|
if (check_stack_overrun(current_thd, STACK_MIN_SIZE , NULL))
|
2022-06-21 14:58:34 +05:30
|
|
|
return 1;
|
|
|
|
|
2019-05-17 11:53:58 +04:00
|
|
|
if (json_read_value(je1) || json_read_value(je2))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (je1->value_type == JSON_VALUE_OBJECT &&
|
|
|
|
je2->value_type == JSON_VALUE_OBJECT)
|
|
|
|
{
|
|
|
|
json_engine_t sav_je1= *je1;
|
|
|
|
json_engine_t sav_je2= *je2;
|
|
|
|
|
|
|
|
int first_key= 1;
|
|
|
|
json_string_t key_name;
|
|
|
|
size_t sav_len;
|
|
|
|
bool mrg_empty;
|
2019-05-19 20:55:37 +02:00
|
|
|
|
2019-05-17 11:53:58 +04:00
|
|
|
*empty_result= FALSE;
|
|
|
|
json_string_set_cs(&key_name, je1->s.cs);
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('{'))
|
2019-05-17 11:53:58 +04:00
|
|
|
return 3;
|
|
|
|
while (json_scan_next(je1) == 0 &&
|
|
|
|
je1->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
const uchar *key_start, *key_end;
|
|
|
|
/* Loop through the Json_1 keys and compare with the Json_2 keys. */
|
|
|
|
DBUG_ASSERT(je1->state == JST_KEY);
|
|
|
|
key_start= je1->s.c_str;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
key_end= je1->s.c_str;
|
|
|
|
} while (json_read_keyname_chr(je1) == 0);
|
|
|
|
|
|
|
|
if (je1->s.error)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
sav_len= str->length();
|
|
|
|
|
|
|
|
if (!first_key)
|
|
|
|
{
|
|
|
|
if (str->append(", ", 2))
|
|
|
|
return 3;
|
|
|
|
*je2= sav_je2;
|
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('"') ||
|
2019-05-17 11:53:58 +04:00
|
|
|
append_simple(str, key_start, key_end - key_start) ||
|
|
|
|
str->append("\":", 2))
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
while (json_scan_next(je2) == 0 &&
|
|
|
|
je2->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
int ires;
|
|
|
|
DBUG_ASSERT(je2->state == JST_KEY);
|
|
|
|
json_string_set_str(&key_name, key_start, key_end);
|
|
|
|
if (!json_key_matches(je2, &key_name))
|
|
|
|
{
|
|
|
|
if (je2->s.error || json_skip_key(je2))
|
|
|
|
return 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
/* Json_2 has same key as Json_1. Merge them. */
|
|
|
|
if ((ires= do_merge_patch(str, je1, je2, &mrg_empty)))
|
|
|
|
return ires;
|
|
|
|
|
|
|
|
if (mrg_empty)
|
|
|
|
str->length(sav_len);
|
|
|
|
else
|
|
|
|
first_key= 0;
|
|
|
|
|
|
|
|
goto merged_j1;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (je2->s.error)
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
key_start= je1->s.c_str;
|
|
|
|
/* Just append the Json_1 key value. */
|
|
|
|
if (json_skip_key(je1))
|
|
|
|
return 1;
|
|
|
|
if (append_simple(str, key_start, je1->s.c_str - key_start))
|
|
|
|
return 3;
|
|
|
|
first_key= 0;
|
|
|
|
|
|
|
|
merged_j1:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
*je2= sav_je2;
|
|
|
|
/*
|
|
|
|
Now loop through the Json_2 keys.
|
|
|
|
Skip if there is same key in Json_1
|
|
|
|
*/
|
|
|
|
while (json_scan_next(je2) == 0 &&
|
|
|
|
je2->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
const uchar *key_start, *key_end;
|
|
|
|
DBUG_ASSERT(je2->state == JST_KEY);
|
|
|
|
key_start= je2->s.c_str;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
key_end= je2->s.c_str;
|
|
|
|
} while (json_read_keyname_chr(je2) == 0);
|
|
|
|
|
|
|
|
if (je2->s.error)
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
*je1= sav_je1;
|
|
|
|
while (json_scan_next(je1) == 0 &&
|
|
|
|
je1->state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(je1->state == JST_KEY);
|
|
|
|
json_string_set_str(&key_name, key_start, key_end);
|
|
|
|
if (!json_key_matches(je1, &key_name))
|
|
|
|
{
|
|
|
|
if (je1->s.error || json_skip_key(je1))
|
|
|
|
return 2;
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
if (json_skip_key(je2) ||
|
|
|
|
json_skip_level(je1))
|
|
|
|
return 1;
|
|
|
|
goto continue_j2;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (je1->s.error)
|
|
|
|
return 2;
|
|
|
|
|
|
|
|
|
|
|
|
sav_len= str->length();
|
|
|
|
|
|
|
|
if (!first_key && str->append(", ", 2))
|
|
|
|
return 3;
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('"') ||
|
2019-05-17 11:53:58 +04:00
|
|
|
append_simple(str, key_start, key_end - key_start) ||
|
|
|
|
str->append("\":", 2))
|
|
|
|
return 3;
|
|
|
|
|
|
|
|
if (json_read_value(je2))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
if (je2->value_type == JSON_VALUE_NULL)
|
|
|
|
str->length(sav_len);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (copy_value_patch(str, je2))
|
|
|
|
return 1;
|
|
|
|
first_key= 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
continue_j2:
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('}'))
|
2019-05-17 11:53:58 +04:00
|
|
|
return 3;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!json_value_scalar(je1) && json_skip_level(je1))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
*empty_result= je2->value_type == JSON_VALUE_NULL;
|
|
|
|
if (!(*empty_result) && copy_value_patch(str, je2))
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_merge_patch::val_str(String *str)
|
|
|
|
{
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2019-05-17 11:53:58 +04:00
|
|
|
json_engine_t je1, je2;
|
|
|
|
String *js1= args[0]->val_json(&tmp_js1), *js2=NULL;
|
|
|
|
uint n_arg;
|
|
|
|
bool empty_result, merge_to_null;
|
2021-07-02 19:13:26 +02:00
|
|
|
THD *thd= current_thd;
|
2019-05-17 11:53:58 +04:00
|
|
|
|
2020-10-22 15:51:14 +04:00
|
|
|
/* To report errors properly if some JSON is invalid. */
|
|
|
|
je1.s.error= je2.s.error= 0;
|
2019-05-17 11:53:58 +04:00
|
|
|
merge_to_null= args[0]->null_value;
|
|
|
|
|
|
|
|
for (n_arg=1; n_arg < arg_count; n_arg++)
|
|
|
|
{
|
|
|
|
js2= args[n_arg]->val_json(&tmp_js2);
|
|
|
|
if (args[n_arg]->null_value)
|
|
|
|
{
|
|
|
|
merge_to_null= true;
|
|
|
|
goto cont_point;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_scan_start(&je2, js2->charset(),(const uchar *) js2->ptr(),
|
|
|
|
(const uchar *) js2->ptr() + js2->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je2.killed_ptr= (uchar*)&thd->killed;
|
2019-05-17 11:53:58 +04:00
|
|
|
|
|
|
|
if (merge_to_null)
|
|
|
|
{
|
|
|
|
if (json_read_value(&je2))
|
|
|
|
goto error_return;
|
|
|
|
if (je2.value_type == JSON_VALUE_OBJECT)
|
|
|
|
goto cont_point;
|
2022-07-28 15:29:53 +08:00
|
|
|
|
2019-05-17 11:53:58 +04:00
|
|
|
merge_to_null= false;
|
|
|
|
str->set(js2->ptr(), js2->length(), js2->charset());
|
|
|
|
goto cont_point;
|
|
|
|
}
|
|
|
|
|
|
|
|
str->set_charset(js1->charset());
|
|
|
|
str->length(0);
|
|
|
|
|
|
|
|
|
|
|
|
json_scan_start(&je1, js1->charset(),(const uchar *) js1->ptr(),
|
|
|
|
(const uchar *) js1->ptr() + js1->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je1.killed_ptr= (uchar*)&thd->killed;
|
2019-05-17 11:53:58 +04:00
|
|
|
|
|
|
|
if (do_merge_patch(str, &je1, &je2, &empty_result))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
if (empty_result)
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append(STRING_WITH_LEN("null"));
|
2019-05-17 11:53:58 +04:00
|
|
|
|
|
|
|
cont_point:
|
|
|
|
{
|
|
|
|
/* Swap str and js1. */
|
|
|
|
if (str == &tmp_js1)
|
|
|
|
{
|
|
|
|
str= js1;
|
|
|
|
js1= &tmp_js1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
js1= str;
|
|
|
|
str= &tmp_js1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (merge_to_null)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
json_scan_start(&je1, js1->charset(),(const uchar *) js1->ptr(),
|
|
|
|
(const uchar *) js1->ptr() + js1->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je1.killed_ptr= (uchar*)&thd->killed;
|
2019-05-17 11:53:58 +04:00
|
|
|
if (json_nice(&je1, str, Item_func_json_format::LOOSE))
|
2017-02-06 06:47:48 +04:00
|
|
|
goto error_return;
|
|
|
|
|
2016-12-06 01:32:13 +04:00
|
|
|
null_value= 0;
|
2017-02-06 06:47:48 +04:00
|
|
|
return str;
|
2016-12-06 01:32:13 +04:00
|
|
|
|
|
|
|
error_return:
|
2016-12-13 12:39:48 +04:00
|
|
|
if (je1.s.error)
|
|
|
|
report_json_error(js1, &je1, 0);
|
|
|
|
if (je2.s.error)
|
|
|
|
report_json_error(js2, &je2, n_arg);
|
2021-07-02 19:13:26 +02:00
|
|
|
thd->check_killed(); // to get the error message right
|
2016-12-13 12:39:48 +04:00
|
|
|
null_return:
|
2016-10-19 14:10:03 +04:00
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_length::fix_length_and_dec(THD *thd)
|
2016-12-24 11:40:31 +04:00
|
|
|
{
|
|
|
|
if (arg_count > 1)
|
|
|
|
path.set_constant_flag(args[1]->const_item());
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2017-05-20 16:29:11 +04:00
|
|
|
max_length= 10;
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-12-24 11:40:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-10-19 14:10:03 +04:00
|
|
|
longlong Item_func_json_length::val_int()
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-10-19 14:10:03 +04:00
|
|
|
json_engine_t je;
|
|
|
|
uint length= 0;
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2017-10-05 23:46:25 +04:00
|
|
|
int err;
|
2016-10-19 14:10:03 +04:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
2016-12-24 11:40:31 +04:00
|
|
|
if (arg_count > 1)
|
|
|
|
{
|
|
|
|
/* Path specified - let's apply it. */
|
|
|
|
if (!path.parsed)
|
|
|
|
{
|
|
|
|
String *s_p= args[1]->val_str(&tmp_path);
|
|
|
|
if (s_p &&
|
2017-01-24 17:34:44 +04:00
|
|
|
path_setup_nwc(&path.p, s_p->charset(), (const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
2016-12-24 11:40:31 +04:00
|
|
|
{
|
2017-01-24 17:34:44 +04:00
|
|
|
report_path_error(s_p, &path.p, 1);
|
2016-12-24 11:40:31 +04:00
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
path.parsed= path.constant;
|
|
|
|
}
|
|
|
|
if (args[1]->null_value)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
path.cur_step= path.p.steps;
|
|
|
|
if (json_find_path(&je, &path.p, &path.cur_step, array_counters))
|
|
|
|
{
|
|
|
|
if (je.s.error)
|
|
|
|
goto err_return;
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if (json_read_value(&je))
|
|
|
|
goto err_return;
|
|
|
|
|
|
|
|
if (json_value_scalar(&je))
|
|
|
|
return 1;
|
|
|
|
|
2017-10-05 23:46:25 +04:00
|
|
|
while (!(err= json_scan_next(&je)) &&
|
2016-12-24 11:40:31 +04:00
|
|
|
je.state != JST_OBJ_END && je.state != JST_ARRAY_END)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2016-12-24 11:40:31 +04:00
|
|
|
switch (je.state)
|
|
|
|
{
|
|
|
|
case JST_VALUE:
|
|
|
|
case JST_KEY:
|
2016-10-19 14:10:03 +04:00
|
|
|
length++;
|
2016-12-24 11:40:31 +04:00
|
|
|
break;
|
|
|
|
case JST_OBJ_START:
|
|
|
|
case JST_ARRAY_START:
|
|
|
|
if (json_skip_level(&je))
|
|
|
|
goto err_return;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
};
|
|
|
|
}
|
2016-10-19 14:10:03 +04:00
|
|
|
|
2017-10-05 23:46:25 +04:00
|
|
|
if (!err)
|
|
|
|
{
|
|
|
|
/* Parse to the end of the JSON just to check it's valid. */
|
|
|
|
while (json_scan_next(&je) == 0) {}
|
|
|
|
}
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (likely(!je.s.error))
|
2016-12-24 11:40:31 +04:00
|
|
|
return length;
|
2016-12-13 12:39:48 +04:00
|
|
|
|
2016-12-24 11:40:31 +04:00
|
|
|
err_return:
|
2016-12-13 12:39:48 +04:00
|
|
|
report_json_error(js, &je, 0);
|
2016-12-24 11:40:31 +04:00
|
|
|
null_return:
|
2016-12-13 12:39:48 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
longlong Item_func_json_depth::val_int()
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-10-19 14:10:03 +04:00
|
|
|
json_engine_t je;
|
2016-12-05 07:17:54 +04:00
|
|
|
uint depth= 0, c_depth= 0;
|
2016-10-19 14:10:03 +04:00
|
|
|
bool inc_depth= TRUE;
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
switch (je.state)
|
|
|
|
{
|
|
|
|
case JST_VALUE:
|
2016-12-16 14:06:12 +04:00
|
|
|
case JST_KEY:
|
2016-10-19 14:10:03 +04:00
|
|
|
if (inc_depth)
|
|
|
|
{
|
2016-12-05 07:17:54 +04:00
|
|
|
c_depth++;
|
2016-10-19 14:10:03 +04:00
|
|
|
inc_depth= FALSE;
|
2016-12-05 07:17:54 +04:00
|
|
|
if (c_depth > depth)
|
|
|
|
depth= c_depth;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
break;
|
|
|
|
case JST_OBJ_START:
|
|
|
|
case JST_ARRAY_START:
|
|
|
|
inc_depth= TRUE;
|
|
|
|
break;
|
2016-12-05 07:17:54 +04:00
|
|
|
case JST_OBJ_END:
|
|
|
|
case JST_ARRAY_END:
|
|
|
|
if (!inc_depth)
|
|
|
|
c_depth--;
|
|
|
|
inc_depth= FALSE;
|
|
|
|
break;
|
2016-10-19 14:10:03 +04:00
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
} while (json_scan_next(&je) == 0);
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (likely(!je.s.error))
|
2016-12-13 12:39:48 +04:00
|
|
|
return depth;
|
|
|
|
|
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_type::fix_length_and_dec(THD *thd)
|
2016-10-19 14:10:03 +04:00
|
|
|
{
|
2019-06-28 09:05:12 +04:00
|
|
|
collation.set(&my_charset_utf8mb3_general_ci);
|
2022-06-25 22:03:08 +08:00
|
|
|
max_length= 12 * collation.collation->mbmaxlen;
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-10-19 14:10:03 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_type::val_str(String *str)
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-10-19 14:10:03 +04:00
|
|
|
json_engine_t je;
|
|
|
|
const char *type;
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
|
|
|
if (json_read_value(&je))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
switch (je.value_type)
|
|
|
|
{
|
|
|
|
case JSON_VALUE_OBJECT:
|
|
|
|
type= "OBJECT";
|
|
|
|
break;
|
|
|
|
case JSON_VALUE_ARRAY:
|
|
|
|
type= "ARRAY";
|
|
|
|
break;
|
|
|
|
case JSON_VALUE_STRING:
|
|
|
|
type= "STRING";
|
|
|
|
break;
|
|
|
|
case JSON_VALUE_NUMBER:
|
2016-12-05 01:01:09 +04:00
|
|
|
type= (je.num_flags & JSON_NUM_FRAC_PART) ? "DOUBLE" : "INTEGER";
|
2016-10-19 14:10:03 +04:00
|
|
|
break;
|
|
|
|
case JSON_VALUE_TRUE:
|
|
|
|
case JSON_VALUE_FALSE:
|
|
|
|
type= "BOOLEAN";
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
type= "NULL";
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2019-06-28 09:05:12 +04:00
|
|
|
str->set(type, strlen(type), &my_charset_utf8mb3_general_ci);
|
2016-10-19 14:10:03 +04:00
|
|
|
return str;
|
|
|
|
|
|
|
|
error:
|
2016-12-13 12:39:48 +04:00
|
|
|
report_json_error(js, &je, 0);
|
2016-10-19 14:10:03 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_insert::fix_length_and_dec(THD *thd)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
uint n_arg;
|
|
|
|
ulonglong char_length;
|
|
|
|
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
char_length= args[0]->max_char_length();
|
|
|
|
|
|
|
|
for (n_arg= 1; n_arg < arg_count; n_arg+= 2)
|
|
|
|
{
|
2016-11-16 12:47:46 +04:00
|
|
|
paths[n_arg/2].set_constant_flag(args[n_arg]->const_item());
|
2022-08-13 12:49:48 +08:00
|
|
|
char_length+=
|
|
|
|
static_cast<ulonglong>(args[n_arg+1]->max_char_length()) + 4;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
fix_char_length_ulonglong(char_length);
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_insert::val_str(String *str)
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-11-15 17:04:31 +04:00
|
|
|
uint n_arg, n_path;
|
|
|
|
json_string_t key_name;
|
2021-07-02 19:13:26 +02:00
|
|
|
THD *thd= current_thd;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
2018-08-06 13:37:09 +04:00
|
|
|
str->set_charset(collation.collation);
|
|
|
|
tmp_js.set_charset(collation.collation);
|
|
|
|
json_string_set_cs(&key_name, collation.collation);
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
for (n_arg=1, n_path=0; n_arg < arg_count; n_arg+=2, n_path++)
|
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-11-15 17:04:31 +04:00
|
|
|
json_path_with_flags *c_path= paths + n_path;
|
|
|
|
const char *v_to;
|
2021-11-22 22:59:30 +05:30
|
|
|
json_path_step_t *lp;
|
|
|
|
int corrected_n_item;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (!c_path->parsed)
|
|
|
|
{
|
|
|
|
String *s_p= args[n_arg]->val_str(tmp_paths+n_path);
|
|
|
|
if (s_p)
|
|
|
|
{
|
2016-12-13 12:39:48 +04:00
|
|
|
if (path_setup_nwc(&c_path->p,s_p->charset(),
|
|
|
|
(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
|
|
|
goto return_null;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
/* We search to the last step. */
|
|
|
|
c_path->p.last_step--;
|
|
|
|
}
|
|
|
|
c_path->parsed= c_path->constant;
|
|
|
|
}
|
|
|
|
if (args[n_arg]->null_value)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto return_null;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2017-01-25 00:13:15 +04:00
|
|
|
if (c_path->p.last_step < c_path->p.steps)
|
|
|
|
goto v_found;
|
|
|
|
|
2016-11-15 17:04:31 +04:00
|
|
|
c_path->cur_step= c_path->p.steps;
|
|
|
|
|
|
|
|
if (c_path->p.last_step >= c_path->p.steps &&
|
|
|
|
json_find_path(&je, &c_path->p, &c_path->cur_step, array_counters))
|
|
|
|
{
|
|
|
|
if (je.s.error)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2017-02-06 06:47:48 +04:00
|
|
|
continue;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (json_read_value(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
lp= c_path->p.last_step+1;
|
2016-12-09 12:26:32 +04:00
|
|
|
if (lp->type & JSON_PATH_ARRAY)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
int n_item= 0;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (je.value_type != JSON_VALUE_ARRAY)
|
|
|
|
{
|
|
|
|
const uchar *v_from= je.value_begin;
|
2017-01-26 16:35:05 +04:00
|
|
|
int do_array_autowrap;
|
|
|
|
|
|
|
|
if (mode_insert)
|
2017-02-06 06:47:48 +04:00
|
|
|
{
|
|
|
|
if (mode_replace)
|
|
|
|
do_array_autowrap= lp->n_item > 0;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (lp->n_item == 0)
|
|
|
|
continue;
|
|
|
|
do_array_autowrap= 1;
|
|
|
|
}
|
|
|
|
}
|
2017-01-26 16:35:05 +04:00
|
|
|
else
|
|
|
|
{
|
|
|
|
if (lp->n_item)
|
|
|
|
continue;
|
|
|
|
do_array_autowrap= 0;
|
|
|
|
}
|
|
|
|
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
str->length(0);
|
|
|
|
/* Wrap the value as an array. */
|
|
|
|
if (append_simple(str, js->ptr(), (const char *) v_from - js->ptr()) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
(do_array_autowrap && str->append('[')))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error; /* Out of memory. */
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (je.value_type == JSON_VALUE_OBJECT)
|
|
|
|
{
|
|
|
|
if (json_skip_level(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
2017-01-26 16:35:05 +04:00
|
|
|
if ((do_array_autowrap &&
|
|
|
|
(append_simple(str, v_from, je.s.c_str - v_from) ||
|
|
|
|
str->append(", ", 2))) ||
|
2016-11-15 17:04:31 +04:00
|
|
|
append_json_value(str, args[n_arg+1], &tmp_val) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
(do_array_autowrap && str->append(']')) ||
|
2016-11-15 17:04:31 +04:00
|
|
|
append_simple(str, je.s.c_str, js->end()-(const char *) je.s.c_str))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error; /* Out of memory. */
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
goto continue_point;
|
|
|
|
}
|
2021-11-22 22:59:30 +05:30
|
|
|
corrected_n_item= lp->n_item;
|
|
|
|
if (corrected_n_item < 0)
|
|
|
|
{
|
|
|
|
int array_size;
|
|
|
|
if (json_skip_array_and_count(&je, &array_size))
|
|
|
|
goto js_error;
|
|
|
|
corrected_n_item+= array_size;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
while (json_scan_next(&je) == 0 && je.state != JST_ARRAY_END)
|
|
|
|
{
|
|
|
|
switch (je.state)
|
|
|
|
{
|
|
|
|
case JST_VALUE:
|
2021-11-22 22:59:30 +05:30
|
|
|
if (n_item == corrected_n_item)
|
2016-11-15 17:04:31 +04:00
|
|
|
goto v_found;
|
|
|
|
n_item++;
|
|
|
|
if (json_skip_array_item(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (!mode_insert)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
v_to= (const char *) (je.s.c_str - je.sav_c_len);
|
|
|
|
str->length(0);
|
|
|
|
if (append_simple(str, js->ptr(), v_to - js->ptr()) ||
|
2017-01-26 16:35:05 +04:00
|
|
|
(n_item > 0 && str->append(", ", 2)) ||
|
2016-11-15 17:04:31 +04:00
|
|
|
append_json_value(str, args[n_arg+1], &tmp_val) ||
|
|
|
|
append_simple(str, v_to, js->end() - v_to))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error; /* Out of memory. */
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
else /*JSON_PATH_KEY*/
|
|
|
|
{
|
2017-09-12 14:40:18 +04:00
|
|
|
uint n_key= 0;
|
|
|
|
|
2016-11-15 17:04:31 +04:00
|
|
|
if (je.value_type != JSON_VALUE_OBJECT)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
while (json_scan_next(&je) == 0 && je.state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
switch (je.state)
|
|
|
|
{
|
|
|
|
case JST_KEY:
|
|
|
|
json_string_set_str(&key_name, lp->key, lp->key_end);
|
|
|
|
if (json_key_matches(&je, &key_name))
|
|
|
|
goto v_found;
|
2017-09-12 14:40:18 +04:00
|
|
|
n_key++;
|
2016-11-15 17:04:31 +04:00
|
|
|
if (json_skip_key(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (!mode_insert)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
v_to= (const char *) (je.s.c_str - je.sav_c_len);
|
|
|
|
str->length(0);
|
|
|
|
if (append_simple(str, js->ptr(), v_to - js->ptr()) ||
|
2017-09-12 14:40:18 +04:00
|
|
|
(n_key > 0 && str->append(", ", 2)) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append('"') ||
|
2016-11-15 17:04:31 +04:00
|
|
|
append_simple(str, lp->key, lp->key_end - lp->key) ||
|
|
|
|
str->append("\":", 2) ||
|
|
|
|
append_json_value(str, args[n_arg+1], &tmp_val) ||
|
|
|
|
append_simple(str, v_to, js->end() - v_to))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error; /* Out of memory. */
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
goto continue_point;
|
|
|
|
|
|
|
|
v_found:
|
|
|
|
|
|
|
|
if (!mode_replace)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
if (json_read_value(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
v_to= (const char *) je.value_begin;
|
|
|
|
str->length(0);
|
|
|
|
if (!json_value_scalar(&je))
|
|
|
|
{
|
|
|
|
if (json_skip_level(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (append_simple(str, js->ptr(), v_to - js->ptr()) ||
|
|
|
|
append_json_value(str, args[n_arg+1], &tmp_val) ||
|
|
|
|
append_simple(str, je.s.c_str, js->end()-(const char *) je.s.c_str))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error; /* Out of memory. */
|
2016-11-15 17:04:31 +04:00
|
|
|
continue_point:
|
|
|
|
{
|
2016-12-05 08:43:15 +04:00
|
|
|
/* Swap str and js. */
|
|
|
|
if (str == &tmp_js)
|
|
|
|
{
|
|
|
|
str= js;
|
|
|
|
js= &tmp_js;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
js= str;
|
|
|
|
str= &tmp_js;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2017-02-06 06:47:48 +04:00
|
|
|
if (json_nice(&je, str, Item_func_json_format::LOOSE))
|
|
|
|
goto js_error;
|
|
|
|
|
|
|
|
return str;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
js_error:
|
|
|
|
report_json_error(js, &je, 0);
|
2021-07-02 19:13:26 +02:00
|
|
|
thd->check_killed(); // to get the error message right
|
2016-12-13 12:39:48 +04:00
|
|
|
return_null:
|
2016-11-15 17:04:31 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_remove::fix_length_and_dec(THD *thd)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
max_length= args[0]->max_length;
|
|
|
|
|
|
|
|
mark_constant_paths(paths, args+1, arg_count-1);
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_remove::val_str(String *str)
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-11-15 17:04:31 +04:00
|
|
|
uint n_arg, n_path;
|
|
|
|
json_string_t key_name;
|
2021-07-02 19:13:26 +02:00
|
|
|
THD *thd= current_thd;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (args[0]->null_value)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
str->set_charset(js->charset());
|
|
|
|
json_string_set_cs(&key_name, js->charset());
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
for (n_arg=1, n_path=0; n_arg < arg_count; n_arg++, n_path++)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-11-15 17:04:31 +04:00
|
|
|
json_path_with_flags *c_path= paths + n_path;
|
2018-04-11 02:20:22 +03:00
|
|
|
const char *rem_start= 0, *rem_end;
|
2021-11-22 22:59:30 +05:30
|
|
|
json_path_step_t *lp;
|
|
|
|
int n_item= 0;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (!c_path->parsed)
|
|
|
|
{
|
|
|
|
String *s_p= args[n_arg]->val_str(tmp_paths+n_path);
|
|
|
|
if (s_p)
|
|
|
|
{
|
2016-12-13 12:39:48 +04:00
|
|
|
if (path_setup_nwc(&c_path->p,s_p->charset(),
|
|
|
|
(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
|
|
|
goto null_return;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
/* We search to the last step. */
|
|
|
|
c_path->p.last_step--;
|
|
|
|
if (c_path->p.last_step < c_path->p.steps)
|
2017-01-24 17:34:44 +04:00
|
|
|
{
|
|
|
|
c_path->p.s.error= TRIVIAL_PATH_NOT_ALLOWED;
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
2016-11-15 17:04:31 +04:00
|
|
|
goto null_return;
|
2017-01-24 17:34:44 +04:00
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
c_path->parsed= c_path->constant;
|
|
|
|
}
|
|
|
|
if (args[n_arg]->null_value)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto null_return;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
c_path->cur_step= c_path->p.steps;
|
|
|
|
|
|
|
|
if (json_find_path(&je, &c_path->p, &c_path->cur_step, array_counters))
|
|
|
|
{
|
|
|
|
if (je.s.error)
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
if (json_read_value(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
lp= c_path->p.last_step+1;
|
2021-11-22 22:59:30 +05:30
|
|
|
|
2016-12-09 12:26:32 +04:00
|
|
|
if (lp->type & JSON_PATH_ARRAY)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
int corrected_n_item;
|
2016-11-15 17:04:31 +04:00
|
|
|
if (je.value_type != JSON_VALUE_ARRAY)
|
|
|
|
continue;
|
|
|
|
|
2021-11-22 22:59:30 +05:30
|
|
|
corrected_n_item= lp->n_item;
|
|
|
|
if (corrected_n_item < 0)
|
|
|
|
{
|
|
|
|
int array_size;
|
|
|
|
if (json_skip_array_and_count(&je, &array_size))
|
|
|
|
goto js_error;
|
|
|
|
corrected_n_item+= array_size;
|
|
|
|
}
|
|
|
|
|
2016-11-15 17:04:31 +04:00
|
|
|
while (json_scan_next(&je) == 0 && je.state != JST_ARRAY_END)
|
|
|
|
{
|
|
|
|
switch (je.state)
|
|
|
|
{
|
|
|
|
case JST_VALUE:
|
2021-11-22 22:59:30 +05:30
|
|
|
if (n_item == corrected_n_item)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
rem_start= (const char *) (je.s.c_str -
|
|
|
|
(n_item ? je.sav_c_len : 0));
|
|
|
|
goto v_found;
|
|
|
|
}
|
|
|
|
n_item++;
|
|
|
|
if (json_skip_array_item(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
else /*JSON_PATH_KEY*/
|
|
|
|
{
|
|
|
|
if (je.value_type != JSON_VALUE_OBJECT)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
while (json_scan_next(&je) == 0 && je.state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
switch (je.state)
|
|
|
|
{
|
|
|
|
case JST_KEY:
|
|
|
|
if (n_item == 0)
|
|
|
|
rem_start= (const char *) (je.s.c_str - je.sav_c_len);
|
|
|
|
json_string_set_str(&key_name, lp->key, lp->key_end);
|
|
|
|
if (json_key_matches(&je, &key_name))
|
|
|
|
goto v_found;
|
|
|
|
|
|
|
|
if (json_skip_key(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
rem_start= (const char *) je.s.c_str;
|
|
|
|
n_item++;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
|
|
|
|
v_found:
|
|
|
|
|
|
|
|
if (json_skip_key(&je) || json_scan_next(&je))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2016-12-05 07:42:00 +04:00
|
|
|
rem_end= (je.state == JST_VALUE && n_item == 0) ?
|
2016-11-15 17:04:31 +04:00
|
|
|
(const char *) je.s.c_str : (const char *) (je.s.c_str - je.sav_c_len);
|
|
|
|
|
|
|
|
str->length(0);
|
|
|
|
|
|
|
|
if (append_simple(str, js->ptr(), rem_start - js->ptr()) ||
|
2017-03-20 15:18:06 +04:00
|
|
|
(je.state == JST_KEY && n_item > 0 && str->append(",", 1)) ||
|
2016-11-15 17:04:31 +04:00
|
|
|
append_simple(str, rem_end, js->end() - rem_end))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error; /* Out of memory. */
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
{
|
2016-12-05 08:43:15 +04:00
|
|
|
/* Swap str and js. */
|
|
|
|
if (str == &tmp_js)
|
|
|
|
{
|
|
|
|
str= js;
|
|
|
|
js= &tmp_js;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
js= str;
|
|
|
|
str= &tmp_js;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2017-02-06 06:47:48 +04:00
|
|
|
if (json_nice(&je, str, Item_func_json_format::LOOSE))
|
|
|
|
goto js_error;
|
|
|
|
|
2017-03-20 15:18:06 +04:00
|
|
|
null_value= 0;
|
2017-02-06 06:47:48 +04:00
|
|
|
return str;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
js_error:
|
2021-07-02 19:13:26 +02:00
|
|
|
thd->check_killed(); // to get the error message right
|
2016-12-13 12:39:48 +04:00
|
|
|
report_json_error(js, &je, 0);
|
2016-11-15 17:04:31 +04:00
|
|
|
null_return:
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_keys::fix_length_and_dec(THD *thd)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
max_length= args[0]->max_length;
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2016-11-15 17:04:31 +04:00
|
|
|
if (arg_count > 1)
|
|
|
|
path.set_constant_flag(args[1]->const_item());
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-08-08 15:40:11 +04:00
|
|
|
/*
|
|
|
|
That function is for Item_func_json_keys::val_str exclusively.
|
|
|
|
It utilizes the fact the resulting string is in specific format:
|
|
|
|
["key1", "key2"...]
|
|
|
|
*/
|
|
|
|
static int check_key_in_list(String *res,
|
|
|
|
const uchar *key, int key_len)
|
|
|
|
{
|
|
|
|
const uchar *c= (const uchar *) res->ptr() + 2; /* beginning '["' */
|
|
|
|
const uchar *end= (const uchar *) res->end() - 1; /* ending '"' */
|
|
|
|
|
|
|
|
while (c < end)
|
|
|
|
{
|
|
|
|
int n_char;
|
|
|
|
for (n_char=0; c[n_char] != '"' && n_char < key_len; n_char++)
|
|
|
|
{
|
|
|
|
if (c[n_char] != key[n_char])
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
if (c[n_char] == '"')
|
|
|
|
{
|
|
|
|
if (n_char == key_len)
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
while (c[n_char] != '"')
|
|
|
|
n_char++;
|
|
|
|
}
|
|
|
|
c+= n_char + 4; /* skip ', "' */
|
|
|
|
}
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-11-15 17:04:31 +04:00
|
|
|
String *Item_func_json_keys::val_str(String *str)
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2016-11-15 17:04:31 +04:00
|
|
|
uint n_keys= 0;
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_counters[JSON_DEPTH_LIMIT];
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if ((args[0]->null_value))
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
|
|
|
if (arg_count < 2)
|
|
|
|
goto skip_search;
|
|
|
|
|
|
|
|
if (!path.parsed)
|
|
|
|
{
|
|
|
|
String *s_p= args[1]->val_str(&tmp_path);
|
|
|
|
if (s_p &&
|
2016-12-13 12:39:48 +04:00
|
|
|
path_setup_nwc(&path.p, s_p->charset(), (const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &path.p, 1);
|
|
|
|
goto null_return;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
path.parsed= path.constant;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (args[1]->null_value)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
path.cur_step= path.p.steps;
|
|
|
|
|
|
|
|
if (json_find_path(&je, &path.p, &path.cur_step, array_counters))
|
|
|
|
{
|
|
|
|
if (je.s.error)
|
|
|
|
goto err_return;
|
|
|
|
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
|
|
|
|
skip_search:
|
|
|
|
if (json_read_value(&je))
|
|
|
|
goto err_return;
|
|
|
|
|
|
|
|
if (je.value_type != JSON_VALUE_OBJECT)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
str->length(0);
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('['))
|
2016-11-15 17:04:31 +04:00
|
|
|
goto err_return; /* Out of memory. */
|
|
|
|
/* Parse the OBJECT collecting the keys. */
|
|
|
|
while (json_scan_next(&je) == 0 && je.state != JST_OBJ_END)
|
|
|
|
{
|
|
|
|
const uchar *key_start, *key_end;
|
2017-08-08 15:40:11 +04:00
|
|
|
int key_len;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
switch (je.state)
|
|
|
|
{
|
|
|
|
case JST_KEY:
|
|
|
|
key_start= je.s.c_str;
|
2017-02-09 17:38:53 +04:00
|
|
|
do
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
key_end= je.s.c_str;
|
2017-02-09 17:38:53 +04:00
|
|
|
} while (json_read_keyname_chr(&je) == 0);
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error))
|
2017-08-08 15:40:11 +04:00
|
|
|
goto err_return;
|
2017-09-28 10:38:02 +00:00
|
|
|
key_len= (int)(key_end - key_start);
|
2017-08-08 15:40:11 +04:00
|
|
|
|
|
|
|
if (!check_key_in_list(str, key_start, key_len))
|
|
|
|
{
|
|
|
|
if ((n_keys > 0 && str->append(", ", 2)) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append('"') ||
|
2017-08-08 15:40:11 +04:00
|
|
|
append_simple(str, key_start, key_len) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append('"'))
|
2016-11-15 17:04:31 +04:00
|
|
|
goto err_return;
|
2017-08-08 15:40:11 +04:00
|
|
|
n_keys++;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
break;
|
|
|
|
case JST_OBJ_START:
|
|
|
|
case JST_ARRAY_START:
|
|
|
|
if (json_skip_level(&je))
|
|
|
|
break;
|
|
|
|
break;
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (unlikely(je.s.error || str->append(']')))
|
2016-11-15 17:04:31 +04:00
|
|
|
goto err_return;
|
|
|
|
|
|
|
|
null_value= 0;
|
|
|
|
return str;
|
|
|
|
|
|
|
|
err_return:
|
2016-12-13 12:39:48 +04:00
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
null_return:
|
2016-11-15 17:04:31 +04:00
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_func_json_search::fix_fields(THD *thd, Item **ref)
|
|
|
|
{
|
|
|
|
if (Item_json_str_multipath::fix_fields(thd, ref))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
if (arg_count < 4)
|
2017-02-02 18:56:15 +04:00
|
|
|
{
|
|
|
|
escape= '\\';
|
2016-11-15 17:04:31 +04:00
|
|
|
return FALSE;
|
2017-02-02 18:56:15 +04:00
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
return fix_escape_item(thd, args[3], &tmp_js, true,
|
|
|
|
args[0]->collation.collation, &escape);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2016-12-27 01:30:20 +02:00
|
|
|
static const uint SQR_MAX_BLOB_WIDTH= (uint) sqrt(MAX_BLOB_WIDTH);
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_search::fix_length_and_dec(THD *thd)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
|
|
|
|
/*
|
|
|
|
It's rather difficult to estimate the length of the result.
|
2021-03-11 09:50:29 +08:00
|
|
|
I believe arglen^2 is the reasonable upper limit.
|
2016-11-15 17:04:31 +04:00
|
|
|
*/
|
|
|
|
if (args[0]->max_length > SQR_MAX_BLOB_WIDTH)
|
|
|
|
max_length= MAX_BLOB_WIDTH;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
max_length= args[0]->max_length;
|
|
|
|
max_length*= max_length;
|
|
|
|
}
|
|
|
|
|
|
|
|
ooa_constant= args[1]->const_item();
|
|
|
|
ooa_parsed= FALSE;
|
|
|
|
|
|
|
|
if (arg_count > 4)
|
|
|
|
mark_constant_paths(paths, args+4, arg_count-4);
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Item_func_json_search::compare_json_value_wild(json_engine_t *je,
|
|
|
|
const String *cmp_str)
|
|
|
|
{
|
2017-03-14 15:25:02 +04:00
|
|
|
if (je->value_type != JSON_VALUE_STRING || !je->value_escaped)
|
2020-01-26 20:27:13 +04:00
|
|
|
return collation.collation->wildcmp(
|
2017-03-14 15:25:02 +04:00
|
|
|
(const char *) je->value, (const char *) (je->value + je->value_len),
|
|
|
|
cmp_str->ptr(), cmp_str->end(), escape, wild_one, wild_many) ? 0 : 1;
|
|
|
|
|
|
|
|
{
|
|
|
|
int esc_len;
|
|
|
|
if (esc_value.alloced_length() < (uint) je->value_len &&
|
|
|
|
esc_value.alloc((je->value_len / 1024 + 1) * 1024))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
esc_len= json_unescape(je->s.cs, je->value, je->value + je->value_len,
|
|
|
|
je->s.cs, (uchar *) esc_value.ptr(),
|
|
|
|
(uchar *) (esc_value.ptr() +
|
|
|
|
esc_value.alloced_length()));
|
|
|
|
if (esc_len <= 0)
|
|
|
|
return 0;
|
|
|
|
|
2020-01-26 20:27:13 +04:00
|
|
|
return collation.collation->wildcmp(
|
2017-03-14 15:25:02 +04:00
|
|
|
esc_value.ptr(), esc_value.ptr() + esc_len,
|
|
|
|
cmp_str->ptr(), cmp_str->end(), escape, wild_one, wild_many) ? 0 : 1;
|
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
static int append_json_path(String *str, const json_path_t *p)
|
|
|
|
{
|
|
|
|
const json_path_step_t *c;
|
|
|
|
|
|
|
|
if (str->append("\"$", 2))
|
|
|
|
return TRUE;
|
|
|
|
|
|
|
|
for (c= p->steps+1; c <= p->last_step; c++)
|
|
|
|
{
|
2016-12-09 12:26:32 +04:00
|
|
|
if (c->type & JSON_PATH_KEY)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
|
|
|
if (str->append(".", 1) ||
|
|
|
|
append_simple(str, c->key, c->key_end-c->key))
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else /*JSON_PATH_ARRAY*/
|
|
|
|
{
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('[') ||
|
2016-11-15 17:04:31 +04:00
|
|
|
str->append_ulonglong(c->n_item) ||
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
str->append(']'))
|
2016-11-15 17:04:31 +04:00
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return str->append('"');
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_search::val_str(String *str)
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2018-09-12 21:57:28 +04:00
|
|
|
String *s_str= args[2]->val_str(&tmp_path);
|
2016-11-15 17:04:31 +04:00
|
|
|
json_engine_t je;
|
|
|
|
json_path_t p, sav_path;
|
|
|
|
uint n_arg;
|
2021-11-22 22:59:30 +05:30
|
|
|
int array_sizes[JSON_DEPTH_LIMIT];
|
|
|
|
uint has_negative_path= 0;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
if (args[0]->null_value || args[2]->null_value)
|
|
|
|
goto null_return;
|
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
if (parse_one_or_all(this, args[1], &ooa_parsed, ooa_constant, &mode_one))
|
2016-11-15 17:04:31 +04:00
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
n_path_found= 0;
|
|
|
|
str->set_charset(js->charset());
|
|
|
|
str->length(0);
|
|
|
|
|
|
|
|
for (n_arg=4; n_arg < arg_count; n_arg++)
|
|
|
|
{
|
|
|
|
json_path_with_flags *c_path= paths + n_arg - 4;
|
2022-04-19 21:43:31 +05:30
|
|
|
c_path->p.types_used= JSON_PATH_KEY_NULL;
|
2016-11-15 17:04:31 +04:00
|
|
|
if (!c_path->parsed)
|
|
|
|
{
|
2017-05-02 14:28:57 +04:00
|
|
|
String *s_p= args[n_arg]->val_str(tmp_paths + (n_arg-4));
|
2022-04-19 21:43:31 +05:30
|
|
|
if (s_p)
|
2016-12-13 12:39:48 +04:00
|
|
|
{
|
2022-04-19 21:43:31 +05:30
|
|
|
if (json_path_setup(&c_path->p,s_p->charset(),(const uchar *) s_p->ptr(),
|
|
|
|
(const uchar *) s_p->ptr() + s_p->length()))
|
|
|
|
{
|
|
|
|
report_path_error(s_p, &c_path->p, n_arg);
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
c_path->parsed= c_path->constant;
|
|
|
|
has_negative_path|= c_path->p.types_used & JSON_PATH_NEGATIVE_INDEX;
|
2016-12-13 12:39:48 +04:00
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
2016-12-03 12:11:06 +04:00
|
|
|
if (args[n_arg]->null_value)
|
|
|
|
goto null_return;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
json_get_path_start(&je, js->charset(),(const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length(), &p);
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
while (json_get_path_next(&je, &p) == 0)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
if (has_negative_path && je.value_type == JSON_VALUE_ARRAY &&
|
|
|
|
json_skip_array_and_count(&je, array_sizes + (p.last_step - p.steps)))
|
|
|
|
goto js_error;
|
|
|
|
|
2017-01-24 17:34:44 +04:00
|
|
|
if (json_value_scalar(&je))
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2021-11-22 22:59:30 +05:30
|
|
|
if ((arg_count < 5 ||
|
|
|
|
path_ok(paths, arg_count - 4, &p, je.value_type, array_sizes)) &&
|
2017-01-24 17:34:44 +04:00
|
|
|
compare_json_value_wild(&je, s_str) != 0)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2017-01-24 17:34:44 +04:00
|
|
|
++n_path_found;
|
|
|
|
if (n_path_found == 1)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
2017-01-24 17:34:44 +04:00
|
|
|
sav_path= p;
|
|
|
|
sav_path.last_step= sav_path.steps + (p.last_step - p.steps);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (n_path_found == 2)
|
2016-11-15 17:04:31 +04:00
|
|
|
{
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append('[') ||
|
2017-01-24 17:34:44 +04:00
|
|
|
append_json_path(str, &sav_path))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
2017-01-24 17:34:44 +04:00
|
|
|
if (str->append(", ", 2) || append_json_path(str, &p))
|
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
2017-01-24 17:34:44 +04:00
|
|
|
if (mode_one)
|
|
|
|
goto end;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
}
|
2017-01-24 17:34:44 +04:00
|
|
|
}
|
2016-11-15 17:04:31 +04:00
|
|
|
|
2018-04-04 12:16:12 +03:00
|
|
|
if (unlikely(je.s.error))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
|
|
|
|
end:
|
|
|
|
if (n_path_found == 0)
|
|
|
|
goto null_return;
|
|
|
|
if (n_path_found == 1)
|
|
|
|
{
|
|
|
|
if (append_json_path(str, &sav_path))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
if (str->append(']'))
|
2016-12-13 12:39:48 +04:00
|
|
|
goto js_error;
|
2016-11-15 17:04:31 +04:00
|
|
|
}
|
|
|
|
|
2016-12-03 12:11:06 +04:00
|
|
|
null_value= 0;
|
2016-11-15 17:04:31 +04:00
|
|
|
return str;
|
|
|
|
|
|
|
|
|
2016-12-13 12:39:48 +04:00
|
|
|
js_error:
|
|
|
|
report_json_error(js, &je, 0);
|
2016-11-15 17:04:31 +04:00
|
|
|
null_return:
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
LEX_CSTRING Item_func_json_format::func_name_cstring() const
|
2017-02-06 06:47:48 +04:00
|
|
|
{
|
|
|
|
switch (fmt)
|
|
|
|
{
|
|
|
|
case COMPACT:
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return { STRING_WITH_LEN("json_compact") };
|
2017-02-06 06:47:48 +04:00
|
|
|
case LOOSE:
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return { STRING_WITH_LEN("json_loose") };
|
2017-02-06 06:47:48 +04:00
|
|
|
case DETAILED:
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return { STRING_WITH_LEN("json_detailed") };
|
2017-02-06 06:47:48 +04:00
|
|
|
default:
|
|
|
|
DBUG_ASSERT(0);
|
|
|
|
};
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
return NULL_clex_str;
|
2017-02-06 06:47:48 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_format::fix_length_and_dec(THD *thd)
|
2017-02-06 06:47:48 +04:00
|
|
|
{
|
|
|
|
decimals= 0;
|
2021-10-11 15:05:44 +04:00
|
|
|
collation.set(args[0]->collation);
|
2017-02-06 06:47:48 +04:00
|
|
|
max_length= args[0]->max_length;
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2018-05-08 15:26:26 +02:00
|
|
|
return FALSE;
|
2017-02-06 06:47:48 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String *Item_func_json_format::val_str(String *str)
|
|
|
|
{
|
2017-02-14 11:11:47 +01:00
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
2017-02-06 06:47:48 +04:00
|
|
|
json_engine_t je;
|
2017-02-14 18:04:35 +01:00
|
|
|
int tab_size= 4;
|
2021-07-02 19:13:26 +02:00
|
|
|
THD *thd= current_thd;
|
2017-02-14 17:51:03 +04:00
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
2017-02-14 17:51:03 +04:00
|
|
|
if (fmt == DETAILED)
|
|
|
|
{
|
|
|
|
if (arg_count > 1)
|
|
|
|
{
|
2017-09-28 10:38:02 +00:00
|
|
|
tab_size= (int)args[1]->val_int();
|
2017-02-14 17:51:03 +04:00
|
|
|
if (args[1]->null_value)
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (tab_size < 0)
|
|
|
|
tab_size= 0;
|
|
|
|
else if (tab_size > TAB_SIZE_LIMIT)
|
|
|
|
tab_size= TAB_SIZE_LIMIT;
|
|
|
|
}
|
|
|
|
|
2017-02-06 06:47:48 +04:00
|
|
|
json_scan_start(&je, js->charset(), (const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr()+js->length());
|
2021-07-02 19:13:26 +02:00
|
|
|
je.killed_ptr= (uchar*)&thd->killed;
|
2017-02-06 06:47:48 +04:00
|
|
|
|
2017-02-14 17:51:03 +04:00
|
|
|
if (json_nice(&je, str, fmt, tab_size))
|
2017-02-06 06:47:48 +04:00
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
report_json_error(js, &je, 0);
|
2021-07-02 19:13:26 +02:00
|
|
|
thd->check_killed(); // to get the error message right
|
2017-02-06 06:47:48 +04:00
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2017-02-14 11:11:47 +01:00
|
|
|
String *Item_func_json_format::val_json(String *str)
|
|
|
|
{
|
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
return js;
|
|
|
|
}
|
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
int Arg_comparator::compare_json_str_basic(Item *j, Item *s)
|
|
|
|
{
|
2018-06-18 23:00:34 +04:00
|
|
|
String *js,*str;
|
|
|
|
int c_len;
|
|
|
|
json_engine_t je;
|
2017-08-11 00:50:29 +04:00
|
|
|
|
2018-06-18 23:00:34 +04:00
|
|
|
if ((js= j->val_str(&value1)))
|
|
|
|
{
|
|
|
|
json_scan_start(&je, js->charset(), (const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr()+js->length());
|
|
|
|
if (json_read_value(&je))
|
|
|
|
goto error;
|
|
|
|
if (je.value_type == JSON_VALUE_STRING)
|
|
|
|
{
|
|
|
|
if (value2.realloc_with_extra_if_needed(je.value_len) ||
|
|
|
|
(c_len= json_unescape(js->charset(), je.value,
|
|
|
|
je.value + je.value_len,
|
2019-06-28 09:05:12 +04:00
|
|
|
&my_charset_utf8mb3_general_ci,
|
2018-06-18 23:00:34 +04:00
|
|
|
(uchar *) value2.ptr(),
|
|
|
|
(uchar *) (value2.ptr() + je.value_len))) < 0)
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
value2.length(c_len);
|
|
|
|
js= &value2;
|
|
|
|
str= &value1;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
str= &value2;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
if ((str= s->val_str(str)))
|
|
|
|
{
|
|
|
|
if (set_null)
|
|
|
|
owner->null_value= 0;
|
|
|
|
return sortcmp(js, str, compare_collation());
|
|
|
|
}
|
2017-08-11 00:50:29 +04:00
|
|
|
}
|
2018-06-18 23:00:34 +04:00
|
|
|
|
2017-08-11 00:50:29 +04:00
|
|
|
error:
|
|
|
|
if (set_null)
|
|
|
|
owner->null_value= 1;
|
|
|
|
return -1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
int Arg_comparator::compare_e_json_str_basic(Item *j, Item *s)
|
|
|
|
{
|
|
|
|
String *res1,*res2;
|
|
|
|
json_value_types type;
|
|
|
|
char *value;
|
|
|
|
int value_len, c_len;
|
|
|
|
Item_func_json_extract *e= (Item_func_json_extract *) j;
|
|
|
|
|
|
|
|
res1= e->read_json(&value1, &type, &value, &value_len);
|
|
|
|
res2= s->val_str(&value2);
|
|
|
|
|
|
|
|
if (!res1 || !res2)
|
|
|
|
return MY_TEST(res1 == res2);
|
|
|
|
|
|
|
|
if (type == JSON_VALUE_STRING)
|
|
|
|
{
|
|
|
|
if (value1.realloc_with_extra_if_needed(value_len) ||
|
|
|
|
(c_len= json_unescape(value1.charset(), (uchar *) value,
|
|
|
|
(uchar *) value+value_len,
|
2019-06-28 09:05:12 +04:00
|
|
|
&my_charset_utf8mb3_general_ci,
|
2017-08-11 00:50:29 +04:00
|
|
|
(uchar *) value1.ptr(),
|
|
|
|
(uchar *) (value1.ptr() + value_len))) < 0)
|
|
|
|
return 1;
|
|
|
|
value1.length(c_len);
|
|
|
|
res1= &value1;
|
|
|
|
}
|
|
|
|
|
|
|
|
return MY_TEST(sortcmp(res1, res2, compare_collation()) == 0);
|
|
|
|
}
|
2019-07-04 13:12:08 +03:00
|
|
|
|
|
|
|
|
2020-06-04 13:53:14 +04:00
|
|
|
String *Item_func_json_arrayagg::get_str_from_item(Item *i, String *tmp)
|
|
|
|
{
|
|
|
|
m_tmp_json.length(0);
|
|
|
|
if (append_json_value(&m_tmp_json, i, tmp))
|
|
|
|
return NULL;
|
|
|
|
return &m_tmp_json;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-30 21:17:57 +02:00
|
|
|
String *Item_func_json_arrayagg::get_str_from_field(Item *i,Field *f,
|
2020-06-04 13:53:14 +04:00
|
|
|
String *tmp, const uchar *key, size_t offset)
|
2019-07-04 13:12:08 +03:00
|
|
|
{
|
2020-06-04 10:00:56 +04:00
|
|
|
m_tmp_json.length(0);
|
2020-06-04 13:53:14 +04:00
|
|
|
|
2021-06-30 21:17:57 +02:00
|
|
|
if (append_json_value_from_field(&m_tmp_json, i, f, key, offset, tmp))
|
2020-06-04 13:53:14 +04:00
|
|
|
return NULL;
|
|
|
|
|
2020-06-04 10:00:56 +04:00
|
|
|
return &m_tmp_json;
|
2020-06-04 13:53:14 +04:00
|
|
|
|
2019-07-04 13:12:08 +03:00
|
|
|
}
|
|
|
|
|
|
|
|
|
2020-06-15 22:51:21 +04:00
|
|
|
void Item_func_json_arrayagg::cut_max_length(String *result,
|
|
|
|
uint old_length, uint max_length) const
|
|
|
|
{
|
|
|
|
if (result->length() == 0)
|
|
|
|
return;
|
|
|
|
|
|
|
|
if (result->ptr()[result->length() - 1] != '"' ||
|
|
|
|
max_length == 0)
|
|
|
|
{
|
|
|
|
Item_func_group_concat::cut_max_length(result, old_length, max_length);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
Item_func_group_concat::cut_max_length(result, old_length, max_length-1);
|
|
|
|
result->append('"');
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-06-24 13:28:28 +04:00
|
|
|
Item *Item_func_json_arrayagg::copy_or_same(THD* thd)
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_json_arrayagg(thd, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-07-04 13:12:08 +03:00
|
|
|
String* Item_func_json_arrayagg::val_str(String *str)
|
|
|
|
{
|
2020-04-28 16:08:18 +04:00
|
|
|
if ((str= Item_func_group_concat::val_str(str)))
|
|
|
|
{
|
|
|
|
String s;
|
|
|
|
s.append('[');
|
|
|
|
s.swap(*str);
|
|
|
|
str->append(s);
|
|
|
|
str->append(']');
|
|
|
|
}
|
2019-07-04 13:12:08 +03:00
|
|
|
return str;
|
|
|
|
}
|
2019-10-14 14:24:22 +04:00
|
|
|
|
|
|
|
|
|
|
|
Item_func_json_objectagg::
|
|
|
|
Item_func_json_objectagg(THD *thd, Item_func_json_objectagg *item)
|
|
|
|
:Item_sum(thd, item)
|
|
|
|
{
|
2020-06-29 20:06:28 +05:30
|
|
|
quick_group= FALSE;
|
2019-10-14 14:24:22 +04:00
|
|
|
result.set_charset(collation.collation);
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
result.append('{');
|
2019-10-14 14:24:22 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool
|
|
|
|
Item_func_json_objectagg::fix_fields(THD *thd, Item **ref)
|
|
|
|
{
|
|
|
|
uint i; /* for loop variable */
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed() == 0);
|
2019-10-14 14:24:22 +04:00
|
|
|
|
2020-10-24 00:16:56 +04:00
|
|
|
memcpy(orig_args, args, sizeof(Item*) * arg_count);
|
|
|
|
|
2019-10-14 14:24:22 +04:00
|
|
|
if (init_sum_func_check(thd))
|
|
|
|
return TRUE;
|
|
|
|
|
2020-09-02 03:13:32 +03:00
|
|
|
set_maybe_null();
|
2019-10-14 14:24:22 +04:00
|
|
|
|
|
|
|
/*
|
|
|
|
Fix fields for select list and ORDER clause
|
|
|
|
*/
|
|
|
|
|
|
|
|
for (i=0 ; i < arg_count ; i++)
|
|
|
|
{
|
|
|
|
if (args[i]->fix_fields_if_needed_for_scalar(thd, &args[i]))
|
|
|
|
return TRUE;
|
2020-09-02 03:13:32 +03:00
|
|
|
with_flags|= args[i]->with_flags;
|
2019-10-14 14:24:22 +04:00
|
|
|
}
|
|
|
|
|
|
|
|
/* skip charset aggregation for order columns */
|
|
|
|
if (agg_arg_charsets_for_string_result(collation, args, arg_count))
|
|
|
|
return 1;
|
|
|
|
|
|
|
|
result.set_charset(collation.collation);
|
|
|
|
result_field= 0;
|
|
|
|
null_value= 1;
|
|
|
|
max_length= (uint32)(thd->variables.group_concat_max_len
|
|
|
|
/ collation.collation->mbminlen
|
|
|
|
* collation.collation->mbmaxlen);
|
|
|
|
|
|
|
|
if (check_sum_func(thd, ref))
|
|
|
|
return TRUE;
|
|
|
|
|
2020-09-02 03:13:32 +03:00
|
|
|
base_flags|= item_base_t::FIXED;
|
2019-10-14 14:24:22 +04:00
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Item_func_json_objectagg::cleanup()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_func_json_objectagg::cleanup");
|
|
|
|
Item_sum::cleanup();
|
|
|
|
|
|
|
|
result.length(1);
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
Item *Item_func_json_objectagg::copy_or_same(THD* thd)
|
|
|
|
{
|
|
|
|
return new (thd->mem_root) Item_func_json_objectagg(thd, this);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
void Item_func_json_objectagg::clear()
|
|
|
|
{
|
|
|
|
result.length(1);
|
|
|
|
null_value= 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_func_json_objectagg::add()
|
|
|
|
{
|
|
|
|
StringBuffer<MAX_FIELD_WIDTH> buf;
|
|
|
|
String *key;
|
|
|
|
|
|
|
|
key= args[0]->val_str(&buf);
|
|
|
|
if (args[0]->is_null())
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
null_value= 0;
|
|
|
|
if (result.length() > 1)
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
result.append(STRING_WITH_LEN(", "));
|
2019-10-14 14:24:22 +04:00
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
result.append('"');
|
2023-02-12 18:42:23 +01:00
|
|
|
st_append_escaped(&result,key);
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
result.append(STRING_WITH_LEN("\":"));
|
2019-10-14 14:24:22 +04:00
|
|
|
|
|
|
|
buf.length(0);
|
|
|
|
append_json_value(&result, args[1], &buf);
|
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String* Item_func_json_objectagg::val_str(String* str)
|
|
|
|
{
|
2020-08-02 12:31:14 +03:00
|
|
|
DBUG_ASSERT(fixed());
|
2019-10-14 14:24:22 +04:00
|
|
|
if (null_value)
|
|
|
|
return 0;
|
|
|
|
|
Reduce usage of strlen()
Changes:
- To detect automatic strlen() I removed the methods in String that
uses 'const char *' without a length:
- String::append(const char*)
- Binary_string(const char *str)
- String(const char *str, CHARSET_INFO *cs)
- append_for_single_quote(const char *)
All usage of append(const char*) is changed to either use
String::append(char), String::append(const char*, size_t length) or
String::append(LEX_CSTRING)
- Added STRING_WITH_LEN() around constant string arguments to
String::append()
- Added overflow argument to escape_string_for_mysql() and
escape_quotes_for_mysql() instead of returning (size_t) -1 on overflow.
This was needed as most usage of the above functions never tested the
result for -1 and would have given wrong results or crashes in case
of overflows.
- Added Item_func_or_sum::func_name_cstring(), which returns LEX_CSTRING.
Changed all Item_func::func_name()'s to func_name_cstring()'s.
The old Item_func_or_sum::func_name() is now an inline function that
returns func_name_cstring().str.
- Changed Item::mode_name() and Item::func_name_ext() to return
LEX_CSTRING.
- Changed for some functions the name argument from const char * to
to const LEX_CSTRING &:
- Item::Item_func_fix_attributes()
- Item::check_type_...()
- Type_std_attributes::agg_item_collations()
- Type_std_attributes::agg_item_set_converter()
- Type_std_attributes::agg_arg_charsets...()
- Type_handler_hybrid_field_type::aggregate_for_result()
- Type_handler_geometry::check_type_geom_or_binary()
- Type_handler::Item_func_or_sum_illegal_param()
- Predicant_to_list_comparator::add_value_skip_null()
- Predicant_to_list_comparator::add_value()
- cmp_item_row::prepare_comparators()
- cmp_item_row::aggregate_row_elements_for_comparison()
- Cursor_ref::print_func()
- Removes String_space() as it was only used in one cases and that
could be simplified to not use String_space(), thanks to the fixed
my_vsnprintf().
- Added some const LEX_CSTRING's for common strings:
- NULL_clex_str, DATA_clex_str, INDEX_clex_str.
- Changed primary_key_name to a LEX_CSTRING
- Renamed String::set_quick() to String::set_buffer_if_not_allocated() to
clarify what the function really does.
- Rename of protocol function:
bool store(const char *from, CHARSET_INFO *cs) to
bool store_string_or_null(const char *from, CHARSET_INFO *cs).
This was done to both clarify the difference between this 'store' function
and also to make it easier to find unoptimal usage of store() calls.
- Added Protocol::store(const LEX_CSTRING*, CHARSET_INFO*)
- Changed some 'const char*' arrays to instead be of type LEX_CSTRING.
- class Item_func_units now used LEX_CSTRING for name.
Other things:
- Fixed a bug in mysql.cc:construct_prompt() where a wrong escape character
in the prompt would cause some part of the prompt to be duplicated.
- Fixed a lot of instances where the length of the argument to
append is known or easily obtain but was not used.
- Removed some not needed 'virtual' definition for functions that was
inherited from the parent. I added override to these.
- Fixed Ordered_key::print() to preallocate needed buffer. Old code could
case memory overruns.
- Simplified some loops when adding char * to a String with delimiters.
2020-08-12 20:29:55 +03:00
|
|
|
result.append('}');
|
2019-10-14 14:24:22 +04:00
|
|
|
return &result;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2021-07-02 08:18:16 +02:00
|
|
|
String *Item_func_json_normalize::val_str(String *buf)
|
|
|
|
{
|
|
|
|
String tmp;
|
|
|
|
String *raw_json= args[0]->val_str(&tmp);
|
|
|
|
|
|
|
|
DYNAMIC_STRING normalized_json;
|
|
|
|
if (init_dynamic_string(&normalized_json, NULL, 0, 0))
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
null_value= args[0]->null_value;
|
|
|
|
if (null_value)
|
|
|
|
goto end;
|
|
|
|
|
|
|
|
if (json_normalize(&normalized_json,
|
2021-08-06 11:27:52 +03:00
|
|
|
raw_json->ptr(), raw_json->length(),
|
2021-07-02 08:18:16 +02:00
|
|
|
raw_json->charset()))
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
buf->length(0);
|
|
|
|
if (buf->append(normalized_json.str, normalized_json.length))
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
goto end;
|
|
|
|
}
|
|
|
|
|
|
|
|
end:
|
|
|
|
dynstr_free(&normalized_json);
|
|
|
|
return null_value ? NULL : buf;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_normalize::fix_length_and_dec(THD *thd)
|
2021-07-02 08:18:16 +02:00
|
|
|
{
|
|
|
|
collation.set(&my_charset_utf8mb4_bin);
|
|
|
|
/* 0 becomes 0.0E0, thus one character becomes 5 chars */
|
|
|
|
fix_char_length_ulonglong((ulonglong) args[0]->max_char_length() * 5);
|
|
|
|
set_maybe_null();
|
|
|
|
return FALSE;
|
|
|
|
}
|
2022-01-30 15:45:25 +05:30
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
When the two values match or don't match we need to return true or false.
|
|
|
|
But we can have some more elements in the array left or some more keys
|
|
|
|
left in the object that we no longer want to compare. In this case,
|
|
|
|
we want to skip the current item.
|
|
|
|
*/
|
2023-06-17 18:37:22 +02:00
|
|
|
static void json_skip_current_level(json_engine_t *js, json_engine_t *value)
|
2022-01-30 15:45:25 +05:30
|
|
|
{
|
|
|
|
json_skip_level(js);
|
|
|
|
json_skip_level(value);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/* At least one of the two arguments is a scalar. */
|
2023-06-17 18:37:22 +02:00
|
|
|
static bool json_find_overlap_with_scalar(json_engine_t *js, json_engine_t *value)
|
2022-01-30 15:45:25 +05:30
|
|
|
{
|
|
|
|
if (json_value_scalar(value))
|
|
|
|
{
|
|
|
|
if (js->value_type == value->value_type)
|
|
|
|
{
|
|
|
|
if (js->value_type == JSON_VALUE_NUMBER)
|
|
|
|
{
|
|
|
|
double d_j, d_v;
|
|
|
|
char *end;
|
|
|
|
int err;
|
|
|
|
|
|
|
|
d_j= js->s.cs->strntod((char *) js->value, js->value_len, &end, &err);
|
|
|
|
d_v= value->s.cs->strntod((char *) value->value, value->value_len,
|
|
|
|
&end, &err);
|
|
|
|
|
|
|
|
return (fabs(d_j - d_v) < 1e-12);
|
|
|
|
}
|
|
|
|
else if (js->value_type == JSON_VALUE_STRING)
|
|
|
|
{
|
|
|
|
return value->value_len == js->value_len &&
|
|
|
|
memcmp(value->value, js->value, value->value_len) == 0;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return value->value_type == js->value_type;
|
|
|
|
}
|
|
|
|
else if (value->value_type == JSON_VALUE_ARRAY)
|
|
|
|
{
|
|
|
|
while (json_scan_next(value) == 0 && value->state == JST_VALUE)
|
|
|
|
{
|
|
|
|
if (json_read_value(value))
|
|
|
|
return FALSE;
|
|
|
|
if (js->value_type == value->value_type)
|
|
|
|
{
|
|
|
|
int res1= json_find_overlap_with_scalar(js, value);
|
|
|
|
if (res1)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
if (!json_value_scalar(value))
|
|
|
|
json_skip_level(value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Compare when one is object and other is array. This means we are looking
|
|
|
|
for the object in the array. Hence, when value type of an element of the
|
|
|
|
array is object, then compare the two objects entirely. If they are
|
|
|
|
equal return true else return false.
|
|
|
|
*/
|
2023-06-17 18:37:22 +02:00
|
|
|
static bool json_compare_arr_and_obj(json_engine_t *js, json_engine_t *value)
|
2022-01-30 15:45:25 +05:30
|
|
|
{
|
|
|
|
st_json_engine_t loc_val= *value;
|
|
|
|
while (json_scan_next(js) == 0 && js->state == JST_VALUE)
|
|
|
|
{
|
|
|
|
if (json_read_value(js))
|
|
|
|
return FALSE;
|
|
|
|
if (js->value_type == JSON_VALUE_OBJECT)
|
|
|
|
{
|
|
|
|
int res1= json_find_overlap_with_object(js, value, true);
|
|
|
|
if (res1)
|
|
|
|
return TRUE;
|
|
|
|
*value= loc_val;
|
|
|
|
}
|
2022-07-30 16:56:18 +05:30
|
|
|
if (js->value_type == JSON_VALUE_ARRAY)
|
2022-01-30 15:45:25 +05:30
|
|
|
json_skip_level(js);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool json_compare_arrays_in_order(json_engine_t *js, json_engine_t *value)
|
|
|
|
{
|
|
|
|
bool res= false;
|
|
|
|
while (json_scan_next(js) == 0 && json_scan_next(value) == 0 &&
|
|
|
|
js->state == JST_VALUE && value->state == JST_VALUE)
|
|
|
|
{
|
|
|
|
if (json_read_value(js) || json_read_value(value))
|
|
|
|
return FALSE;
|
|
|
|
if (js->value_type != value->value_type)
|
|
|
|
{
|
|
|
|
json_skip_current_level(js, value);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
res= check_overlaps(js, value, true);
|
|
|
|
if (!res)
|
|
|
|
{
|
|
|
|
json_skip_current_level(js, value);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
res= (value->state == JST_ARRAY_END || value->state == JST_OBJ_END ?
|
|
|
|
TRUE : FALSE);
|
|
|
|
json_skip_current_level(js, value);
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-06-17 18:37:22 +02:00
|
|
|
static int json_find_overlap_with_array(json_engine_t *js, json_engine_t *value,
|
2022-01-30 15:45:25 +05:30
|
|
|
bool compare_whole)
|
|
|
|
{
|
|
|
|
if (value->value_type == JSON_VALUE_ARRAY)
|
|
|
|
{
|
|
|
|
if (compare_whole)
|
|
|
|
return json_compare_arrays_in_order(js, value);
|
|
|
|
|
|
|
|
json_engine_t loc_value= *value, current_js= *js;
|
|
|
|
|
|
|
|
while (json_scan_next(js) == 0 && js->state == JST_VALUE)
|
|
|
|
{
|
|
|
|
if (json_read_value(js))
|
|
|
|
return FALSE;
|
|
|
|
current_js= *js;
|
|
|
|
while (json_scan_next(value) == 0 && value->state == JST_VALUE)
|
|
|
|
{
|
|
|
|
if (json_read_value(value))
|
|
|
|
return FALSE;
|
|
|
|
if (js->value_type == value->value_type)
|
|
|
|
{
|
|
|
|
int res1= check_overlaps(js, value, true);
|
|
|
|
if (res1)
|
|
|
|
return TRUE;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (!json_value_scalar(value))
|
|
|
|
json_skip_level(value);
|
|
|
|
}
|
|
|
|
*js= current_js;
|
|
|
|
}
|
|
|
|
*value= loc_value;
|
|
|
|
if (!json_value_scalar(js))
|
|
|
|
json_skip_level(js);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
else if (value->value_type == JSON_VALUE_OBJECT)
|
|
|
|
{
|
|
|
|
if (compare_whole)
|
|
|
|
{
|
|
|
|
json_skip_current_level(js, value);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return json_compare_arr_and_obj(js, value);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
return json_find_overlap_with_scalar(value, js);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2022-07-30 16:56:18 +05:30
|
|
|
int compare_nested_object(json_engine_t *js, json_engine_t *value)
|
|
|
|
{
|
|
|
|
int result= 0;
|
|
|
|
const char *value_begin= (const char*)value->s.c_str-1;
|
|
|
|
const char *js_begin= (const char*)js->s.c_str-1;
|
|
|
|
json_skip_level(value);
|
|
|
|
json_skip_level(js);
|
|
|
|
const char *value_end= (const char*)value->s.c_str;
|
|
|
|
const char *js_end= (const char*)js->s.c_str;
|
|
|
|
|
|
|
|
String a(value_begin, value_end-value_begin,value->s.cs);
|
|
|
|
String b(js_begin, js_end-js_begin, js->s.cs);
|
|
|
|
|
|
|
|
DYNAMIC_STRING a_res, b_res;
|
|
|
|
if (init_dynamic_string(&a_res, NULL, 4096, 1024) ||
|
|
|
|
init_dynamic_string(&b_res, NULL, 4096, 1024))
|
|
|
|
{
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
if (json_normalize(&a_res, a.ptr(), a.length(), value->s.cs) ||
|
|
|
|
json_normalize(&b_res, b.ptr(), b.length(), value->s.cs))
|
|
|
|
{
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
result= strcmp(a_res.str, b_res.str) ? 0 : 1;
|
|
|
|
|
|
|
|
error:
|
|
|
|
dynstr_free(&a_res);
|
|
|
|
dynstr_free(&b_res);
|
|
|
|
|
|
|
|
return MY_TEST(result);
|
|
|
|
}
|
2023-06-17 18:37:22 +02:00
|
|
|
|
|
|
|
|
|
|
|
static int json_find_overlap_with_object(json_engine_t *js, json_engine_t *value,
|
2022-01-30 15:45:25 +05:30
|
|
|
bool compare_whole)
|
|
|
|
{
|
|
|
|
if (value->value_type == JSON_VALUE_OBJECT)
|
|
|
|
{
|
2022-07-30 16:56:18 +05:30
|
|
|
if (compare_whole)
|
2022-01-30 15:45:25 +05:30
|
|
|
{
|
2022-07-30 16:56:18 +05:30
|
|
|
return compare_nested_object(js, value);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/* Find at least one common key-value pair */
|
|
|
|
json_string_t key_name;
|
|
|
|
bool found_key= false, found_value= false;
|
|
|
|
json_engine_t loc_js= *js;
|
|
|
|
const uchar *k_start, *k_end;
|
2022-01-30 15:45:25 +05:30
|
|
|
|
2022-07-30 16:56:18 +05:30
|
|
|
json_string_set_cs(&key_name, value->s.cs);
|
2022-01-30 15:45:25 +05:30
|
|
|
|
2022-07-30 16:56:18 +05:30
|
|
|
while (json_scan_next(value) == 0 && value->state == JST_KEY)
|
2022-01-30 15:45:25 +05:30
|
|
|
{
|
2022-07-30 16:56:18 +05:30
|
|
|
k_start= value->s.c_str;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
k_end= value->s.c_str;
|
|
|
|
} while (json_read_keyname_chr(value) == 0);
|
|
|
|
|
|
|
|
if (unlikely(value->s.error))
|
2022-01-30 15:45:25 +05:30
|
|
|
return FALSE;
|
|
|
|
|
2022-07-30 16:56:18 +05:30
|
|
|
json_string_set_str(&key_name, k_start, k_end);
|
|
|
|
found_key= find_key_in_object(js, &key_name);
|
|
|
|
found_value= 0;
|
|
|
|
|
|
|
|
if (found_key)
|
2022-01-30 15:45:25 +05:30
|
|
|
{
|
2022-07-30 16:56:18 +05:30
|
|
|
if (json_read_value(js) || json_read_value(value))
|
|
|
|
return FALSE;
|
|
|
|
|
|
|
|
/*
|
|
|
|
The value of key-value pair can be an be anything. If it is an object
|
|
|
|
then we need to compare the whole value and if it is an array then
|
|
|
|
we need to compare the elements in that order. So set compare_whole
|
|
|
|
to true.
|
|
|
|
*/
|
|
|
|
if (js->value_type == value->value_type)
|
|
|
|
found_value= check_overlaps(js, value, true);
|
|
|
|
if (found_value)
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
We have found at least one common key-value pair now.
|
|
|
|
No need to check for more key-value pairs. So skip remaining
|
|
|
|
jsons and return TRUE.
|
|
|
|
*/
|
|
|
|
json_skip_current_level(js, value);
|
2022-01-30 15:45:25 +05:30
|
|
|
return TRUE;
|
2022-07-30 16:56:18 +05:30
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
/*
|
|
|
|
Key is found but value is not found. We have already
|
|
|
|
exhausted both values for current key. Hence "reset"
|
|
|
|
only js (first argument i.e json document) and
|
|
|
|
continue.
|
|
|
|
*/
|
|
|
|
*js= loc_js;
|
|
|
|
continue;
|
|
|
|
}
|
2022-01-30 15:45:25 +05:30
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
2022-07-30 16:56:18 +05:30
|
|
|
/*
|
|
|
|
key is not found. So no need to check for value for that key.
|
|
|
|
Read the value anyway so we get the "type" of json value.
|
|
|
|
If is is non-scalar then skip the entire value
|
|
|
|
(scalar values get exhausted while reading so no need to skip them).
|
|
|
|
Then reset the json doc again.
|
|
|
|
*/
|
|
|
|
if (json_read_value(value))
|
2022-01-30 15:45:25 +05:30
|
|
|
return FALSE;
|
2022-07-30 16:56:18 +05:30
|
|
|
if (!json_value_scalar(value))
|
|
|
|
json_skip_level(value);
|
2022-01-30 15:45:25 +05:30
|
|
|
*js= loc_js;
|
|
|
|
}
|
|
|
|
}
|
2022-07-30 16:56:18 +05:30
|
|
|
/*
|
|
|
|
At this point we have already returned true if any intersection exists.
|
|
|
|
So skip jsons if not exhausted and return false.
|
|
|
|
*/
|
|
|
|
json_skip_current_level(js, value);
|
|
|
|
return FALSE;
|
2022-01-30 15:45:25 +05:30
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (value->value_type == JSON_VALUE_ARRAY)
|
|
|
|
{
|
|
|
|
if (compare_whole)
|
|
|
|
{
|
|
|
|
json_skip_current_level(js, value);
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
return json_compare_arr_and_obj(value, js);
|
|
|
|
}
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Find if two json documents overlap
|
|
|
|
|
|
|
|
SYNOPSIS
|
|
|
|
check_overlaps()
|
|
|
|
js - json document
|
|
|
|
value - value
|
|
|
|
compare_whole - If true then find full overlap with the document in case of
|
|
|
|
object and comparing in-order in case of array.
|
|
|
|
Else find at least one match between two objects or array.
|
|
|
|
|
|
|
|
IMPLEMENTATION
|
|
|
|
We can compare two json datatypes if they are of same type to check if
|
|
|
|
they are equal. When comparing between a json document and json value,
|
|
|
|
there can be following cases:
|
|
|
|
1) When at least one of the two json documents is of scalar type:
|
|
|
|
1.a) If value and json document both are scalar, then return true
|
|
|
|
if they have same type and value.
|
|
|
|
1.b) If json document is scalar but other is array (or vice versa),
|
|
|
|
then return true if array has at least one element of same type
|
|
|
|
and value as scalar.
|
|
|
|
1.c) If one is scalar and other is object, then return false because
|
|
|
|
it can't be compared.
|
|
|
|
|
|
|
|
2) When both arguments are of non-scalar type:
|
|
|
|
2.a) If both arguments are arrays:
|
|
|
|
Iterate over the value and json document. If there exists at least
|
|
|
|
one element in other array of same type and value as that of
|
|
|
|
element in value, then return true else return false.
|
|
|
|
2.b) If both arguments are objects:
|
|
|
|
Iterate over value and json document and if there exists at least
|
|
|
|
one key-value pair common between two objects, then return true,
|
|
|
|
else return false.
|
|
|
|
2.c) If either of json document or value is array and other is object:
|
|
|
|
Iterate over the array, if an element of type object is found,
|
|
|
|
then compare it with the object (which is the other arguemnt).
|
|
|
|
If the entire object matches i.e all they key value pairs match,
|
|
|
|
then return true else return false.
|
|
|
|
|
|
|
|
When we are comparing an object which is nested in other object or nested
|
|
|
|
in an array, we need to compare all the key-value pairs, irrespective of
|
|
|
|
what order they are in as opposed to non-nested where we return true if
|
|
|
|
at least one match is found. However, if we have an array nested in another
|
|
|
|
array, then we compare two arrays in that order i.e we compare
|
|
|
|
i-th element of array 1 with i-th element of array 2.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE - If two json documents do not overlap
|
|
|
|
TRUE - if two json documents overlap
|
|
|
|
*/
|
2023-06-17 18:37:22 +02:00
|
|
|
static int check_overlaps(json_engine_t *js, json_engine_t *value, bool compare_whole)
|
2022-01-30 15:45:25 +05:30
|
|
|
{
|
2022-06-21 19:10:11 +05:30
|
|
|
DBUG_EXECUTE_IF("json_check_min_stack_requirement",
|
2022-07-29 16:17:09 +05:30
|
|
|
{
|
|
|
|
long arbitrary_var;
|
|
|
|
long stack_used_up= (available_stack_size(current_thd->thread_stack, &arbitrary_var));
|
|
|
|
ALLOCATE_MEM_ON_STACK(my_thread_stack_size-stack_used_up-STACK_MIN_SIZE);
|
|
|
|
});
|
2022-06-21 19:10:11 +05:30
|
|
|
if (check_stack_overrun(current_thd, STACK_MIN_SIZE , NULL))
|
|
|
|
return 1;
|
2022-06-21 19:10:11 +05:30
|
|
|
|
2022-01-30 15:45:25 +05:30
|
|
|
switch (js->value_type)
|
|
|
|
{
|
|
|
|
case JSON_VALUE_OBJECT:
|
|
|
|
return json_find_overlap_with_object(js, value, compare_whole);
|
|
|
|
case JSON_VALUE_ARRAY:
|
|
|
|
return json_find_overlap_with_array(js, value, compare_whole);
|
|
|
|
default:
|
|
|
|
return json_find_overlap_with_scalar(js, value);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
longlong Item_func_json_overlaps::val_int()
|
|
|
|
{
|
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
|
|
|
json_engine_t je, ve;
|
|
|
|
int result;
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
|
|
|
return 0;
|
|
|
|
|
|
|
|
if (!a2_parsed)
|
|
|
|
{
|
|
|
|
val= args[1]->val_json(&tmp_val);
|
|
|
|
a2_parsed= a2_constant;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (val == 0)
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_scan_start(&je, js->charset(), (const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
|
|
|
|
json_scan_start(&ve, val->charset(), (const uchar *) val->ptr(),
|
|
|
|
(const uchar *) val->end());
|
|
|
|
|
|
|
|
if (json_read_value(&je) || json_read_value(&ve))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
result= check_overlaps(&je, &ve, false);
|
|
|
|
if (unlikely(je.s.error || ve.s.error))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
return result;
|
|
|
|
|
|
|
|
error:
|
|
|
|
if (je.s.error)
|
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
if (ve.s.error)
|
|
|
|
report_json_error(val, &ve, 1);
|
2022-03-03 14:40:55 +05:30
|
|
|
return 0;
|
2022-01-30 15:45:25 +05:30
|
|
|
}
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
bool Item_func_json_overlaps::fix_length_and_dec(THD *thd)
|
2022-01-30 15:45:25 +05:30
|
|
|
{
|
|
|
|
a2_constant= args[1]->const_item();
|
|
|
|
a2_parsed= FALSE;
|
|
|
|
set_maybe_null();
|
|
|
|
|
2022-01-26 03:02:45 +05:30
|
|
|
return Item_bool_func::fix_length_and_dec(thd);
|
2022-01-30 15:45:25 +05:30
|
|
|
}
|
2022-10-28 13:03:13 +05:30
|
|
|
|
|
|
|
longlong Item_func_json_schema_valid::val_int()
|
|
|
|
{
|
|
|
|
json_engine_t ve;
|
|
|
|
int is_valid= 1;
|
|
|
|
|
|
|
|
if (!schema_parsed)
|
2023-03-02 17:50:19 +05:30
|
|
|
{
|
|
|
|
null_value= 1;
|
2023-05-02 16:34:07 +05:30
|
|
|
return 0;
|
2023-03-02 17:50:19 +05:30
|
|
|
}
|
2022-10-28 13:03:13 +05:30
|
|
|
|
2023-05-02 16:34:07 +05:30
|
|
|
val= args[1]->val_json(&tmp_val);
|
2022-10-28 13:03:13 +05:30
|
|
|
|
2023-05-02 16:34:07 +05:30
|
|
|
if (!val)
|
2022-10-28 13:03:13 +05:30
|
|
|
{
|
2023-05-02 16:34:07 +05:30
|
|
|
null_value= 1;
|
|
|
|
return 0;
|
2022-10-28 13:03:13 +05:30
|
|
|
}
|
2023-05-02 16:34:07 +05:30
|
|
|
null_value= 0;
|
|
|
|
|
|
|
|
if (!val->length())
|
|
|
|
return 1;
|
2022-10-28 13:03:13 +05:30
|
|
|
|
|
|
|
json_scan_start(&ve, val->charset(), (const uchar *) val->ptr(),
|
|
|
|
(const uchar *) val->end());
|
|
|
|
|
|
|
|
if (json_read_value(&ve))
|
|
|
|
goto end;
|
|
|
|
|
|
|
|
if (!keyword_list.is_empty())
|
|
|
|
{
|
|
|
|
List_iterator <Json_schema_keyword> it(keyword_list);;
|
|
|
|
Json_schema_keyword* curr_keyword= NULL;
|
|
|
|
while ((curr_keyword=it++))
|
|
|
|
{
|
|
|
|
if (curr_keyword->validate(&ve, NULL, NULL))
|
|
|
|
{
|
|
|
|
is_valid= 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-25 13:54:05 +05:30
|
|
|
if (is_valid && !ve.s.error && !json_scan_ended(&ve))
|
|
|
|
{
|
|
|
|
while (json_scan_next(&ve) == 0) /* no-op */;
|
|
|
|
}
|
|
|
|
|
2022-10-28 13:03:13 +05:30
|
|
|
end:
|
|
|
|
if (unlikely(ve.s.error))
|
|
|
|
{
|
|
|
|
is_valid= 0;
|
2023-04-25 13:54:05 +05:30
|
|
|
report_json_error(val, &ve, 1);
|
2022-10-28 13:03:13 +05:30
|
|
|
}
|
|
|
|
|
|
|
|
return is_valid;
|
|
|
|
}
|
|
|
|
|
|
|
|
/*
|
|
|
|
Idea behind implementation:
|
|
|
|
JSON schema basically has same structure as that of json object, consisting of
|
|
|
|
key-value pairs. So it can be parsed in the same manner as any json object.
|
|
|
|
|
|
|
|
However, none of the keywords are mandatory, so making guess about the json value
|
|
|
|
type based only on the keywords would be incorrect. Hence we need separate objects
|
|
|
|
denoting each keyword.
|
|
|
|
|
|
|
|
So during create_object_and_handle_keyword() we create appropriate objects
|
|
|
|
based on the keywords and validate each of them individually on the json
|
|
|
|
document by calling respective validate() function if the type matches.
|
|
|
|
If any of them fails, return false, else return true.
|
|
|
|
*/
|
|
|
|
bool Item_func_json_schema_valid::fix_length_and_dec(THD *thd)
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
|
|
|
bool res= 0;
|
|
|
|
|
|
|
|
String *js= args[0]->val_json(&tmp_js);
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value))
|
2023-05-02 16:34:07 +05:30
|
|
|
{
|
|
|
|
null_value= 1;
|
2022-10-28 13:03:13 +05:30
|
|
|
return 0;
|
2023-05-02 16:34:07 +05:30
|
|
|
}
|
2022-10-28 13:03:13 +05:30
|
|
|
json_scan_start(&je, js->charset(), (const uchar *) js->ptr(),
|
|
|
|
(const uchar *) js->ptr() + js->length());
|
|
|
|
if (!create_object_and_handle_keyword(thd, &je, &keyword_list,
|
|
|
|
&all_keywords))
|
|
|
|
schema_parsed= true;
|
|
|
|
else
|
2023-03-02 17:50:19 +05:30
|
|
|
schema_parsed= false;
|
2022-10-28 13:03:13 +05:30
|
|
|
|
2023-03-02 17:50:19 +05:30
|
|
|
/*
|
|
|
|
create_object_and_handle_keyword fails when either the json value for
|
|
|
|
keyword is invalid or when there is syntax error. Return NULL in both
|
|
|
|
these cases.
|
|
|
|
*/
|
|
|
|
if (!schema_parsed)
|
|
|
|
{
|
|
|
|
if (je.s.error)
|
|
|
|
report_json_error(js, &je, 0);
|
|
|
|
set_maybe_null();
|
|
|
|
}
|
2022-10-28 13:03:13 +05:30
|
|
|
|
|
|
|
return res || Item_bool_func::fix_length_and_dec(thd);
|
|
|
|
}
|
|
|
|
|
|
|
|
void Item_func_json_schema_valid::cleanup()
|
|
|
|
{
|
|
|
|
DBUG_ENTER("Item_func_json_schema_valid::cleanup");
|
|
|
|
Item_bool_func::cleanup();
|
|
|
|
|
|
|
|
List_iterator<Json_schema_keyword> it2(all_keywords);
|
|
|
|
Json_schema_keyword *curr_schema;
|
|
|
|
while ((curr_schema= it2++))
|
|
|
|
{
|
|
|
|
delete curr_schema;
|
|
|
|
curr_schema= nullptr;
|
|
|
|
}
|
|
|
|
all_keywords.empty();
|
|
|
|
keyword_list.empty();
|
|
|
|
|
|
|
|
DBUG_VOID_RETURN;
|
|
|
|
}
|
2023-05-25 15:45:43 +05:30
|
|
|
|
|
|
|
|
|
|
|
bool Item_func_json_key_value::get_key_value(json_engine_t *je, String *str)
|
|
|
|
{
|
|
|
|
int level= je->stack_p;
|
|
|
|
|
|
|
|
if (str->append('['))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
while (json_scan_next(je) == 0 && je->stack_p >= level)
|
|
|
|
{
|
|
|
|
const uchar *key_start, *key_end, *value_begin;
|
|
|
|
size_t v_len;
|
|
|
|
|
|
|
|
switch (je->state)
|
|
|
|
{
|
|
|
|
case JST_KEY:
|
|
|
|
|
|
|
|
key_start= je->s.c_str;
|
|
|
|
do
|
|
|
|
{
|
|
|
|
key_end= je->s.c_str;
|
|
|
|
} while (json_read_keyname_chr(je) == 0);
|
|
|
|
|
|
|
|
if (unlikely(je->s.error))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
if (json_read_value(je))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
value_begin= je->value_begin;
|
|
|
|
if (json_value_scalar(je))
|
|
|
|
v_len= je->value_end - value_begin;
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (json_skip_level(je))
|
|
|
|
goto error_return;
|
|
|
|
v_len= je->s.c_str - value_begin;
|
|
|
|
}
|
|
|
|
|
|
|
|
size_t key_len= (size_t)(key_end-key_start);
|
|
|
|
|
|
|
|
if (str->append('{') ||
|
|
|
|
str->append('"') || str->append("key", 3) || str->append('"') ||
|
|
|
|
str->append(": ", 2) ||
|
|
|
|
str->append('"') || str->append((const char*)key_start, key_len) || str->append('"') ||
|
|
|
|
str->append(", ",2) ||
|
|
|
|
str->append('"') || str->append("value", 5) || str->append('"') ||
|
|
|
|
str->append(": ", 2) ||
|
|
|
|
str->append((const char*)value_begin, v_len) ||
|
|
|
|
str->append('}') ||
|
|
|
|
str->append(", ", 2))
|
|
|
|
goto error_return;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (je->s.error)
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
if (str->length() > 1)
|
|
|
|
{
|
|
|
|
/* remove the last comma and space. */
|
|
|
|
str->chop();
|
|
|
|
str->chop();
|
|
|
|
}
|
|
|
|
|
|
|
|
/* close the array */
|
|
|
|
if (str->append(']'))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
return false;
|
|
|
|
|
|
|
|
error_return:
|
|
|
|
str->length(0);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
String* Item_func_json_key_value::val_str(String *str)
|
|
|
|
{
|
|
|
|
json_engine_t je;
|
|
|
|
|
|
|
|
if ((null_value= args[0]->null_value) ||
|
|
|
|
(null_value= args[1]->null_value))
|
|
|
|
{
|
|
|
|
goto return_null;
|
|
|
|
}
|
|
|
|
|
|
|
|
null_value= Json_path_extractor::extract(&tmp_str, args[0], args[1],
|
|
|
|
collation.collation);
|
|
|
|
if (null_value)
|
|
|
|
return NULL;
|
|
|
|
|
|
|
|
json_scan_start(&je, tmp_str.charset(), (const uchar *) tmp_str.ptr(),
|
|
|
|
(const uchar *) tmp_str.ptr() + tmp_str.length());
|
|
|
|
if (json_read_value(&je))
|
|
|
|
{
|
|
|
|
report_json_error(str, &je, 0);
|
|
|
|
goto return_null;
|
|
|
|
}
|
|
|
|
|
|
|
|
str->length(0);
|
|
|
|
if (get_key_value(&je, str))
|
|
|
|
{
|
|
|
|
report_json_error(str, &je, 0);
|
|
|
|
goto return_null;
|
|
|
|
}
|
|
|
|
|
|
|
|
return str;
|
|
|
|
|
|
|
|
return_null:
|
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_func_json_key_value::fix_length_and_dec(THD *thd)
|
|
|
|
{
|
|
|
|
collation.set(args[0]->collation);
|
|
|
|
|
|
|
|
tmp_str.set("", 0, collation.collation);
|
|
|
|
|
|
|
|
max_length= args[0]->max_length*2;
|
|
|
|
set_constant_flag(args[1]->const_item());
|
|
|
|
set_maybe_null();
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
MDEV-26182: Implement JSON_INTERSECT()
The idea is to have simple functions that the user can combine to produce
the exact result one wants, whether the user wants JSON object that has
common keys with another JSON object, or same key/value pair etc. So
making simpler function helps here.
We accomplish this by making three separate functions.
1) JSON_OBJECT_FILTER_KEYS(Obj, Arr_keys):
Put keys ( which are basically strings ) in hash, go over the object and
get key one by one. If the key is present in the hash,
add the key-value pair to result.
2) JSON_OBJECT_TO_ARRAY(Obj) : Create a string variable, Go over the json
object, and add each key value pair as an array into the result.
3) JSON_ARRAY_INTERSECT(arr1, arr2) :
Go over one of the json and add each item of the array
in hash (after normalizing each item). Go over the second array,
search the normalized item one by one in the hash. If item is found,
add it to the result.
Implementation Idea: Holyfoot ( Alexey Botchkov)
Author: tanruixiang and Rucha Deodhar
2022-06-16 15:05:35 +08:00
|
|
|
|
|
|
|
|
2023-06-17 18:37:22 +02:00
|
|
|
static bool create_hash(json_engine_t *value, HASH *items, bool &hash_inited,
|
|
|
|
MEM_ROOT *hash_root)
|
MDEV-26182: Implement JSON_INTERSECT()
The idea is to have simple functions that the user can combine to produce
the exact result one wants, whether the user wants JSON object that has
common keys with another JSON object, or same key/value pair etc. So
making simpler function helps here.
We accomplish this by making three separate functions.
1) JSON_OBJECT_FILTER_KEYS(Obj, Arr_keys):
Put keys ( which are basically strings ) in hash, go over the object and
get key one by one. If the key is present in the hash,
add the key-value pair to result.
2) JSON_OBJECT_TO_ARRAY(Obj) : Create a string variable, Go over the json
object, and add each key value pair as an array into the result.
3) JSON_ARRAY_INTERSECT(arr1, arr2) :
Go over one of the json and add each item of the array
in hash (after normalizing each item). Go over the second array,
search the normalized item one by one in the hash. If item is found,
add it to the result.
Implementation Idea: Holyfoot ( Alexey Botchkov)
Author: tanruixiang and Rucha Deodhar
2022-06-16 15:05:35 +08:00
|
|
|
{
|
|
|
|
int level= value->stack_p;
|
|
|
|
if (my_hash_init(PSI_INSTRUMENT_ME, items, value->s.cs, 0, 0, 0,
|
|
|
|
(my_hash_get_key) get_key_name, NULL, 0))
|
|
|
|
return true;
|
|
|
|
hash_inited= true;
|
|
|
|
|
|
|
|
while (json_scan_next(value) == 0 && value->stack_p >= level)
|
|
|
|
{
|
|
|
|
const uchar *value_start= NULL;
|
|
|
|
size_t value_len= 0;
|
|
|
|
DYNAMIC_STRING norm_val;
|
|
|
|
|
|
|
|
if (json_read_value(value) ||
|
|
|
|
get_current_value(value, value_start, value_len) ||
|
|
|
|
init_dynamic_string(&norm_val, NULL, 0, 0))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
if (json_normalize(&norm_val, (const char*) value_start,
|
|
|
|
value_len, value->s.cs))
|
|
|
|
{
|
|
|
|
dynstr_free(&norm_val);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *new_entry= (char*)alloc_root(hash_root,
|
|
|
|
norm_val.length+1);
|
|
|
|
if (!new_entry)
|
|
|
|
{
|
|
|
|
dynstr_free(&norm_val);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
strncpy(new_entry, norm_val.str, norm_val.length);
|
|
|
|
new_entry[norm_val.length]='\0';
|
|
|
|
|
|
|
|
dynstr_free(&norm_val);
|
|
|
|
|
|
|
|
if (my_hash_insert(items, (const uchar *) new_entry))
|
|
|
|
{
|
|
|
|
my_free(new_entry);
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
Get the starting pointer and length of the value of the current layer.
|
|
|
|
RETURN
|
|
|
|
FALSE - The function was successfully completed without errors.
|
|
|
|
TRUE - An error occurred while running.
|
|
|
|
*/
|
2023-06-17 18:37:22 +02:00
|
|
|
static bool get_current_value(json_engine_t *js, const uchar *&value_start,
|
|
|
|
size_t &value_len)
|
MDEV-26182: Implement JSON_INTERSECT()
The idea is to have simple functions that the user can combine to produce
the exact result one wants, whether the user wants JSON object that has
common keys with another JSON object, or same key/value pair etc. So
making simpler function helps here.
We accomplish this by making three separate functions.
1) JSON_OBJECT_FILTER_KEYS(Obj, Arr_keys):
Put keys ( which are basically strings ) in hash, go over the object and
get key one by one. If the key is present in the hash,
add the key-value pair to result.
2) JSON_OBJECT_TO_ARRAY(Obj) : Create a string variable, Go over the json
object, and add each key value pair as an array into the result.
3) JSON_ARRAY_INTERSECT(arr1, arr2) :
Go over one of the json and add each item of the array
in hash (after normalizing each item). Go over the second array,
search the normalized item one by one in the hash. If item is found,
add it to the result.
Implementation Idea: Holyfoot ( Alexey Botchkov)
Author: tanruixiang and Rucha Deodhar
2022-06-16 15:05:35 +08:00
|
|
|
{
|
|
|
|
value_start= js->value_begin;
|
|
|
|
|
|
|
|
if (json_value_scalar(js))
|
|
|
|
{
|
|
|
|
value_len= js->value_end - value_start;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (json_skip_level(js))
|
|
|
|
return true;
|
|
|
|
value_len= js->s.c_str - value_start;
|
|
|
|
}
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
/*
|
|
|
|
If the outermost layer of JSON is an array,
|
|
|
|
the intersection of arrays is independent of order.
|
|
|
|
Create a hash containing all elements in the array,
|
|
|
|
itterate over another array and add the common elements
|
|
|
|
to the result.
|
|
|
|
|
|
|
|
RETURN
|
|
|
|
FALSE - if two array documents have intersection
|
|
|
|
TRUE - If two array documents do not have intersection
|
|
|
|
*/
|
2023-06-17 18:37:22 +02:00
|
|
|
static bool get_intersect_between_arrays(String *str, json_engine_t *value,
|
|
|
|
HASH items)
|
MDEV-26182: Implement JSON_INTERSECT()
The idea is to have simple functions that the user can combine to produce
the exact result one wants, whether the user wants JSON object that has
common keys with another JSON object, or same key/value pair etc. So
making simpler function helps here.
We accomplish this by making three separate functions.
1) JSON_OBJECT_FILTER_KEYS(Obj, Arr_keys):
Put keys ( which are basically strings ) in hash, go over the object and
get key one by one. If the key is present in the hash,
add the key-value pair to result.
2) JSON_OBJECT_TO_ARRAY(Obj) : Create a string variable, Go over the json
object, and add each key value pair as an array into the result.
3) JSON_ARRAY_INTERSECT(arr1, arr2) :
Go over one of the json and add each item of the array
in hash (after normalizing each item). Go over the second array,
search the normalized item one by one in the hash. If item is found,
add it to the result.
Implementation Idea: Holyfoot ( Alexey Botchkov)
Author: tanruixiang and Rucha Deodhar
2022-06-16 15:05:35 +08:00
|
|
|
{
|
|
|
|
bool res= true, has_value= false;
|
|
|
|
int level= value->stack_p;
|
|
|
|
String temp_str(0);
|
|
|
|
|
|
|
|
temp_str.append('[');
|
|
|
|
while (json_scan_next(value) == 0 && value->stack_p >= level)
|
|
|
|
{
|
|
|
|
const uchar *value_start= NULL;
|
|
|
|
size_t value_len= 0;
|
|
|
|
DYNAMIC_STRING norm_val;
|
|
|
|
|
|
|
|
if (json_read_value(value) ||
|
|
|
|
get_current_value(value, value_start, value_len) ||
|
|
|
|
init_dynamic_string(&norm_val, NULL, 0, 0))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
if (json_normalize(&norm_val, (const char*) value_start,
|
|
|
|
value_len, value->s.cs))
|
|
|
|
{
|
|
|
|
dynstr_free(&norm_val);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
|
|
|
|
char *new_entry= (char*)malloc(norm_val.length+1);
|
|
|
|
if (!new_entry)
|
|
|
|
{
|
|
|
|
dynstr_free(&norm_val);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
strncpy(new_entry, norm_val.str, norm_val.length);
|
|
|
|
new_entry[norm_val.length]='\0';
|
|
|
|
|
|
|
|
dynstr_free(&norm_val);
|
|
|
|
|
|
|
|
/*
|
|
|
|
If the same value is found in the hash table, add
|
|
|
|
that value to str. At the same time, update the number
|
|
|
|
of times the value appears in the hash table.
|
|
|
|
*/
|
|
|
|
uchar * found= NULL;
|
|
|
|
if ((found= my_hash_search(&items,
|
|
|
|
(const uchar *) new_entry,
|
|
|
|
strlen(new_entry))))
|
|
|
|
{
|
|
|
|
has_value= true;
|
|
|
|
temp_str.append( (const char*) value_start, value_len);
|
|
|
|
temp_str.append(',');
|
|
|
|
if (my_hash_delete(&items, found))
|
|
|
|
{
|
|
|
|
free(new_entry);
|
|
|
|
goto error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
free(new_entry);
|
|
|
|
}
|
|
|
|
|
|
|
|
res= false;
|
|
|
|
|
|
|
|
if (has_value)
|
|
|
|
{
|
|
|
|
temp_str.chop(); /* remove last comma because there are no values after that. */
|
|
|
|
temp_str.append(']');
|
|
|
|
str->append(temp_str.ptr(), temp_str.length());
|
|
|
|
}
|
|
|
|
|
|
|
|
error:
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
String* Item_func_json_array_intersect::val_str(String *str)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(fixed());
|
|
|
|
|
|
|
|
json_engine_t je2, res_je;
|
|
|
|
String *js2= args[1]->val_json(&tmp_js2);
|
|
|
|
|
|
|
|
if (null_value || args[1]->null_value)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
str->set_charset(js2->charset());
|
|
|
|
str->length(0);
|
|
|
|
|
|
|
|
json_scan_start(&je2, js2->charset(), (const uchar *) js2->ptr(),
|
|
|
|
(const uchar *) js2->ptr() + js2->length());
|
|
|
|
|
|
|
|
if (json_read_value(&je2) || je2.value_type != JSON_VALUE_ARRAY)
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
if (get_intersect_between_arrays(str, &je2, items))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
if (str->length())
|
|
|
|
{
|
|
|
|
json_scan_start(&res_je, str->charset(), (const uchar *) str->ptr(),
|
|
|
|
(const uchar *) str->ptr() + str->length());
|
|
|
|
str= &tmp_js1;
|
|
|
|
if (json_nice(&res_je, str, Item_func_json_format::LOOSE))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
null_value= 0;
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
|
|
|
|
error_return:
|
|
|
|
if (je2.s.error)
|
|
|
|
report_json_error(js2, &je2, 1);
|
|
|
|
null_return:
|
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
bool Item_func_json_array_intersect::fix_length_and_dec(THD *thd)
|
|
|
|
{
|
|
|
|
json_engine_t je1;
|
|
|
|
String *js1= args[0]->val_json(&tmp_js1);
|
|
|
|
|
|
|
|
if (args[0]->null_value)
|
|
|
|
{
|
|
|
|
null_value= true;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
json_scan_start(&je1, js1->charset(), (const uchar *) js1->ptr(),
|
|
|
|
(const uchar *) js1->ptr() + js1->length());
|
|
|
|
/*
|
|
|
|
Scan value uses the hash table to get the intersection of two arrays.
|
|
|
|
*/
|
|
|
|
|
2023-06-17 19:00:56 +02:00
|
|
|
if (!root_inited)
|
|
|
|
init_alloc_root(PSI_NOT_INSTRUMENTED, &hash_root, 1024, 0, MYF(0));
|
MDEV-26182: Implement JSON_INTERSECT()
The idea is to have simple functions that the user can combine to produce
the exact result one wants, whether the user wants JSON object that has
common keys with another JSON object, or same key/value pair etc. So
making simpler function helps here.
We accomplish this by making three separate functions.
1) JSON_OBJECT_FILTER_KEYS(Obj, Arr_keys):
Put keys ( which are basically strings ) in hash, go over the object and
get key one by one. If the key is present in the hash,
add the key-value pair to result.
2) JSON_OBJECT_TO_ARRAY(Obj) : Create a string variable, Go over the json
object, and add each key value pair as an array into the result.
3) JSON_ARRAY_INTERSECT(arr1, arr2) :
Go over one of the json and add each item of the array
in hash (after normalizing each item). Go over the second array,
search the normalized item one by one in the hash. If item is found,
add it to the result.
Implementation Idea: Holyfoot ( Alexey Botchkov)
Author: tanruixiang and Rucha Deodhar
2022-06-16 15:05:35 +08:00
|
|
|
root_inited= true;
|
|
|
|
|
|
|
|
if (json_read_value(&je1) || je1.value_type != JSON_VALUE_ARRAY ||
|
|
|
|
create_hash(&je1, &items, hash_inited, &hash_root))
|
|
|
|
{
|
|
|
|
if (je1.s.error)
|
|
|
|
report_json_error(js1, &je1, 0);
|
|
|
|
null_value= 1;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
max_length= (args[0]->max_length < args[1]->max_length) ?
|
|
|
|
args[0]->max_length : args[1]->max_length;
|
|
|
|
|
|
|
|
set_maybe_null();
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2023-06-17 18:37:22 +02:00
|
|
|
static bool filter_keys(json_engine_t *je1, String *str, HASH items)
|
MDEV-26182: Implement JSON_INTERSECT()
The idea is to have simple functions that the user can combine to produce
the exact result one wants, whether the user wants JSON object that has
common keys with another JSON object, or same key/value pair etc. So
making simpler function helps here.
We accomplish this by making three separate functions.
1) JSON_OBJECT_FILTER_KEYS(Obj, Arr_keys):
Put keys ( which are basically strings ) in hash, go over the object and
get key one by one. If the key is present in the hash,
add the key-value pair to result.
2) JSON_OBJECT_TO_ARRAY(Obj) : Create a string variable, Go over the json
object, and add each key value pair as an array into the result.
3) JSON_ARRAY_INTERSECT(arr1, arr2) :
Go over one of the json and add each item of the array
in hash (after normalizing each item). Go over the second array,
search the normalized item one by one in the hash. If item is found,
add it to the result.
Implementation Idea: Holyfoot ( Alexey Botchkov)
Author: tanruixiang and Rucha Deodhar
2022-06-16 15:05:35 +08:00
|
|
|
{
|
|
|
|
int level= je1->stack_p;
|
|
|
|
String temp_str(0);
|
|
|
|
bool res= true, has_value= false;
|
|
|
|
|
|
|
|
temp_str.append('{');
|
|
|
|
while (json_scan_next(je1)==0 && level <= je1->stack_p)
|
|
|
|
{
|
|
|
|
switch(je1->state)
|
|
|
|
{
|
|
|
|
case JST_KEY:
|
|
|
|
{
|
|
|
|
const uchar *key_start= je1->s.c_str;
|
|
|
|
const uchar *key_end;
|
|
|
|
String str("", 0, je1->s.cs);
|
|
|
|
str.append('"');
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
key_end= je1->s.c_str;
|
|
|
|
} while (json_read_keyname_chr(je1) == 0);
|
|
|
|
|
|
|
|
if (unlikely(je1->s.error))
|
|
|
|
goto error;
|
|
|
|
if (json_read_value(je1))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
const uchar* value_start= NULL;
|
|
|
|
size_t value_len= 0;
|
|
|
|
if (get_current_value(je1, value_start, value_len))
|
|
|
|
goto error;
|
|
|
|
|
|
|
|
str.append((const char*)key_start, (size_t)(key_end-key_start));
|
|
|
|
str.append('"');
|
|
|
|
str.append('\0');
|
|
|
|
|
|
|
|
char *curr_key= (char*)malloc((size_t)(key_end-key_start+3));
|
|
|
|
strncpy(curr_key, str.ptr(), str.length());
|
|
|
|
|
|
|
|
if (my_hash_search(&items, (const uchar*)curr_key, strlen(curr_key)))
|
|
|
|
{
|
|
|
|
has_value= true;
|
|
|
|
|
|
|
|
temp_str.append('"');
|
|
|
|
temp_str.append((const char*)key_start, (size_t)(key_end-key_start));
|
|
|
|
temp_str.append('"');
|
|
|
|
|
|
|
|
temp_str.append(':');
|
|
|
|
|
|
|
|
temp_str.append((const char*)value_start, value_len);
|
|
|
|
|
|
|
|
temp_str.append(',');
|
|
|
|
}
|
|
|
|
free(curr_key);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
res= false;
|
|
|
|
|
|
|
|
if (has_value)
|
|
|
|
{
|
|
|
|
temp_str.chop();
|
|
|
|
temp_str.append('}');
|
|
|
|
str->append(temp_str.ptr(), temp_str.length());
|
|
|
|
}
|
|
|
|
|
|
|
|
error:
|
|
|
|
return res;
|
|
|
|
}
|
|
|
|
|
|
|
|
String* Item_func_json_object_filter_keys::val_str(String *str)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(fixed());
|
|
|
|
|
|
|
|
json_engine_t je1, res_je;
|
|
|
|
String *js1= args[0]->val_json(&tmp_js1);
|
|
|
|
|
|
|
|
if (null_value || args[0]->null_value)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
str->set_charset(js1->charset());
|
|
|
|
str->length(0);
|
|
|
|
|
|
|
|
json_scan_start(&je1, js1->charset(),(const uchar *) js1->ptr(),
|
|
|
|
(const uchar *) js1->ptr() + js1->length());
|
|
|
|
|
|
|
|
if (json_read_value(&je1) || je1.value_type != JSON_VALUE_OBJECT)
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
if(filter_keys(&je1, str, items))
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
if (str->length())
|
|
|
|
{
|
|
|
|
json_scan_start(&res_je, str->charset(), (const uchar *) str->ptr(),
|
|
|
|
(const uchar *) str->ptr() + str->length());
|
|
|
|
str= &tmp_js1;
|
|
|
|
if (json_nice(&res_je, str, Item_func_json_format::LOOSE))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
null_value= 0;
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
error_return:
|
|
|
|
if (je1.s.error)
|
|
|
|
report_json_error(js1, &je1, 0);
|
|
|
|
null_return:
|
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_func_json_object_filter_keys::fix_length_and_dec(THD *thd)
|
|
|
|
{
|
|
|
|
String *js2= args[1]->val_json(&tmp_js2);
|
|
|
|
json_engine_t je2;
|
|
|
|
|
|
|
|
if (args[1]->null_value)
|
|
|
|
{
|
|
|
|
null_value= 1;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
json_scan_start(&je2, js2->charset(),(const uchar *) js2->ptr(),
|
|
|
|
(const uchar *) js2->ptr() + js2->length());
|
2023-06-17 19:00:56 +02:00
|
|
|
if (!root_inited)
|
|
|
|
init_alloc_root(PSI_NOT_INSTRUMENTED, &hash_root, 1024, 0, MYF(0));
|
MDEV-26182: Implement JSON_INTERSECT()
The idea is to have simple functions that the user can combine to produce
the exact result one wants, whether the user wants JSON object that has
common keys with another JSON object, or same key/value pair etc. So
making simpler function helps here.
We accomplish this by making three separate functions.
1) JSON_OBJECT_FILTER_KEYS(Obj, Arr_keys):
Put keys ( which are basically strings ) in hash, go over the object and
get key one by one. If the key is present in the hash,
add the key-value pair to result.
2) JSON_OBJECT_TO_ARRAY(Obj) : Create a string variable, Go over the json
object, and add each key value pair as an array into the result.
3) JSON_ARRAY_INTERSECT(arr1, arr2) :
Go over one of the json and add each item of the array
in hash (after normalizing each item). Go over the second array,
search the normalized item one by one in the hash. If item is found,
add it to the result.
Implementation Idea: Holyfoot ( Alexey Botchkov)
Author: tanruixiang and Rucha Deodhar
2022-06-16 15:05:35 +08:00
|
|
|
root_inited= true;
|
|
|
|
|
|
|
|
if (json_read_value(&je2) || je2.value_type != JSON_VALUE_ARRAY ||
|
|
|
|
create_hash(&je2, &items, hash_inited, &hash_root))
|
|
|
|
{
|
|
|
|
if (je2.s.error)
|
|
|
|
report_json_error(js2, &je2, 0);
|
|
|
|
null_value= 1;
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
|
|
|
max_length= args[0]->max_length;
|
|
|
|
set_maybe_null();
|
|
|
|
|
|
|
|
return FALSE;
|
|
|
|
}
|
|
|
|
|
2023-06-17 18:37:22 +02:00
|
|
|
static bool convert_to_array(json_engine_t *je, String *str)
|
MDEV-26182: Implement JSON_INTERSECT()
The idea is to have simple functions that the user can combine to produce
the exact result one wants, whether the user wants JSON object that has
common keys with another JSON object, or same key/value pair etc. So
making simpler function helps here.
We accomplish this by making three separate functions.
1) JSON_OBJECT_FILTER_KEYS(Obj, Arr_keys):
Put keys ( which are basically strings ) in hash, go over the object and
get key one by one. If the key is present in the hash,
add the key-value pair to result.
2) JSON_OBJECT_TO_ARRAY(Obj) : Create a string variable, Go over the json
object, and add each key value pair as an array into the result.
3) JSON_ARRAY_INTERSECT(arr1, arr2) :
Go over one of the json and add each item of the array
in hash (after normalizing each item). Go over the second array,
search the normalized item one by one in the hash. If item is found,
add it to the result.
Implementation Idea: Holyfoot ( Alexey Botchkov)
Author: tanruixiang and Rucha Deodhar
2022-06-16 15:05:35 +08:00
|
|
|
{
|
|
|
|
int level= je->stack_p;
|
|
|
|
String temp_str(0);
|
|
|
|
temp_str.append('[');
|
|
|
|
|
|
|
|
while (json_scan_next(je)==0 && level <= je->stack_p)
|
|
|
|
{
|
|
|
|
switch(je->state)
|
|
|
|
{
|
|
|
|
case JST_KEY:
|
|
|
|
{
|
|
|
|
temp_str.append('[');
|
|
|
|
|
|
|
|
const uchar *key_start= je->s.c_str;
|
|
|
|
const uchar *key_end;
|
|
|
|
|
|
|
|
do
|
|
|
|
{
|
|
|
|
key_end= je->s.c_str;
|
|
|
|
} while (json_read_keyname_chr(je) == 0);
|
|
|
|
|
|
|
|
if (unlikely(je->s.error))
|
|
|
|
return true;
|
|
|
|
|
|
|
|
temp_str.append('"');
|
|
|
|
temp_str.append((const char*)key_start, (size_t)(key_end-key_start));
|
|
|
|
temp_str.append('"');
|
|
|
|
|
|
|
|
temp_str.append(',');
|
|
|
|
|
|
|
|
int v_len= 0;
|
|
|
|
const uchar *value= NULL;
|
|
|
|
|
|
|
|
if (json_read_value(je))
|
|
|
|
return true;
|
|
|
|
value= je->value_begin;
|
|
|
|
if (json_value_scalar(je))
|
|
|
|
v_len= (int)(je->value_end - value);
|
|
|
|
else
|
|
|
|
{
|
|
|
|
if (json_skip_level(je))
|
|
|
|
return true;
|
|
|
|
v_len= (int)(je->s.c_str - value);
|
|
|
|
}
|
|
|
|
temp_str.append((const char *) value, v_len);
|
|
|
|
|
|
|
|
temp_str.append(']');
|
|
|
|
temp_str.append(',');
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (je->s.error)
|
|
|
|
return true;
|
|
|
|
|
|
|
|
temp_str.chop(); /* remove the last comma. */
|
|
|
|
temp_str.append(']');
|
|
|
|
str->append(temp_str.ptr(), temp_str.length());
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
String* Item_func_json_object_to_array::val_str(String *str)
|
|
|
|
{
|
|
|
|
DBUG_ASSERT(fixed());
|
|
|
|
|
|
|
|
json_engine_t je;
|
|
|
|
String *js1= args[0]->val_str(&tmp);
|
|
|
|
|
|
|
|
if (args[0]->null_value)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
str->set_charset(js1->charset());
|
|
|
|
str->length(0);
|
|
|
|
|
|
|
|
json_scan_start(&je, js1->charset(),(const uchar *) js1->ptr(),
|
|
|
|
(const uchar *) js1->ptr() + js1->length());
|
|
|
|
|
|
|
|
if (json_read_value(&je))
|
|
|
|
goto error_return;
|
|
|
|
if (je.value_type != JSON_VALUE_OBJECT)
|
|
|
|
goto null_return;
|
|
|
|
|
|
|
|
if (convert_to_array(&je, str))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
if (str->length())
|
|
|
|
{
|
|
|
|
json_scan_start(&je, str->charset(), (const uchar *) str->ptr(),
|
|
|
|
(const uchar *) str->ptr() + str->length());
|
|
|
|
str= &tmp;
|
|
|
|
if (json_nice(&je, str, Item_func_json_format::LOOSE))
|
|
|
|
goto error_return;
|
|
|
|
|
|
|
|
null_value= 0;
|
|
|
|
return str;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
goto null_return;
|
|
|
|
}
|
|
|
|
|
|
|
|
error_return:
|
|
|
|
if (je.s.error)
|
|
|
|
report_json_error(js1, &je, 0);
|
|
|
|
null_return:
|
|
|
|
null_value= 1;
|
|
|
|
return NULL;
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
bool Item_func_json_object_to_array::fix_length_and_dec(THD *thd)
|
|
|
|
{
|
|
|
|
max_length= args[0]->max_length + (args[0]->max_length/2);
|
|
|
|
set_maybe_null();
|
|
|
|
return FALSE;
|
|
|
|
}
|