support json type
This commit is contained in:
266
src/sql/engine/expr/ob_expr_json_overlaps.cpp
Normal file
266
src/sql/engine/expr/ob_expr_json_overlaps.cpp
Normal file
@ -0,0 +1,266 @@
|
||||
/**
|
||||
* Copyright (c) 2021 OceanBase
|
||||
* OceanBase CE is licensed under Mulan PubL v2.
|
||||
* You can use this software according to the terms and conditions of the Mulan PubL v2.
|
||||
* You may obtain a copy of Mulan PubL v2 at:
|
||||
* http://license.coscl.org.cn/MulanPubL-2.0
|
||||
* THIS SOFTWARE IS PROVIDED ON AN "AS IS" BASIS, WITHOUT WARRANTIES OF ANY KIND,
|
||||
* EITHER EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO NON-INFRINGEMENT,
|
||||
* MERCHANTABILITY OR FIT FOR A PARTICULAR PURPOSE.
|
||||
* See the Mulan PubL v2 for more details.
|
||||
*/
|
||||
|
||||
// This file contains implementation for json_overlaps.
|
||||
#define USING_LOG_PREFIX SQL_ENG
|
||||
#include "ob_expr_json_overlaps.h"
|
||||
#include "ob_expr_json_func_helper.h"
|
||||
|
||||
using namespace oceanbase::common;
|
||||
using namespace oceanbase::sql;
|
||||
|
||||
namespace oceanbase
|
||||
{
|
||||
namespace sql
|
||||
{
|
||||
|
||||
ObExprJsonOverlaps::ObExprJsonOverlaps(ObIAllocator &alloc)
|
||||
: ObFuncExprOperator(alloc, T_FUN_SYS_JSON_OVERLAPS, N_JSON_OVERLAPS, 2, NOT_ROW_DIMENSION)
|
||||
{
|
||||
}
|
||||
|
||||
ObExprJsonOverlaps::~ObExprJsonOverlaps()
|
||||
{
|
||||
}
|
||||
|
||||
int ObExprJsonOverlaps::calc_result_type2(ObExprResType &type,
|
||||
ObExprResType &type1,
|
||||
ObExprResType &type2,
|
||||
ObExprTypeCtx &type_ctx) const
|
||||
{
|
||||
UNUSED(type_ctx);
|
||||
int ret = OB_SUCCESS;
|
||||
|
||||
type.set_int32();
|
||||
type.set_precision(DEFAULT_PRECISION_FOR_BOOL);
|
||||
type.set_scale(ObAccuracy::DDL_DEFAULT_ACCURACY[ObIntType].scale_);
|
||||
|
||||
if (OB_FAIL(ObJsonExprHelper::is_valid_for_json(type1, 1, N_JSON_OVERLAPS))) {
|
||||
LOG_WARN("wrong type for json doc.", K(ret), K(type1.get_type()));
|
||||
} else if (OB_FAIL(ObJsonExprHelper::is_valid_for_json(type2, 2, N_JSON_OVERLAPS))) {
|
||||
LOG_WARN("wrong type for json doc.", K(ret), K(type2.get_type()));
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprJsonOverlaps::calc_result2(common::ObObj &result,
|
||||
const common::ObObj &obj1,
|
||||
const common::ObObj &obj2,
|
||||
common::ObExprCtx &expr_ctx) const
|
||||
{
|
||||
INIT_SUCC(ret);
|
||||
ObIAllocator *allocator = expr_ctx.calc_buf_;
|
||||
|
||||
if (OB_ISNULL(allocator)) { // check allocator
|
||||
ret = OB_NOT_INIT;
|
||||
LOG_WARN("varchar buffer not init", K(ret));
|
||||
} else {
|
||||
ObIJsonBase *json_a = NULL;
|
||||
ObIJsonBase *json_b = NULL;
|
||||
bool is_null_result = false;
|
||||
if (OB_FAIL(ObJsonExprHelper::get_json_doc(&obj1, allocator, 0,
|
||||
json_a, is_null_result, false))) {
|
||||
LOG_WARN("get_json_doc failed", K(ret));
|
||||
} else if (OB_FAIL(ObJsonExprHelper::get_json_doc(&obj2, allocator, 0,
|
||||
json_b, is_null_result, false))) {
|
||||
LOG_WARN("get_json_doc failed", K(ret));
|
||||
} else {
|
||||
bool is_overlaps = false;
|
||||
if (!is_null_result) {
|
||||
if (OB_FAIL(json_overlaps(json_a, json_b, &is_overlaps))) {
|
||||
LOG_WARN("json_overlaps in sub_json_targets failed", K(ret));
|
||||
}
|
||||
}
|
||||
// set result
|
||||
if (OB_FAIL(ret)) {
|
||||
LOG_WARN("json_overlaps failed", K(ret));
|
||||
} else if (is_null_result) {
|
||||
result.set_null();
|
||||
} else {
|
||||
result.set_int(static_cast<int64_t>(is_overlaps));
|
||||
}
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprJsonOverlaps::eval_json_overlaps(const ObExpr &expr, ObEvalCtx &ctx, ObDatum &res)
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
ObIJsonBase *json_a = NULL;
|
||||
ObIJsonBase *json_b = NULL;
|
||||
bool is_null_result = false;
|
||||
common::ObArenaAllocator &temp_allocator = ctx.get_reset_tmp_alloc();
|
||||
if (OB_FAIL(ObJsonExprHelper::get_json_doc(expr, ctx, temp_allocator, 0, json_a, is_null_result, false))) {
|
||||
LOG_WARN("get_json_doc failed", K(ret));
|
||||
} else if (OB_FAIL(ObJsonExprHelper::get_json_doc(expr, ctx, temp_allocator, 1, json_b, is_null_result, false))) {
|
||||
LOG_WARN("get_json_doc failed", K(ret));
|
||||
} else {
|
||||
bool is_overlaps = false;
|
||||
if (!is_null_result) {
|
||||
if (OB_FAIL(json_overlaps(json_a, json_b, &is_overlaps))) {
|
||||
LOG_WARN("json_overlaps in sub_json_targets failed", K(ret));
|
||||
}
|
||||
|
||||
}
|
||||
|
||||
// set result
|
||||
if (OB_FAIL(ret)) {
|
||||
LOG_WARN("json_overlaps failed", K(ret));
|
||||
} else if (is_null_result) {
|
||||
res.set_null();
|
||||
} else {
|
||||
res.set_int(static_cast<int64_t>(is_overlaps));
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprJsonOverlaps::cg_expr(ObExprCGCtx &expr_cg_ctx,
|
||||
const ObRawExpr &raw_expr,
|
||||
ObExpr &rt_expr) const
|
||||
{
|
||||
UNUSED(expr_cg_ctx);
|
||||
UNUSED(raw_expr);
|
||||
rt_expr.eval_func_ = eval_json_overlaps;
|
||||
return OB_SUCCESS;
|
||||
}
|
||||
|
||||
int ObExprJsonOverlaps::json_overlaps_object(ObIJsonBase *json_a,
|
||||
ObIJsonBase *json_b,
|
||||
bool *result)
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
if (json_b->json_type() != ObJsonNodeType::J_OBJECT) {
|
||||
*result = false;
|
||||
} else if (json_a->element_count() == 0 && json_b->element_count() == 0) {
|
||||
*result = true;
|
||||
} else {
|
||||
JsonObjectIterator iter_a = json_a->object_iterator();
|
||||
JsonObjectIterator iter_b = json_b->object_iterator();
|
||||
while (!iter_b.empty() && OB_SUCC(ret)) {
|
||||
ObString key_b;
|
||||
ObIJsonBase *a_tmp = NULL;
|
||||
ObIJsonBase *b_tmp = NULL;
|
||||
if (OB_FAIL(iter_b.get_key(key_b))) {
|
||||
LOG_WARN("fail to get key from iterator", K(ret));
|
||||
} else if(OB_FAIL(iter_a.get_value(key_b, a_tmp))) {
|
||||
if (ret == OB_SEARCH_NOT_FOUND) {
|
||||
ret = OB_SUCCESS;
|
||||
} else {
|
||||
LOG_WARN("fail to get object_value from wrapper", K(ret));
|
||||
}
|
||||
} else if (OB_FAIL(iter_b.get_value(b_tmp))) {
|
||||
LOG_WARN("fail to get value from iterator", K(ret));
|
||||
} else {
|
||||
int tmp_result;
|
||||
if (OB_FAIL(a_tmp->compare(*b_tmp, tmp_result))) {
|
||||
LOG_WARN("json_overlaps_object fail to compare with object type", K(ret));
|
||||
}
|
||||
if (tmp_result == 0) {
|
||||
*result = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
iter_b.next();
|
||||
}
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprJsonOverlaps::json_overlaps_array(ObIJsonBase *json_a,
|
||||
ObIJsonBase *json_b,
|
||||
bool *result)
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
bool ret_tmp = true;
|
||||
|
||||
ObIAllocator *allocator = json_a->get_allocator();
|
||||
ObJsonArray tmp_arr(allocator);
|
||||
if (json_b->json_type() != ObJsonNodeType::J_ARRAY) {
|
||||
// convert to array if needed
|
||||
ObIJsonBase *jb_node = NULL;
|
||||
if (ObJsonBaseFactory::transform(allocator, json_b, ObJsonInType::JSON_TREE, jb_node)) {
|
||||
LOG_WARN("fail to transform to tree", K(ret), K(*json_b));
|
||||
} else {
|
||||
ObJsonNode *j_node = static_cast<ObJsonNode *>(jb_node);
|
||||
if (OB_FAIL(tmp_arr.array_append(j_node->clone(allocator)))) {
|
||||
LOG_WARN("result array append failed", K(ret), K(*j_node));
|
||||
} else {
|
||||
json_b = &tmp_arr;
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
ObSortedVector<ObIJsonBase *> vec_a;
|
||||
if (OB_SUCC(ret) && OB_FAIL(ObJsonBaseUtil::sort_array_pointer(json_a, vec_a))) {
|
||||
LOG_WARN("sort_array_pointer failed.", K(ret));
|
||||
} else {
|
||||
uint64_t b_len = json_b->element_count();
|
||||
for (uint64_t i = 0; i < b_len; i++) {
|
||||
ObIJsonBase *b_tmp = NULL;
|
||||
if (OB_FAIL(json_b->get_array_element(i, b_tmp))) {
|
||||
LOG_WARN("fail to get_array_element",K(ret), K(i));
|
||||
} else if (ObJsonBaseUtil::binary_search(vec_a, b_tmp)) {
|
||||
*result = true;
|
||||
break;
|
||||
}
|
||||
}
|
||||
}
|
||||
return ret;
|
||||
}
|
||||
|
||||
int ObExprJsonOverlaps::json_overlaps(ObIJsonBase *json_a,
|
||||
ObIJsonBase *json_b,
|
||||
bool *result)
|
||||
{
|
||||
int ret = OB_SUCCESS;
|
||||
|
||||
// make sure json_a is array.
|
||||
if (json_a->json_type() != ObJsonNodeType::J_ARRAY && json_b->json_type() == ObJsonNodeType::J_ARRAY) {
|
||||
std::swap(json_a, json_b);
|
||||
}
|
||||
|
||||
// make sure json_a has bigger size.
|
||||
if (json_a->json_type() == ObJsonNodeType::J_ARRAY
|
||||
&& json_b->json_type() == ObJsonNodeType::J_ARRAY
|
||||
&& json_a->element_count() < json_b->element_count()) {
|
||||
std::swap(json_a, json_b);
|
||||
}
|
||||
switch (json_a->json_type()) {
|
||||
case ObJsonNodeType::J_ARRAY:
|
||||
if (OB_FAIL( json_overlaps_array(json_a, json_b, result))) {
|
||||
LOG_WARN("fail to json_overlaps with ARRAY type", K(ret));
|
||||
}
|
||||
break;
|
||||
|
||||
case ObJsonNodeType::J_OBJECT:
|
||||
if (OB_FAIL( json_overlaps_object(json_a, json_b, result))) {
|
||||
LOG_WARN("fail to json_overlaps with OBJECT type", K(ret));
|
||||
}
|
||||
break;
|
||||
|
||||
default:
|
||||
int res_int = -1;
|
||||
if (OB_FAIL( json_a->compare(*json_b, res_int))) {
|
||||
LOG_WARN("fail to json_overlaps with other type", K(ret));
|
||||
}
|
||||
*result = (res_int == 0);
|
||||
}
|
||||
|
||||
return ret;
|
||||
}
|
||||
|
||||
}
|
||||
}
|
||||
Reference in New Issue
Block a user