Select.cpp 6.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183
  1. /*
  2. * Copyright (c) 2021, Jan de Visser <jan@de-visser.net>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/NumericLimits.h>
  7. #include <LibSQL/AST/AST.h>
  8. #include <LibSQL/Database.h>
  9. #include <LibSQL/Meta.h>
  10. #include <LibSQL/Row.h>
  11. namespace SQL::AST {
  12. static ByteString result_column_name(ResultColumn const& column, size_t column_index)
  13. {
  14. auto fallback_column_name = [column_index]() {
  15. return ByteString::formatted("Column{}", column_index);
  16. };
  17. if (auto const& alias = column.column_alias(); !alias.is_empty())
  18. return alias;
  19. if (column.select_from_expression()) {
  20. if (is<ColumnNameExpression>(*column.expression())) {
  21. auto const& column_name_expression = verify_cast<ColumnNameExpression>(*column.expression());
  22. return column_name_expression.column_name();
  23. }
  24. // FIXME: Generate column names from other result column expressions.
  25. return fallback_column_name();
  26. }
  27. VERIFY(column.select_from_table());
  28. // FIXME: Generate column names from select-from-table result columns.
  29. return fallback_column_name();
  30. }
  31. ResultOr<ResultSet> Select::execute(ExecutionContext& context) const
  32. {
  33. Vector<NonnullRefPtr<ResultColumn const>> columns;
  34. Vector<ByteString> column_names;
  35. auto const& result_column_list = this->result_column_list();
  36. VERIFY(!result_column_list.is_empty());
  37. for (auto& table_descriptor : table_or_subquery_list()) {
  38. if (!table_descriptor->is_table())
  39. return Result { SQLCommand::Select, SQLErrorCode::NotYetImplemented, "Sub-selects are not yet implemented"sv };
  40. auto table_def = TRY(context.database->get_table(table_descriptor->schema_name(), table_descriptor->table_name()));
  41. if (result_column_list.size() == 1 && result_column_list[0]->type() == ResultType::All) {
  42. TRY(columns.try_ensure_capacity(columns.size() + table_def->columns().size()));
  43. TRY(column_names.try_ensure_capacity(column_names.size() + table_def->columns().size()));
  44. for (auto& col : table_def->columns()) {
  45. columns.unchecked_append(
  46. create_ast_node<ResultColumn>(
  47. create_ast_node<ColumnNameExpression>(table_def->parent()->name(), table_def->name(), col->name()),
  48. ""));
  49. column_names.unchecked_append(col->name());
  50. }
  51. }
  52. }
  53. if (result_column_list.size() != 1 || result_column_list[0]->type() != ResultType::All) {
  54. TRY(columns.try_ensure_capacity(result_column_list.size()));
  55. TRY(column_names.try_ensure_capacity(result_column_list.size()));
  56. for (size_t i = 0; i < result_column_list.size(); ++i) {
  57. auto const& col = result_column_list[i];
  58. if (col->type() == ResultType::All) {
  59. // FIXME can have '*' for example in conjunction with computed columns
  60. return Result { SQLCommand::Select, SQLErrorCode::SyntaxError, "*"sv };
  61. }
  62. columns.unchecked_append(col);
  63. column_names.unchecked_append(result_column_name(col, i));
  64. }
  65. }
  66. ResultSet result { SQLCommand::Select, move(column_names) };
  67. auto descriptor = adopt_ref(*new TupleDescriptor);
  68. Tuple tuple(descriptor);
  69. Vector<Tuple> rows;
  70. descriptor->empend("__unity__"sv);
  71. tuple.append(Value { true });
  72. rows.append(tuple);
  73. for (auto& table_descriptor : table_or_subquery_list()) {
  74. if (!table_descriptor->is_table())
  75. return Result { SQLCommand::Select, SQLErrorCode::NotYetImplemented, "Sub-selects are not yet implemented"sv };
  76. auto table_def = TRY(context.database->get_table(table_descriptor->schema_name(), table_descriptor->table_name()));
  77. if (table_def->num_columns() == 0)
  78. continue;
  79. auto old_descriptor_size = descriptor->size();
  80. descriptor->extend(table_def->to_tuple_descriptor());
  81. while (!rows.is_empty() && (rows.first().size() == old_descriptor_size)) {
  82. auto cartesian_row = rows.take_first();
  83. auto table_rows = TRY(context.database->select_all(*table_def));
  84. for (auto& table_row : table_rows) {
  85. auto new_row = cartesian_row;
  86. new_row.extend(table_row);
  87. rows.append(new_row);
  88. }
  89. }
  90. }
  91. bool has_ordering { false };
  92. auto sort_descriptor = adopt_ref(*new TupleDescriptor);
  93. for (auto& term : m_ordering_term_list) {
  94. sort_descriptor->append(TupleElementDescriptor { .order = term->order() });
  95. has_ordering = true;
  96. }
  97. Tuple sort_key(sort_descriptor);
  98. for (auto& row : rows) {
  99. context.current_row = &row;
  100. if (where_clause()) {
  101. auto where_result = TRY(where_clause()->evaluate(context)).to_bool();
  102. if (!where_result.has_value() || !where_result.value())
  103. continue;
  104. }
  105. tuple.clear();
  106. for (auto& col : columns) {
  107. auto value = TRY(col->expression()->evaluate(context));
  108. tuple.append(value);
  109. }
  110. if (has_ordering) {
  111. sort_key.clear();
  112. for (auto& term : m_ordering_term_list) {
  113. auto value = TRY(term->expression()->evaluate(context));
  114. sort_key.append(value);
  115. }
  116. }
  117. result.insert_row(tuple, sort_key);
  118. }
  119. if (m_limit_clause != nullptr) {
  120. size_t limit_value = NumericLimits<size_t>::max();
  121. size_t offset_value = 0;
  122. auto limit = TRY(m_limit_clause->limit_expression()->evaluate(context));
  123. if (!limit.is_null()) {
  124. auto limit_value_maybe = limit.to_int<size_t>();
  125. if (!limit_value_maybe.has_value())
  126. return Result { SQLCommand::Select, SQLErrorCode::SyntaxError, "LIMIT clause must evaluate to an integer value"sv };
  127. limit_value = limit_value_maybe.value();
  128. }
  129. if (m_limit_clause->offset_expression() != nullptr) {
  130. auto offset = TRY(m_limit_clause->offset_expression()->evaluate(context));
  131. if (!offset.is_null()) {
  132. auto offset_value_maybe = offset.to_int<size_t>();
  133. if (!offset_value_maybe.has_value())
  134. return Result { SQLCommand::Select, SQLErrorCode::SyntaxError, "OFFSET clause must evaluate to an integer value"sv };
  135. offset_value = offset_value_maybe.value();
  136. }
  137. }
  138. result.limit(offset_value, limit_value);
  139. }
  140. return result;
  141. }
  142. }