Select.cpp 5.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155
  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/ResultSet.h>
  11. #include <LibSQL/Row.h>
  12. namespace SQL::AST {
  13. Result Select::execute(ExecutionContext& context) const
  14. {
  15. NonnullRefPtrVector<ResultColumn> columns;
  16. auto const& result_column_list = this->result_column_list();
  17. VERIFY(!result_column_list.is_empty());
  18. for (auto& table_descriptor : table_or_subquery_list()) {
  19. if (!table_descriptor.is_table())
  20. return { SQLCommand::Select, SQLErrorCode::NotYetImplemented, "Sub-selects are not yet implemented"sv };
  21. auto table_def = TRY(context.database->get_table(table_descriptor.schema_name(), table_descriptor.table_name()));
  22. if (!table_def)
  23. return { SQLCommand::Select, SQLErrorCode::TableDoesNotExist, table_descriptor.table_name() };
  24. if (result_column_list.size() == 1 && result_column_list[0].type() == ResultType::All) {
  25. for (auto& col : table_def->columns()) {
  26. columns.append(
  27. create_ast_node<ResultColumn>(
  28. create_ast_node<ColumnNameExpression>(table_def->parent()->name(), table_def->name(), col.name()),
  29. ""));
  30. }
  31. }
  32. }
  33. if (result_column_list.size() != 1 || result_column_list[0].type() != ResultType::All) {
  34. for (auto& col : result_column_list) {
  35. if (col.type() == ResultType::All) {
  36. // FIXME can have '*' for example in conjunction with computed columns
  37. return { SQLCommand::Select, SQLErrorCode::SyntaxError, "*"sv };
  38. }
  39. columns.append(col);
  40. }
  41. }
  42. context.result = Result { SQLCommand::Select };
  43. auto descriptor = adopt_ref(*new TupleDescriptor);
  44. Tuple tuple(descriptor);
  45. Vector<Tuple> rows;
  46. descriptor->empend("__unity__"sv);
  47. tuple.append(Value(SQLType::Boolean, true));
  48. rows.append(tuple);
  49. for (auto& table_descriptor : table_or_subquery_list()) {
  50. if (!table_descriptor.is_table())
  51. return { SQLCommand::Select, SQLErrorCode::NotYetImplemented, "Sub-selects are not yet implemented"sv };
  52. auto table_def = TRY(context.database->get_table(table_descriptor.schema_name(), table_descriptor.table_name()));
  53. if (table_def->num_columns() == 0)
  54. continue;
  55. auto old_descriptor_size = descriptor->size();
  56. descriptor->extend(table_def->to_tuple_descriptor());
  57. for (auto cartesian_row = rows.first(); cartesian_row.size() == old_descriptor_size; cartesian_row = rows.first()) {
  58. rows.remove(0);
  59. auto table_rows = TRY(context.database->select_all(*table_def));
  60. for (auto& table_row : table_rows) {
  61. auto new_row = cartesian_row;
  62. new_row.extend(table_row);
  63. rows.append(new_row);
  64. }
  65. }
  66. }
  67. bool has_ordering { false };
  68. auto sort_descriptor = adopt_ref(*new TupleDescriptor);
  69. for (auto& term : m_ordering_term_list) {
  70. sort_descriptor->append(TupleElementDescriptor { .order = term.order() });
  71. has_ordering = true;
  72. }
  73. Tuple sort_key(sort_descriptor);
  74. for (auto& row : rows) {
  75. context.current_row = &row;
  76. if (where_clause()) {
  77. auto where_result = where_clause()->evaluate(context);
  78. if (context.result->is_error())
  79. return context.result.release_value();
  80. if (!where_result)
  81. continue;
  82. }
  83. tuple.clear();
  84. for (auto& col : columns) {
  85. auto value = col.expression()->evaluate(context);
  86. if (context.result->is_error())
  87. return context.result.release_value();
  88. tuple.append(value);
  89. }
  90. if (has_ordering) {
  91. sort_key.clear();
  92. for (auto& term : m_ordering_term_list) {
  93. auto value = term.expression()->evaluate(context);
  94. if (context.result->is_error())
  95. return context.result.release_value();
  96. sort_key.append(value);
  97. }
  98. }
  99. context.result->insert(tuple, sort_key);
  100. }
  101. if (m_limit_clause != nullptr) {
  102. size_t limit_value = NumericLimits<size_t>::max();
  103. size_t offset_value = 0;
  104. auto limit = m_limit_clause->limit_expression()->evaluate(context);
  105. if (!limit.is_null()) {
  106. auto limit_value_maybe = limit.to_u32();
  107. if (!limit_value_maybe.has_value())
  108. return { SQLCommand::Select, SQLErrorCode::SyntaxError, "LIMIT clause must evaluate to an integer value"sv };
  109. limit_value = limit_value_maybe.value();
  110. }
  111. if (m_limit_clause->offset_expression() != nullptr) {
  112. auto offset = m_limit_clause->offset_expression()->evaluate(context);
  113. if (!offset.is_null()) {
  114. auto offset_value_maybe = offset.to_u32();
  115. if (!offset_value_maybe.has_value())
  116. return { SQLCommand::Select, SQLErrorCode::SyntaxError, "OFFSET clause must evaluate to an integer value"sv };
  117. offset_value = offset_value_maybe.value();
  118. }
  119. }
  120. context.result->limit(offset_value, limit_value);
  121. }
  122. return context.result.release_value();
  123. }
  124. }