Select.cpp 4.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120
  1. /*
  2. * Copyright (c) 2021, Jan de Visser <jan@de-visser.net>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <LibSQL/AST/AST.h>
  7. #include <LibSQL/Database.h>
  8. #include <LibSQL/Meta.h>
  9. #include <LibSQL/ResultSet.h>
  10. #include <LibSQL/Row.h>
  11. namespace SQL::AST {
  12. RefPtr<SQLResult> Select::execute(ExecutionContext& context) const
  13. {
  14. NonnullRefPtrVector<ResultColumn> columns;
  15. for (auto& table_descriptor : table_or_subquery_list()) {
  16. if (!table_descriptor.is_table())
  17. return SQLResult::construct(SQLCommand::Select, SQLErrorCode::NotYetImplemented, "Sub-selects are not yet implemented");
  18. auto table_def_or_error = context.database->get_table(table_descriptor.schema_name(), table_descriptor.table_name());
  19. if (table_def_or_error.is_error())
  20. return SQLResult::construct(SQLCommand::Select, SQLErrorCode::InternalError, table_def_or_error.error());
  21. auto table = table_def_or_error.value();
  22. if (!table) {
  23. return SQLResult::construct(SQLCommand::Select, SQLErrorCode::TableDoesNotExist, table_descriptor.table_name());
  24. }
  25. if (result_column_list().size() == 1 && result_column_list()[0].type() == ResultType::All) {
  26. for (auto& col : table->columns()) {
  27. columns.append(
  28. create_ast_node<ResultColumn>(
  29. create_ast_node<ColumnNameExpression>(table->parent()->name(), table->name(), col.name()),
  30. ""));
  31. }
  32. }
  33. }
  34. VERIFY(!result_column_list().is_empty());
  35. if (result_column_list().size() != 1 || result_column_list()[0].type() != ResultType::All) {
  36. for (auto& col : result_column_list()) {
  37. if (col.type() == ResultType::All)
  38. // FIXME can have '*' for example in conjunction with computed columns
  39. return SQLResult::construct(SQL::SQLCommand::Select, SQLErrorCode::SyntaxError, "*");
  40. columns.append(col);
  41. }
  42. }
  43. context.result = SQLResult::construct();
  44. AK::NonnullRefPtr<TupleDescriptor> descriptor = AK::adopt_ref(*new TupleDescriptor);
  45. Tuple tuple(descriptor);
  46. Vector<Tuple> rows;
  47. descriptor->empend("__unity__");
  48. tuple.append(Value(SQLType::Boolean, true));
  49. rows.append(tuple);
  50. for (auto& table_descriptor : table_or_subquery_list()) {
  51. if (!table_descriptor.is_table())
  52. return SQLResult::construct(SQLCommand::Select, SQLErrorCode::NotYetImplemented, "Sub-selects are not yet implemented");
  53. auto table_def_or_error = context.database->get_table(table_descriptor.schema_name(), table_descriptor.table_name());
  54. if (table_def_or_error.is_error())
  55. return SQLResult::construct(SQLCommand::Select, SQLErrorCode::InternalError, table_def_or_error.error());
  56. auto table = table_def_or_error.value();
  57. if (table->num_columns() == 0)
  58. continue;
  59. auto old_descriptor_size = descriptor->size();
  60. descriptor->extend(table->to_tuple_descriptor());
  61. for (auto cartesian_row = rows.first(); cartesian_row.size() == old_descriptor_size; cartesian_row = rows.first()) {
  62. rows.remove(0);
  63. auto table_rows_or_error = context.database->select_all(*table);
  64. if (table_rows_or_error.is_error())
  65. return SQLResult::construct(SQLCommand::Create, SQLErrorCode::InternalError, table_rows_or_error.error());
  66. for (auto& table_row : table_rows_or_error.value()) {
  67. auto new_row = cartesian_row;
  68. new_row.extend(table_row);
  69. rows.append(new_row);
  70. }
  71. }
  72. }
  73. bool has_ordering { false };
  74. AK::NonnullRefPtr<TupleDescriptor> sort_descriptor = AK::adopt_ref(*new TupleDescriptor);
  75. for (auto& term : m_ordering_term_list) {
  76. sort_descriptor->append(TupleElementDescriptor { .order = term.order() });
  77. has_ordering = true;
  78. }
  79. Tuple sort_key(sort_descriptor);
  80. for (auto& row : rows) {
  81. context.current_row = &row;
  82. if (where_clause()) {
  83. auto where_result = where_clause()->evaluate(context);
  84. if (context.result->has_error())
  85. return context.result;
  86. if (!where_result)
  87. continue;
  88. }
  89. tuple.clear();
  90. for (auto& col : columns) {
  91. auto value = col.expression()->evaluate(context);
  92. if (context.result->has_error())
  93. return context.result;
  94. tuple.append(value);
  95. }
  96. if (has_ordering) {
  97. sort_key.clear();
  98. for (auto& term : m_ordering_term_list) {
  99. auto value = term.expression()->evaluate(context);
  100. if (context.result->has_error())
  101. return context.result;
  102. sort_key.append(value);
  103. }
  104. }
  105. context.result->insert(tuple, sort_key);
  106. }
  107. return context.result;
  108. }
  109. }