aboutsummaryrefslogtreecommitdiff
path: root/updater/sql/mac/src/statement.h
diff options
context:
space:
mode:
authorRaindropsSys <raindrops@equestria.dev>2024-06-13 15:46:03 +0200
committerRaindropsSys <raindrops@equestria.dev>2024-06-13 15:46:03 +0200
commite44e2fe070484e06d384a31ef2699c3a2d5d474e (patch)
tree2d5eb5d1b01646270d18cf1f2d94519966d6e7de /updater/sql/mac/src/statement.h
downloadfaunerie-e44e2fe070484e06d384a31ef2699c3a2d5d474e.tar.gz
faunerie-e44e2fe070484e06d384a31ef2699c3a2d5d474e.tar.bz2
faunerie-e44e2fe070484e06d384a31ef2699c3a2d5d474e.zip
GitHub migration
Diffstat (limited to 'updater/sql/mac/src/statement.h')
-rwxr-xr-xupdater/sql/mac/src/statement.h244
1 files changed, 244 insertions, 0 deletions
diff --git a/updater/sql/mac/src/statement.h b/updater/sql/mac/src/statement.h
new file mode 100755
index 0000000..c522c0f
--- /dev/null
+++ b/updater/sql/mac/src/statement.h
@@ -0,0 +1,244 @@
+#ifndef NODE_SQLITE3_SRC_STATEMENT_H
+#define NODE_SQLITE3_SRC_STATEMENT_H
+
+#include <cstdlib>
+#include <cstring>
+#include <string>
+#include <queue>
+#include <vector>
+#include <sqlite3.h>
+#include <napi.h>
+#include <uv.h>
+
+#include "database.h"
+#include "threading.h"
+
+using namespace Napi;
+
+namespace node_sqlite3 {
+
+namespace Values {
+ struct Field {
+ inline Field(unsigned short _index, unsigned short _type = SQLITE_NULL) :
+ type(_type), index(_index) {}
+ inline Field(const char* _name, unsigned short _type = SQLITE_NULL) :
+ type(_type), index(0), name(_name) {}
+
+ unsigned short type;
+ unsigned short index;
+ std::string name;
+
+ virtual ~Field() = default;
+ };
+
+ struct Integer : Field {
+ template <class T> inline Integer(T _name, int64_t val) :
+ Field(_name, SQLITE_INTEGER), value(val) {}
+ int64_t value;
+ virtual ~Integer() override = default;
+ };
+
+ struct Float : Field {
+ template <class T> inline Float(T _name, double val) :
+ Field(_name, SQLITE_FLOAT), value(val) {}
+ double value;
+ virtual ~Float() override = default;
+ };
+
+ struct Text : Field {
+ template <class T> inline Text(T _name, size_t len, const char* val) :
+ Field(_name, SQLITE_TEXT), value(val, len) {}
+ std::string value;
+ virtual ~Text() override = default;
+ };
+
+ struct Blob : Field {
+ template <class T> inline Blob(T _name, size_t len, const void* val) :
+ Field(_name, SQLITE_BLOB), length(len) {
+ value = new char[len];
+ assert(value != nullptr);
+ memcpy(value, val, len);
+ }
+ inline virtual ~Blob() override {
+ delete[] value;
+ }
+ int length;
+ char* value;
+ };
+
+ typedef Field Null;
+}
+
+typedef std::vector<std::unique_ptr<Values::Field> > Row;
+typedef std::vector<std::unique_ptr<Row> > Rows;
+typedef Row Parameters;
+
+
+
+class Statement : public Napi::ObjectWrap<Statement> {
+public:
+ static Napi::Object Init(Napi::Env env, Napi::Object exports);
+ static Napi::Value New(const Napi::CallbackInfo& info);
+
+ struct Baton {
+ napi_async_work request = NULL;
+ Statement* stmt;
+ Napi::FunctionReference callback;
+ Parameters parameters;
+
+ Baton(Statement* stmt_, Napi::Function cb_) : stmt(stmt_) {
+ stmt->Ref();
+ callback.Reset(cb_, 1);
+ }
+ virtual ~Baton() {
+ parameters.clear();
+ if (request) napi_delete_async_work(stmt->Env(), request);
+ stmt->Unref();
+ callback.Reset();
+ }
+ };
+
+ struct RowBaton : Baton {
+ RowBaton(Statement* stmt_, Napi::Function cb_) :
+ Baton(stmt_, cb_) {}
+ Row row;
+ virtual ~RowBaton() override = default;
+ };
+
+ struct RunBaton : Baton {
+ RunBaton(Statement* stmt_, Napi::Function cb_) :
+ Baton(stmt_, cb_), inserted_id(0), changes(0) {}
+ sqlite3_int64 inserted_id;
+ int changes;
+ virtual ~RunBaton() override = default;
+ };
+
+ struct RowsBaton : Baton {
+ RowsBaton(Statement* stmt_, Napi::Function cb_) :
+ Baton(stmt_, cb_) {}
+ Rows rows;
+ virtual ~RowsBaton() override = default;
+ };
+
+ struct Async;
+
+ struct EachBaton : Baton {
+ Napi::FunctionReference completed;
+ Async* async; // Isn't deleted when the baton is deleted.
+
+ EachBaton(Statement* stmt_, Napi::Function cb_) :
+ Baton(stmt_, cb_) {}
+ virtual ~EachBaton() override {
+ completed.Reset();
+ }
+ };
+
+ struct PrepareBaton : Database::Baton {
+ Statement* stmt;
+ std::string sql;
+ PrepareBaton(Database* db_, Napi::Function cb_, Statement* stmt_) :
+ Baton(db_, cb_), stmt(stmt_) {
+ stmt->Ref();
+ }
+ virtual ~PrepareBaton() override {
+ stmt->Unref();
+ if (!db->IsOpen() && db->IsLocked()) {
+ // The database handle was closed before the statement could be
+ // prepared.
+ stmt->Finalize_();
+ }
+ }
+ };
+
+ typedef void (*Work_Callback)(Baton* baton);
+
+ struct Call {
+ Call(Work_Callback cb_, Baton* baton_) : callback(cb_), baton(baton_) {};
+ Work_Callback callback;
+ Baton* baton;
+ };
+
+ struct Async {
+ uv_async_t watcher;
+ Statement* stmt;
+ Rows data;
+ NODE_SQLITE3_MUTEX_t;
+ bool completed;
+ int retrieved;
+
+ // Store the callbacks here because we don't have
+ // access to the baton in the async callback.
+ Napi::FunctionReference item_cb;
+ Napi::FunctionReference completed_cb;
+
+ Async(Statement* st, uv_async_cb async_cb) :
+ stmt(st), completed(false), retrieved(0) {
+ watcher.data = this;
+ NODE_SQLITE3_MUTEX_INIT
+ stmt->Ref();
+ uv_loop_t *loop;
+ napi_get_uv_event_loop(stmt->Env(), &loop);
+ uv_async_init(loop, &watcher, async_cb);
+ }
+
+ ~Async() {
+ stmt->Unref();
+ item_cb.Reset();
+ completed_cb.Reset();
+ NODE_SQLITE3_MUTEX_DESTROY
+ }
+ };
+
+ Statement(const Napi::CallbackInfo& info);
+
+ ~Statement() {
+ if (!finalized) Finalize_();
+ }
+
+ WORK_DEFINITION(Bind)
+ WORK_DEFINITION(Get)
+ WORK_DEFINITION(Run)
+ WORK_DEFINITION(All)
+ WORK_DEFINITION(Each)
+ WORK_DEFINITION(Reset)
+
+ Napi::Value Finalize_(const Napi::CallbackInfo& info);
+
+protected:
+ static void Work_BeginPrepare(Database::Baton* baton);
+ static void Work_Prepare(napi_env env, void* data);
+ static void Work_AfterPrepare(napi_env env, napi_status status, void* data);
+
+ static void AsyncEach(uv_async_t* handle);
+ static void CloseCallback(uv_handle_t* handle);
+
+ static void Finalize_(Baton* baton);
+ void Finalize_();
+
+ template <class T> inline std::unique_ptr<Values::Field> BindParameter(const Napi::Value source, T pos);
+ template <class T> T* Bind(const Napi::CallbackInfo& info, int start = 0, int end = -1);
+ bool Bind(const Parameters &parameters);
+
+ static void GetRow(Row* row, sqlite3_stmt* stmt);
+ static Napi::Value RowToJS(Napi::Env env, Row* row);
+ void Schedule(Work_Callback callback, Baton* baton);
+ void Process();
+ void CleanQueue();
+ template <class T> static void Error(T* baton);
+
+protected:
+ Database* db;
+
+ sqlite3_stmt* _handle = NULL;
+ int status = SQLITE_OK;
+ bool prepared = false;
+ bool locked = true;
+ bool finalized = false;
+
+ std::queue<Call*> queue;
+ std::string message;
+};
+
+}
+
+#endif