/* * Copyright (c) 2021, Jan de Visser * * SPDX-License-Identifier: BSD-2-Clause */ #include #include #include #include #include namespace SQLServer { static HashMap> s_statements; static SQL::StatementID s_next_statement_id = 0; RefPtr SQLStatement::statement_for(SQL::StatementID statement_id) { if (s_statements.contains(statement_id)) return *s_statements.get(statement_id).value(); dbgln_if(SQLSERVER_DEBUG, "Invalid statement_id {}", statement_id); return nullptr; } SQL::ResultOr> SQLStatement::create(DatabaseConnection& connection, StringView sql) { auto parser = SQL::AST::Parser(SQL::AST::Lexer(sql)); auto statement = parser.next_statement(); if (parser.has_errors()) return SQL::Result { SQL::SQLCommand::Unknown, SQL::SQLErrorCode::SyntaxError, parser.errors()[0].to_deprecated_string() }; return TRY(adopt_nonnull_ref_or_enomem(new (nothrow) SQLStatement(connection, move(statement)))); } SQLStatement::SQLStatement(DatabaseConnection& connection, NonnullRefPtr statement) : m_connection(connection) , m_statement_id(s_next_statement_id++) , m_statement(move(statement)) { dbgln_if(SQLSERVER_DEBUG, "SQLStatement({})", connection.connection_id()); s_statements.set(m_statement_id, *this); } void SQLStatement::report_error(SQL::Result result, SQL::ExecutionID execution_id) { dbgln_if(SQLSERVER_DEBUG, "SQLStatement::report_error(statement_id {}, error {}", statement_id(), result.error_string()); auto client_connection = ConnectionFromClient::client_connection_for(connection().client_id()); s_statements.remove(statement_id()); if (client_connection) client_connection->async_execution_error(statement_id(), execution_id, result.error(), result.error_string()); else warnln("Cannot return execution error. Client disconnected"); } Optional SQLStatement::execute(Vector placeholder_values) { dbgln_if(SQLSERVER_DEBUG, "SQLStatement::execute(statement_id {}", statement_id()); auto client_connection = ConnectionFromClient::client_connection_for(connection().client_id()); if (!client_connection) { warnln("Cannot yield next result. Client disconnected"); return {}; } auto execution_id = m_next_execution_id++; m_ongoing_executions.set(execution_id); Core::deferred_invoke([this, strong_this = NonnullRefPtr(*this), placeholder_values = move(placeholder_values), execution_id] { auto execution_result = m_statement->execute(connection().database(), placeholder_values); m_ongoing_executions.remove(execution_id); if (execution_result.is_error()) { report_error(execution_result.release_error(), execution_id); return; } auto client_connection = ConnectionFromClient::client_connection_for(connection().client_id()); if (!client_connection) { warnln("Cannot return statement execution results. Client disconnected"); return; } auto result = execution_result.release_value(); if (should_send_result_rows(result)) { client_connection->async_execution_success(statement_id(), execution_id, result.column_names(), true, 0, 0, 0); auto result_size = result.size(); next(execution_id, move(result), result_size); } else { if (result.command() == SQL::SQLCommand::Insert) client_connection->async_execution_success(statement_id(), execution_id, result.column_names(), false, result.size(), 0, 0); else if (result.command() == SQL::SQLCommand::Update) client_connection->async_execution_success(statement_id(), execution_id, result.column_names(), false, 0, result.size(), 0); else if (result.command() == SQL::SQLCommand::Delete) client_connection->async_execution_success(statement_id(), execution_id, result.column_names(), false, 0, 0, result.size()); else client_connection->async_execution_success(statement_id(), execution_id, result.column_names(), false, 0, 0, 0); } }); return execution_id; } bool SQLStatement::should_send_result_rows(SQL::ResultSet const& result) const { if (result.is_empty()) return false; switch (result.command()) { case SQL::SQLCommand::Describe: case SQL::SQLCommand::Select: return true; default: return false; } } void SQLStatement::next(SQL::ExecutionID execution_id, SQL::ResultSet result, size_t result_size) { auto client_connection = ConnectionFromClient::client_connection_for(connection().client_id()); if (!client_connection) { warnln("Cannot yield next result. Client disconnected"); return; } if (!result.is_empty()) { auto result_row = result.take_first(); client_connection->async_next_result(statement_id(), execution_id, result_row.row.take_data()); Core::deferred_invoke([this, strong_this = NonnullRefPtr(*this), execution_id, result = move(result), result_size]() mutable { next(execution_id, move(result), result_size); }); } else { client_connection->async_results_exhausted(statement_id(), execution_id, result_size); } } }