#include "row.h" namespace usql { int ColNullValue::compare(ColValue * other) { return other->isNull() ? 0 : -1; // null goes to end } int ColIntegerValue::compare(ColValue * other) { return other->isNull() ? 1 : m_integer - other->getIntValue(); // TODO implicit conversion from long to int } int ColDoubleValue::compare(ColValue * other) { if (other->isNull()) return 1; // null goes to end double c = m_double - other->getDoubleValue(); return c < 0 ? -1 : c == 0.0 ? 0 : 1; } int ColStringValue::compare(ColValue * other) { return other->isNull() ? 1 : m_string.compare(other->getStringValue()); // null goes to end } int ColDateValue::compare(ColValue * other) { return other->isNull() ? 1 : m_date - other->getIntValue(); // TODO implicit conversion from long to int } int ColBooleanValue::compare(ColValue * other) { if (other->isNull()) return 1; // null goes to end return m_bool==other->getBoolValue() ? 0 : m_bool && !other->getBoolValue() ? -1 : 1; // true first } Row::Row(int cols_count) { m_columns.reserve(cols_count); for (int i = 0; i < cols_count; i++) { m_columns.push_back(std::make_unique()); } } Row::Row(const Row &other) { m_columns.reserve(other.m_columns.size()); // TODO fixme, here first set cols null and then immediately replace it for (int i = 0; i < other.m_columns.size(); i++) { m_columns.push_back(std::make_unique()); } // TODO get rid of dynamic_cast for (int i = 0; i < other.m_columns.size(); i++) { if (auto *other_v = dynamic_cast(other.m_columns[i].get())) { setIntColumnValue(i, other_v->getIntValue()); } else if (auto *other_v = dynamic_cast(other.m_columns[i].get())) { setFloatColumnValue(i, other_v->getDoubleValue()); } else if (auto *other_v = dynamic_cast(other.m_columns[i].get())) { setStringColumnValue(i, other_v->getStringValue()); } else if (auto *other_v = dynamic_cast(other.m_columns[i].get())) { setDateColumnValue(i, other_v->getDateValue()); } else if (auto *other_v = dynamic_cast(other.m_columns[i].get())) { setBoolColumnValue(i, other_v->getBoolValue()); } else if (auto *other_v = dynamic_cast(other.m_columns[i].get())) { // NOP } else { throw Exception("unsupported data type"); } } } Row &Row::operator=(Row other) { std::swap(m_columns, other.m_columns); return *this; } void Row::setColumnNull(int col_index) { m_columns[col_index] = std::make_unique(); } void Row::setIntColumnValue(int col_index, long value) { m_columns[col_index] = std::make_unique(value); } void Row::setFloatColumnValue(int col_index, double value) { m_columns[col_index] = std::make_unique(value); } void Row::setStringColumnValue(int col_index, const std::string &value) { m_columns[col_index] = std::make_unique(value); } void Row::setDateColumnValue(int col_index, long value) { m_columns[col_index] = std::make_unique(value); } void Row::setDateColumnValue(int col_index, const std::string &value) { m_columns[col_index] = std::make_unique(Settings::string_to_date(value)); } void Row::setBoolColumnValue(int col_index, bool value) { m_columns[col_index] = std::make_unique(value); } void Row::setBoolColumnValue(int col_index, const std::string &value) { bool v = (value=="Y" || value=="1"); m_columns[col_index] = std::make_unique(v); } void Row::setColumnValue(ColDefNode *col_def, ColValue *col_value) { if (!col_value->isNull()) { if (col_def->type == ColumnType::integer_type) setIntColumnValue(col_def->order, col_value->getIntValue()); else if (col_def->type == ColumnType::float_type) setFloatColumnValue(col_def->order, col_value->getDoubleValue()); else if (col_def->type == ColumnType::varchar_type) setStringColumnValue(col_def->order, col_value->getStringValue()); else if (col_def->type == ColumnType::date_type) setDateColumnValue(col_def->order, col_value->getDateValue()); else if (col_def->type == ColumnType::bool_type) setBoolColumnValue(col_def->order, col_value->getBoolValue()); } else { setColumnNull(col_def->order); } } void Row::setColumnValue(ColDefNode *col_def, ValueNode *col_value) { if (!col_value->isNull()) { if (col_def->type == ColumnType::integer_type) setIntColumnValue(col_def->order, col_value->getIntegerValue()); else if (col_def->type == ColumnType::float_type) setFloatColumnValue(col_def->order, col_value->getDoubleValue()); else if (col_def->type == ColumnType::varchar_type) setStringColumnValue(col_def->order, col_value->getStringValue()); else if (col_def->type == ColumnType::date_type) setIntColumnValue(col_def->order, col_value->getDateValue()); else if (col_def->type == ColumnType::bool_type) setBoolColumnValue(col_def->order, col_value->getBooleanValue()); else throw Exception("unsupported data type"); } else { setColumnNull(col_def->order); } } int Row::compare(const Row & other) const { for (int ci = 0; ci < m_columns.size(); ci++) { int cmp = m_columns[ci]->compare(other.ith_column(ci)); if (cmp != 0) return cmp; } return 0; } void Row::print(const std::vector & col_char_sizes) { std::string out{"| "}; for (int ci = 0; ci < m_columns.size(); ci++) { auto value = m_columns[ci]->getStringValue(); // TODO use string functions out.append(value + std::string(col_char_sizes[ci] - value.size(), ' ') + " | "); } std::cout << out << std::endl; } }