Serenity Operating System
at master 61 lines 2.2 kB view raw
1/* 2 * Copyright (c) 2021, Jan de Visser <jan@de-visser.net> 3 * Copyright (c) 2021, Mahmoud Mandour <ma.mandourr@gmail.com> 4 * 5 * SPDX-License-Identifier: BSD-2-Clause 6 */ 7 8#include <LibSQL/AST/AST.h> 9#include <LibSQL/Database.h> 10#include <LibSQL/Meta.h> 11#include <LibSQL/Row.h> 12 13namespace SQL::AST { 14 15ResultOr<ResultSet> Insert::execute(ExecutionContext& context) const 16{ 17 auto table_def = TRY(context.database->get_table(m_schema_name, m_table_name)); 18 19 Row row(table_def); 20 for (auto& column : m_column_names) { 21 if (!row.has(column)) 22 return Result { SQLCommand::Insert, SQLErrorCode::ColumnDoesNotExist, column }; 23 } 24 25 ResultSet result { SQLCommand::Insert }; 26 TRY(result.try_ensure_capacity(m_chained_expressions.size())); 27 28 for (auto& row_expr : m_chained_expressions) { 29 for (auto& column_def : table_def->columns()) { 30 if (!m_column_names.contains_slow(column_def->name())) 31 row[column_def->name()] = column_def->default_value(); 32 } 33 34 auto row_value = TRY(row_expr->evaluate(context)); 35 VERIFY(row_value.type() == SQLType::Tuple); 36 37 auto values = row_value.to_vector().release_value(); 38 39 if (m_column_names.is_empty() && values.size() != row.size()) 40 return Result { SQLCommand::Insert, SQLErrorCode::InvalidNumberOfValues, DeprecatedString::empty() }; 41 42 for (auto ix = 0u; ix < values.size(); ix++) { 43 auto& tuple_descriptor = *row.descriptor(); 44 // In case of having column names, this must succeed since we checked for every column name for existence in the table. 45 auto element_index = m_column_names.is_empty() ? ix : tuple_descriptor.find_if([&](auto element) { return element.name == m_column_names[ix]; }).index(); 46 auto element_type = tuple_descriptor[element_index].type; 47 48 if (!values[ix].is_type_compatible_with(element_type)) 49 return Result { SQLCommand::Insert, SQLErrorCode::InvalidValueType, table_def->columns()[element_index]->name() }; 50 51 row[element_index] = move(values[ix]); 52 } 53 54 TRY(context.database->insert(row)); 55 result.insert_row(row, {}); 56 } 57 58 return result; 59} 60 61}