Database.cpp 3.5 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798
  1. /*
  2. * Copyright (c) 2022-2023, Tim Flynn <trflynn89@serenityos.org>
  3. *
  4. * SPDX-License-Identifier: BSD-2-Clause
  5. */
  6. #include <AK/StringView.h>
  7. #include <LibWebView/Database.h>
  8. namespace WebView {
  9. static constexpr auto database_name = "Browser"sv;
  10. ErrorOr<NonnullRefPtr<Database>> Database::create()
  11. {
  12. auto sql_client = TRY(SQL::SQLClient::try_create());
  13. return create(move(sql_client));
  14. }
  15. #if !defined(AK_OS_SERENITY)
  16. ErrorOr<NonnullRefPtr<Database>> Database::create(Vector<String> candidate_sql_server_paths)
  17. {
  18. auto sql_client = TRY(SQL::SQLClient::launch_server_and_create_client(move(candidate_sql_server_paths)));
  19. return create(move(sql_client));
  20. }
  21. #endif
  22. ErrorOr<NonnullRefPtr<Database>> Database::create(NonnullRefPtr<SQL::SQLClient> sql_client)
  23. {
  24. auto connection_id = sql_client->connect(database_name);
  25. if (!connection_id.has_value())
  26. return Error::from_string_view("Could not connect to SQL database"sv);
  27. return adopt_nonnull_ref_or_enomem(new (nothrow) Database(move(sql_client), *connection_id));
  28. }
  29. Database::Database(NonnullRefPtr<SQL::SQLClient> sql_client, SQL::ConnectionID connection_id)
  30. : m_sql_client(move(sql_client))
  31. , m_connection_id(connection_id)
  32. {
  33. m_sql_client->on_execution_success = [this](auto result) {
  34. if (result.has_results)
  35. return;
  36. if (auto in_progress_statement = take_pending_execution(result); in_progress_statement.has_value()) {
  37. if (in_progress_statement->on_complete)
  38. in_progress_statement->on_complete();
  39. }
  40. };
  41. m_sql_client->on_next_result = [this](auto result) {
  42. if (auto in_progress_statement = take_pending_execution(result); in_progress_statement.has_value()) {
  43. if (in_progress_statement->on_result)
  44. in_progress_statement->on_result(result.values);
  45. m_pending_executions.set({ result.statement_id, result.execution_id }, in_progress_statement.release_value());
  46. }
  47. };
  48. m_sql_client->on_results_exhausted = [this](auto result) {
  49. if (auto in_progress_statement = take_pending_execution(result); in_progress_statement.has_value()) {
  50. if (in_progress_statement->on_complete)
  51. in_progress_statement->on_complete();
  52. }
  53. };
  54. m_sql_client->on_execution_error = [this](auto result) {
  55. if (auto in_progress_statement = take_pending_execution(result); in_progress_statement.has_value()) {
  56. if (in_progress_statement->on_error)
  57. in_progress_statement->on_error(result.error_message);
  58. }
  59. };
  60. }
  61. ErrorOr<SQL::StatementID> Database::prepare_statement(StringView statement)
  62. {
  63. if (auto statement_id = m_sql_client->prepare_statement(m_connection_id, statement); statement_id.has_value())
  64. return *statement_id;
  65. return Error::from_string_view(statement);
  66. }
  67. void Database::execute_statement(SQL::StatementID statement_id, Vector<SQL::Value> placeholder_values, PendingExecution pending_execution)
  68. {
  69. Core::deferred_invoke([this, statement_id, placeholder_values = move(placeholder_values), pending_execution = move(pending_execution)]() mutable {
  70. auto execution_id = m_sql_client->execute_statement(statement_id, move(placeholder_values));
  71. if (!execution_id.has_value()) {
  72. if (pending_execution.on_error)
  73. pending_execution.on_error("Could not execute statement"sv);
  74. return;
  75. }
  76. m_pending_executions.set({ statement_id, *execution_id }, move(pending_execution));
  77. });
  78. }
  79. }