/* Copyright (c) 2021 OceanBase and/or its affiliates. All rights reserved. miniob is licensed under Mulan PSL v2. You can use this software according to the terms and conditions of the Mulan PSL v2. You may obtain a copy of Mulan PSL v2 at: http://license.coscl.org.cn/MulanPSL2 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 PSL v2 for more details. */ // // Created by WangYunlai on 2022/07/05. // #include #include "sql/expr/tuple_cell.h" #include "storage/common/field.h" #include "common/log/log.h" #include "common/lang/comparator.h" #include "common/lang/string.h" TupleCellSpec::TupleCellSpec(const char *table_name, const char *field_name, const char *alias) { if (table_name) { table_name_ = table_name; } if (field_name) { field_name_ = field_name; } if (alias) { alias_ = alias; } else { if (table_name_.empty()) { alias_ = field_name_; } else { alias_ = table_name_ + "." + field_name_; } } } TupleCellSpec::TupleCellSpec(const char *alias) { if (alias) { alias_ = alias; } } //////////////////////////////////////////////////////////////////////////////// void TupleCell::set_data(char *data, int length) { switch (attr_type_) { case CHARS: { set_string(data, length); } break; case INTS: { num_value_.int_value_ = *(int *)data; length_ = length; } break; case FLOATS: { num_value_.float_value_ = *(float *)data; length_ = length; } break; case BOOLEANS: { num_value_.bool_value_ = *(int *)data != 0; length_ = length; } break; default: { LOG_WARN("unknown data type: %d", attr_type_); } break; } } void TupleCell::set_int(int val) { attr_type_ = INTS; num_value_.int_value_ = val; length_ = sizeof(val); } void TupleCell::set_float(float val) { attr_type_ = FLOATS; num_value_.float_value_ = val; length_ = sizeof(val); } void TupleCell::set_boolean(bool val) { attr_type_ = BOOLEANS; num_value_.bool_value_ = val; length_ = sizeof(val); } void TupleCell::set_string(const char *s, int len /*= 0*/) { attr_type_ = CHARS; if (len > 0) { len = strnlen(s, len); str_value_.assign(s, len); } else { str_value_.assign(s); } length_ = str_value_.length(); } void TupleCell::set_value(const Value &value) { switch (value.type) { case INTS: { set_int(value.int_value); } break; case FLOATS: { set_float(value.float_value); } break; case CHARS: { set_string(value.string_value.c_str()); } break; case BOOLEANS: { set_boolean(value.bool_value); } break; case UNDEFINED: { ASSERT(false, "got an invalid value type"); } break; } } const char *TupleCell::data() const { switch (attr_type_) { case CHARS: { return str_value_.c_str(); } break; default: { return (const char *)&num_value_; } break; } } void TupleCell::to_string(std::ostream &os) const { switch (attr_type_) { case INTS: { os << num_value_.int_value_; } break; case FLOATS: { os << common::double_to_str(num_value_.float_value_); } break; case BOOLEANS: { os << num_value_.bool_value_; } break; case CHARS: { os << str_value_; } break; default: { LOG_WARN("unsupported attr type: %d", attr_type_); } break; } } int TupleCell::compare(const TupleCell &other) const { if (this->attr_type_ == other.attr_type_) { switch (this->attr_type_) { case INTS: { return common::compare_int((void *)&this->num_value_.int_value_, (void *)&other.num_value_.int_value_); } break; case FLOATS: { return common::compare_float((void *)&this->num_value_.float_value_, (void *)&other.num_value_.float_value_); } break; case CHARS: { return common::compare_string((void *)this->str_value_.c_str(), this->str_value_.length(), (void *)other.str_value_.c_str(), other.str_value_.length()); } break; case BOOLEANS: { return common::compare_int((void *)&this->num_value_.bool_value_, (void *)&other.num_value_.bool_value_); } default: { LOG_WARN("unsupported type: %d", this->attr_type_); } } } else if (this->attr_type_ == INTS && other.attr_type_ == FLOATS) { float this_data = this->num_value_.int_value_; return common::compare_float((void *)&this_data, (void *)&other.num_value_.float_value_); } else if (this->attr_type_ == FLOATS && other.attr_type_ == INTS) { float other_data = other.num_value_.int_value_; return common::compare_float((void *)&this->num_value_.float_value_, (void *)&other_data); } LOG_WARN("not supported"); return -1; // TODO return rc? } int TupleCell::get_int() const { switch (attr_type_) { case CHARS: { try { return (int)(std::stol(str_value_)); } catch (std::exception const &ex) { LOG_TRACE("failed to convert string to number. s=%s, ex=%s", str_value_.c_str(), ex.what()); return 0; } } case INTS: { return num_value_.int_value_; } case FLOATS: { return (int)(num_value_.float_value_); } case BOOLEANS: { return (int)(num_value_.bool_value_); } default: { LOG_WARN("unknown data type. type=%d", attr_type_); return 0; } } return 0; } float TupleCell::get_float() const { switch (attr_type_) { case CHARS: { try { return std::stof(str_value_); } catch (std::exception const &ex) { LOG_TRACE("failed to convert string to float. s=%s, ex=%s", str_value_.c_str(), ex.what()); return 0.0; } } break; case INTS: { return float(num_value_.int_value_); } break; case FLOATS: { return num_value_.float_value_; } break; case BOOLEANS: { return float(num_value_.bool_value_); } break; default: { LOG_WARN("unknown data type. type=%d", attr_type_); return 0; } } return 0; } std::string TupleCell::get_string() const { std::stringstream ss; to_string(ss); return ss.str(); } bool TupleCell::get_boolean() const { switch (attr_type_) { case CHARS: { try { float val = std::stof(str_value_); if (val >= EPSILON || val <= -EPSILON) { return true; } int int_val = std::stol(str_value_); if (int_val != 0) { return true; } return !str_value_.empty(); } catch (std::exception const &ex) { LOG_TRACE("failed to convert string to float or integer. s=%s, ex=%s", str_value_.c_str(), ex.what()); return !str_value_.empty(); } } break; case INTS: { return num_value_.int_value_ != 0; } break; case FLOATS: { float val = num_value_.float_value_; return val >= EPSILON || val <= -EPSILON; } break; case BOOLEANS: { return num_value_.bool_value_; } break; default: { LOG_WARN("unknown data type. type=%d", attr_type_); return false; } } return false; }