@barakxyz/better-sqlite3 12.6.2
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +99 -0
- package/binding.gyp +42 -0
- package/deps/common.gypi +68 -0
- package/deps/copy.js +31 -0
- package/deps/defines.gypi +43 -0
- package/deps/download.sh +122 -0
- package/deps/patches/1208.patch +15 -0
- package/deps/sqlite3/sqlite3.c +265969 -0
- package/deps/sqlite3/sqlite3.h +13968 -0
- package/deps/sqlite3/sqlite3ext.h +730 -0
- package/deps/sqlite3.gyp +80 -0
- package/deps/test_extension.c +21 -0
- package/lib/database.js +93 -0
- package/lib/index.js +3 -0
- package/lib/methods/aggregate.js +43 -0
- package/lib/methods/backup.js +67 -0
- package/lib/methods/function.js +31 -0
- package/lib/methods/inspect.js +7 -0
- package/lib/methods/pragma.js +12 -0
- package/lib/methods/serialize.js +16 -0
- package/lib/methods/table.js +189 -0
- package/lib/methods/transaction.js +78 -0
- package/lib/methods/wrappers.js +73 -0
- package/lib/sqlite-error.js +20 -0
- package/lib/util.js +12 -0
- package/package.json +59 -0
- package/src/addon.cpp +48 -0
- package/src/better_sqlite3.cpp +79 -0
- package/src/objects/backup.cpp +120 -0
- package/src/objects/backup.hpp +36 -0
- package/src/objects/database.cpp +521 -0
- package/src/objects/database.hpp +116 -0
- package/src/objects/session.cpp +148 -0
- package/src/objects/session.hpp +33 -0
- package/src/objects/statement-iterator.cpp +113 -0
- package/src/objects/statement-iterator.hpp +50 -0
- package/src/objects/statement.cpp +383 -0
- package/src/objects/statement.hpp +58 -0
- package/src/util/bind-map.cpp +73 -0
- package/src/util/binder.cpp +193 -0
- package/src/util/constants.cpp +172 -0
- package/src/util/custom-aggregate.cpp +121 -0
- package/src/util/custom-function.cpp +59 -0
- package/src/util/custom-table.cpp +409 -0
- package/src/util/data-converter.cpp +17 -0
- package/src/util/data.cpp +194 -0
- package/src/util/helpers.cpp +109 -0
- package/src/util/macros.cpp +70 -0
- package/src/util/query-macros.cpp +71 -0
- package/src/util/row-builder.cpp +49 -0
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
class CS {
|
|
2
|
+
public:
|
|
3
|
+
|
|
4
|
+
v8::Local<v8::String> Code(v8::Isolate* isolate, int code) {
|
|
5
|
+
auto element = codes.find(code);
|
|
6
|
+
if (element != codes.end()) return element->second.Get(isolate);
|
|
7
|
+
return StringFromUtf8(isolate, (std::string("UNKNOWN_SQLITE_ERROR_") + std::to_string(code)).c_str(), -1);
|
|
8
|
+
}
|
|
9
|
+
|
|
10
|
+
explicit CS(v8::Isolate* isolate) {
|
|
11
|
+
SetString(isolate, database, "database");
|
|
12
|
+
SetString(isolate, reader, "reader");
|
|
13
|
+
SetString(isolate, source, "source");
|
|
14
|
+
SetString(isolate, memory, "memory");
|
|
15
|
+
SetString(isolate, readonly, "readonly");
|
|
16
|
+
SetString(isolate, name, "name");
|
|
17
|
+
SetString(isolate, next, "next");
|
|
18
|
+
SetString(isolate, length, "length");
|
|
19
|
+
SetString(isolate, done, "done");
|
|
20
|
+
SetString(isolate, value, "value");
|
|
21
|
+
SetString(isolate, changes, "changes");
|
|
22
|
+
SetString(isolate, lastInsertRowid, "lastInsertRowid");
|
|
23
|
+
SetString(isolate, statement, "statement");
|
|
24
|
+
SetString(isolate, column, "column");
|
|
25
|
+
SetString(isolate, table, "table");
|
|
26
|
+
SetString(isolate, type, "type");
|
|
27
|
+
SetString(isolate, totalPages, "totalPages");
|
|
28
|
+
SetString(isolate, remainingPages, "remainingPages");
|
|
29
|
+
|
|
30
|
+
SetCode(isolate, SQLITE_OK, "SQLITE_OK");
|
|
31
|
+
SetCode(isolate, SQLITE_ERROR, "SQLITE_ERROR");
|
|
32
|
+
SetCode(isolate, SQLITE_INTERNAL, "SQLITE_INTERNAL");
|
|
33
|
+
SetCode(isolate, SQLITE_PERM, "SQLITE_PERM");
|
|
34
|
+
SetCode(isolate, SQLITE_ABORT, "SQLITE_ABORT");
|
|
35
|
+
SetCode(isolate, SQLITE_BUSY, "SQLITE_BUSY");
|
|
36
|
+
SetCode(isolate, SQLITE_LOCKED, "SQLITE_LOCKED");
|
|
37
|
+
SetCode(isolate, SQLITE_NOMEM, "SQLITE_NOMEM");
|
|
38
|
+
SetCode(isolate, SQLITE_READONLY, "SQLITE_READONLY");
|
|
39
|
+
SetCode(isolate, SQLITE_INTERRUPT, "SQLITE_INTERRUPT");
|
|
40
|
+
SetCode(isolate, SQLITE_IOERR, "SQLITE_IOERR");
|
|
41
|
+
SetCode(isolate, SQLITE_CORRUPT, "SQLITE_CORRUPT");
|
|
42
|
+
SetCode(isolate, SQLITE_NOTFOUND, "SQLITE_NOTFOUND");
|
|
43
|
+
SetCode(isolate, SQLITE_FULL, "SQLITE_FULL");
|
|
44
|
+
SetCode(isolate, SQLITE_CANTOPEN, "SQLITE_CANTOPEN");
|
|
45
|
+
SetCode(isolate, SQLITE_PROTOCOL, "SQLITE_PROTOCOL");
|
|
46
|
+
SetCode(isolate, SQLITE_EMPTY, "SQLITE_EMPTY");
|
|
47
|
+
SetCode(isolate, SQLITE_SCHEMA, "SQLITE_SCHEMA");
|
|
48
|
+
SetCode(isolate, SQLITE_TOOBIG, "SQLITE_TOOBIG");
|
|
49
|
+
SetCode(isolate, SQLITE_CONSTRAINT, "SQLITE_CONSTRAINT");
|
|
50
|
+
SetCode(isolate, SQLITE_MISMATCH, "SQLITE_MISMATCH");
|
|
51
|
+
SetCode(isolate, SQLITE_MISUSE, "SQLITE_MISUSE");
|
|
52
|
+
SetCode(isolate, SQLITE_NOLFS, "SQLITE_NOLFS");
|
|
53
|
+
SetCode(isolate, SQLITE_AUTH, "SQLITE_AUTH");
|
|
54
|
+
SetCode(isolate, SQLITE_FORMAT, "SQLITE_FORMAT");
|
|
55
|
+
SetCode(isolate, SQLITE_RANGE, "SQLITE_RANGE");
|
|
56
|
+
SetCode(isolate, SQLITE_NOTADB, "SQLITE_NOTADB");
|
|
57
|
+
SetCode(isolate, SQLITE_NOTICE, "SQLITE_NOTICE");
|
|
58
|
+
SetCode(isolate, SQLITE_WARNING, "SQLITE_WARNING");
|
|
59
|
+
SetCode(isolate, SQLITE_ROW, "SQLITE_ROW");
|
|
60
|
+
SetCode(isolate, SQLITE_DONE, "SQLITE_DONE");
|
|
61
|
+
|
|
62
|
+
SetCode(isolate, SQLITE_ERROR_MISSING_COLLSEQ, "SQLITE_ERROR_MISSING_COLLSEQ");
|
|
63
|
+
SetCode(isolate, SQLITE_ERROR_RETRY, "SQLITE_ERROR_RETRY");
|
|
64
|
+
SetCode(isolate, SQLITE_ERROR_SNAPSHOT, "SQLITE_ERROR_SNAPSHOT");
|
|
65
|
+
SetCode(isolate, SQLITE_IOERR_READ, "SQLITE_IOERR_READ");
|
|
66
|
+
SetCode(isolate, SQLITE_IOERR_SHORT_READ, "SQLITE_IOERR_SHORT_READ");
|
|
67
|
+
SetCode(isolate, SQLITE_IOERR_WRITE, "SQLITE_IOERR_WRITE");
|
|
68
|
+
SetCode(isolate, SQLITE_IOERR_FSYNC, "SQLITE_IOERR_FSYNC");
|
|
69
|
+
SetCode(isolate, SQLITE_IOERR_DIR_FSYNC, "SQLITE_IOERR_DIR_FSYNC");
|
|
70
|
+
SetCode(isolate, SQLITE_IOERR_TRUNCATE, "SQLITE_IOERR_TRUNCATE");
|
|
71
|
+
SetCode(isolate, SQLITE_IOERR_FSTAT, "SQLITE_IOERR_FSTAT");
|
|
72
|
+
SetCode(isolate, SQLITE_IOERR_UNLOCK, "SQLITE_IOERR_UNLOCK");
|
|
73
|
+
SetCode(isolate, SQLITE_IOERR_RDLOCK, "SQLITE_IOERR_RDLOCK");
|
|
74
|
+
SetCode(isolate, SQLITE_IOERR_DELETE, "SQLITE_IOERR_DELETE");
|
|
75
|
+
SetCode(isolate, SQLITE_IOERR_BLOCKED, "SQLITE_IOERR_BLOCKED");
|
|
76
|
+
SetCode(isolate, SQLITE_IOERR_NOMEM, "SQLITE_IOERR_NOMEM");
|
|
77
|
+
SetCode(isolate, SQLITE_IOERR_ACCESS, "SQLITE_IOERR_ACCESS");
|
|
78
|
+
SetCode(isolate, SQLITE_IOERR_CHECKRESERVEDLOCK, "SQLITE_IOERR_CHECKRESERVEDLOCK");
|
|
79
|
+
SetCode(isolate, SQLITE_IOERR_LOCK, "SQLITE_IOERR_LOCK");
|
|
80
|
+
SetCode(isolate, SQLITE_IOERR_CLOSE, "SQLITE_IOERR_CLOSE");
|
|
81
|
+
SetCode(isolate, SQLITE_IOERR_DIR_CLOSE, "SQLITE_IOERR_DIR_CLOSE");
|
|
82
|
+
SetCode(isolate, SQLITE_IOERR_SHMOPEN, "SQLITE_IOERR_SHMOPEN");
|
|
83
|
+
SetCode(isolate, SQLITE_IOERR_SHMSIZE, "SQLITE_IOERR_SHMSIZE");
|
|
84
|
+
SetCode(isolate, SQLITE_IOERR_SHMLOCK, "SQLITE_IOERR_SHMLOCK");
|
|
85
|
+
SetCode(isolate, SQLITE_IOERR_SHMMAP, "SQLITE_IOERR_SHMMAP");
|
|
86
|
+
SetCode(isolate, SQLITE_IOERR_SEEK, "SQLITE_IOERR_SEEK");
|
|
87
|
+
SetCode(isolate, SQLITE_IOERR_DELETE_NOENT, "SQLITE_IOERR_DELETE_NOENT");
|
|
88
|
+
SetCode(isolate, SQLITE_IOERR_MMAP, "SQLITE_IOERR_MMAP");
|
|
89
|
+
SetCode(isolate, SQLITE_IOERR_GETTEMPPATH, "SQLITE_IOERR_GETTEMPPATH");
|
|
90
|
+
SetCode(isolate, SQLITE_IOERR_CONVPATH, "SQLITE_IOERR_CONVPATH");
|
|
91
|
+
SetCode(isolate, SQLITE_IOERR_VNODE, "SQLITE_IOERR_VNODE");
|
|
92
|
+
SetCode(isolate, SQLITE_IOERR_AUTH, "SQLITE_IOERR_AUTH");
|
|
93
|
+
SetCode(isolate, SQLITE_IOERR_BEGIN_ATOMIC, "SQLITE_IOERR_BEGIN_ATOMIC");
|
|
94
|
+
SetCode(isolate, SQLITE_IOERR_COMMIT_ATOMIC, "SQLITE_IOERR_COMMIT_ATOMIC");
|
|
95
|
+
SetCode(isolate, SQLITE_IOERR_ROLLBACK_ATOMIC, "SQLITE_IOERR_ROLLBACK_ATOMIC");
|
|
96
|
+
SetCode(isolate, SQLITE_IOERR_DATA, "SQLITE_IOERR_DATA");
|
|
97
|
+
SetCode(isolate, SQLITE_IOERR_CORRUPTFS, "SQLITE_IOERR_CORRUPTFS");
|
|
98
|
+
SetCode(isolate, SQLITE_IOERR_IN_PAGE, "SQLITE_IOERR_IN_PAGE");
|
|
99
|
+
SetCode(isolate, SQLITE_LOCKED_SHAREDCACHE, "SQLITE_LOCKED_SHAREDCACHE");
|
|
100
|
+
SetCode(isolate, SQLITE_LOCKED_VTAB, "SQLITE_LOCKED_VTAB");
|
|
101
|
+
SetCode(isolate, SQLITE_BUSY_RECOVERY, "SQLITE_BUSY_RECOVERY");
|
|
102
|
+
SetCode(isolate, SQLITE_BUSY_SNAPSHOT, "SQLITE_BUSY_SNAPSHOT");
|
|
103
|
+
SetCode(isolate, SQLITE_CANTOPEN_NOTEMPDIR, "SQLITE_CANTOPEN_NOTEMPDIR");
|
|
104
|
+
SetCode(isolate, SQLITE_CANTOPEN_ISDIR, "SQLITE_CANTOPEN_ISDIR");
|
|
105
|
+
SetCode(isolate, SQLITE_CANTOPEN_FULLPATH, "SQLITE_CANTOPEN_FULLPATH");
|
|
106
|
+
SetCode(isolate, SQLITE_CANTOPEN_CONVPATH, "SQLITE_CANTOPEN_CONVPATH");
|
|
107
|
+
SetCode(isolate, SQLITE_CANTOPEN_DIRTYWAL, "SQLITE_CANTOPEN_DIRTYWAL");
|
|
108
|
+
SetCode(isolate, SQLITE_CANTOPEN_SYMLINK, "SQLITE_CANTOPEN_SYMLINK");
|
|
109
|
+
SetCode(isolate, SQLITE_CORRUPT_VTAB, "SQLITE_CORRUPT_VTAB");
|
|
110
|
+
SetCode(isolate, SQLITE_CORRUPT_SEQUENCE, "SQLITE_CORRUPT_SEQUENCE");
|
|
111
|
+
SetCode(isolate, SQLITE_CORRUPT_INDEX, "SQLITE_CORRUPT_INDEX");
|
|
112
|
+
SetCode(isolate, SQLITE_READONLY_RECOVERY, "SQLITE_READONLY_RECOVERY");
|
|
113
|
+
SetCode(isolate, SQLITE_READONLY_CANTLOCK, "SQLITE_READONLY_CANTLOCK");
|
|
114
|
+
SetCode(isolate, SQLITE_READONLY_ROLLBACK, "SQLITE_READONLY_ROLLBACK");
|
|
115
|
+
SetCode(isolate, SQLITE_READONLY_DBMOVED, "SQLITE_READONLY_DBMOVED");
|
|
116
|
+
SetCode(isolate, SQLITE_READONLY_CANTINIT, "SQLITE_READONLY_CANTINIT");
|
|
117
|
+
SetCode(isolate, SQLITE_READONLY_DIRECTORY, "SQLITE_READONLY_DIRECTORY");
|
|
118
|
+
SetCode(isolate, SQLITE_ABORT_ROLLBACK, "SQLITE_ABORT_ROLLBACK");
|
|
119
|
+
SetCode(isolate, SQLITE_CONSTRAINT_CHECK, "SQLITE_CONSTRAINT_CHECK");
|
|
120
|
+
SetCode(isolate, SQLITE_CONSTRAINT_COMMITHOOK, "SQLITE_CONSTRAINT_COMMITHOOK");
|
|
121
|
+
SetCode(isolate, SQLITE_CONSTRAINT_FOREIGNKEY, "SQLITE_CONSTRAINT_FOREIGNKEY");
|
|
122
|
+
SetCode(isolate, SQLITE_CONSTRAINT_FUNCTION, "SQLITE_CONSTRAINT_FUNCTION");
|
|
123
|
+
SetCode(isolate, SQLITE_CONSTRAINT_NOTNULL, "SQLITE_CONSTRAINT_NOTNULL");
|
|
124
|
+
SetCode(isolate, SQLITE_CONSTRAINT_PRIMARYKEY, "SQLITE_CONSTRAINT_PRIMARYKEY");
|
|
125
|
+
SetCode(isolate, SQLITE_CONSTRAINT_TRIGGER, "SQLITE_CONSTRAINT_TRIGGER");
|
|
126
|
+
SetCode(isolate, SQLITE_CONSTRAINT_UNIQUE, "SQLITE_CONSTRAINT_UNIQUE");
|
|
127
|
+
SetCode(isolate, SQLITE_CONSTRAINT_VTAB, "SQLITE_CONSTRAINT_VTAB");
|
|
128
|
+
SetCode(isolate, SQLITE_CONSTRAINT_ROWID, "SQLITE_CONSTRAINT_ROWID");
|
|
129
|
+
SetCode(isolate, SQLITE_CONSTRAINT_PINNED, "SQLITE_CONSTRAINT_PINNED");
|
|
130
|
+
SetCode(isolate, SQLITE_CONSTRAINT_DATATYPE, "SQLITE_CONSTRAINT_DATATYPE");
|
|
131
|
+
SetCode(isolate, SQLITE_NOTICE_RECOVER_WAL, "SQLITE_NOTICE_RECOVER_WAL");
|
|
132
|
+
SetCode(isolate, SQLITE_NOTICE_RECOVER_ROLLBACK, "SQLITE_NOTICE_RECOVER_ROLLBACK");
|
|
133
|
+
SetCode(isolate, SQLITE_NOTICE_RBU, "SQLITE_NOTICE_RBU");
|
|
134
|
+
SetCode(isolate, SQLITE_WARNING_AUTOINDEX, "SQLITE_WARNING_AUTOINDEX");
|
|
135
|
+
SetCode(isolate, SQLITE_AUTH_USER, "SQLITE_AUTH_USER");
|
|
136
|
+
SetCode(isolate, SQLITE_OK_LOAD_PERMANENTLY, "SQLITE_OK_LOAD_PERMANENTLY");
|
|
137
|
+
SetCode(isolate, SQLITE_OK_SYMLINK, "SQLITE_OK_SYMLINK");
|
|
138
|
+
}
|
|
139
|
+
|
|
140
|
+
v8::Global<v8::String> database;
|
|
141
|
+
v8::Global<v8::String> reader;
|
|
142
|
+
v8::Global<v8::String> source;
|
|
143
|
+
v8::Global<v8::String> memory;
|
|
144
|
+
v8::Global<v8::String> readonly;
|
|
145
|
+
v8::Global<v8::String> name;
|
|
146
|
+
v8::Global<v8::String> next;
|
|
147
|
+
v8::Global<v8::String> length;
|
|
148
|
+
v8::Global<v8::String> done;
|
|
149
|
+
v8::Global<v8::String> value;
|
|
150
|
+
v8::Global<v8::String> changes;
|
|
151
|
+
v8::Global<v8::String> lastInsertRowid;
|
|
152
|
+
v8::Global<v8::String> statement;
|
|
153
|
+
v8::Global<v8::String> column;
|
|
154
|
+
v8::Global<v8::String> table;
|
|
155
|
+
v8::Global<v8::String> type;
|
|
156
|
+
v8::Global<v8::String> totalPages;
|
|
157
|
+
v8::Global<v8::String> remainingPages;
|
|
158
|
+
|
|
159
|
+
private:
|
|
160
|
+
|
|
161
|
+
static void SetString(v8::Isolate* isolate, v8::Global<v8::String>& constant, const char* str) {
|
|
162
|
+
constant.Reset(isolate, InternalizedFromLatin1(isolate, str));
|
|
163
|
+
}
|
|
164
|
+
|
|
165
|
+
void SetCode(v8::Isolate* isolate, int code, const char* str) {
|
|
166
|
+
codes.emplace(std::piecewise_construct,
|
|
167
|
+
std::forward_as_tuple(code),
|
|
168
|
+
std::forward_as_tuple(isolate, InternalizedFromLatin1(isolate, str)));
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
std::unordered_map<int, v8::Global<v8::String> > codes;
|
|
172
|
+
};
|
|
@@ -0,0 +1,121 @@
|
|
|
1
|
+
class CustomAggregate : public CustomFunction {
|
|
2
|
+
public:
|
|
3
|
+
|
|
4
|
+
explicit CustomAggregate(
|
|
5
|
+
v8::Isolate* isolate,
|
|
6
|
+
Database* db,
|
|
7
|
+
const char* name,
|
|
8
|
+
v8::Local<v8::Value> start,
|
|
9
|
+
v8::Local<v8::Function> step,
|
|
10
|
+
v8::Local<v8::Value> inverse,
|
|
11
|
+
v8::Local<v8::Value> result,
|
|
12
|
+
bool safe_ints
|
|
13
|
+
) :
|
|
14
|
+
CustomFunction(isolate, db, name, step, safe_ints),
|
|
15
|
+
invoke_result(result->IsFunction()),
|
|
16
|
+
invoke_start(start->IsFunction()),
|
|
17
|
+
inverse(isolate, inverse->IsFunction() ? inverse.As<v8::Function>() : v8::Local<v8::Function>()),
|
|
18
|
+
result(isolate, result->IsFunction() ? result.As<v8::Function>() : v8::Local<v8::Function>()),
|
|
19
|
+
start(isolate, start) {}
|
|
20
|
+
|
|
21
|
+
static void xStep(sqlite3_context* invocation, int argc, sqlite3_value** argv) {
|
|
22
|
+
xStepBase(invocation, argc, argv, &CustomAggregate::fn);
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
static void xInverse(sqlite3_context* invocation, int argc, sqlite3_value** argv) {
|
|
26
|
+
xStepBase(invocation, argc, argv, &CustomAggregate::inverse);
|
|
27
|
+
}
|
|
28
|
+
|
|
29
|
+
static void xValue(sqlite3_context* invocation) {
|
|
30
|
+
xValueBase(invocation, false);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
static void xFinal(sqlite3_context* invocation) {
|
|
34
|
+
xValueBase(invocation, true);
|
|
35
|
+
}
|
|
36
|
+
|
|
37
|
+
private:
|
|
38
|
+
|
|
39
|
+
static inline void xStepBase(sqlite3_context* invocation, int argc, sqlite3_value** argv, const v8::Global<v8::Function> CustomAggregate::*ptrtm) {
|
|
40
|
+
AGGREGATE_START();
|
|
41
|
+
|
|
42
|
+
v8::Local<v8::Value> args_fast[5];
|
|
43
|
+
v8::Local<v8::Value>* args = argc <= 4 ? args_fast : ALLOC_ARRAY<v8::Local<v8::Value>>(argc + 1);
|
|
44
|
+
args[0] = acc->value.Get(isolate);
|
|
45
|
+
if (argc != 0) Data::GetArgumentsJS(isolate, args + 1, argv, argc, self->safe_ints);
|
|
46
|
+
|
|
47
|
+
v8::MaybeLocal<v8::Value> maybeReturnValue = (self->*ptrtm).Get(isolate)->Call(OnlyContext, v8::Undefined(isolate), argc + 1, args);
|
|
48
|
+
if (args != args_fast) delete[] args;
|
|
49
|
+
|
|
50
|
+
if (maybeReturnValue.IsEmpty()) {
|
|
51
|
+
self->PropagateJSError(invocation);
|
|
52
|
+
} else {
|
|
53
|
+
v8::Local<v8::Value> returnValue = maybeReturnValue.ToLocalChecked();
|
|
54
|
+
if (!returnValue->IsUndefined()) acc->value.Reset(isolate, returnValue);
|
|
55
|
+
}
|
|
56
|
+
}
|
|
57
|
+
|
|
58
|
+
static inline void xValueBase(sqlite3_context* invocation, bool is_final) {
|
|
59
|
+
AGGREGATE_START();
|
|
60
|
+
|
|
61
|
+
if (!is_final) {
|
|
62
|
+
acc->is_window = true;
|
|
63
|
+
} else if (acc->is_window) {
|
|
64
|
+
DestroyAccumulator(invocation);
|
|
65
|
+
return;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
v8::Local<v8::Value> result = acc->value.Get(isolate);
|
|
69
|
+
if (self->invoke_result) {
|
|
70
|
+
v8::MaybeLocal<v8::Value> maybeResult = self->result.Get(isolate)->Call(OnlyContext, v8::Undefined(isolate), 1, &result);
|
|
71
|
+
if (maybeResult.IsEmpty()) {
|
|
72
|
+
self->PropagateJSError(invocation);
|
|
73
|
+
return;
|
|
74
|
+
}
|
|
75
|
+
result = maybeResult.ToLocalChecked();
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
Data::ResultValueFromJS(isolate, invocation, result, self);
|
|
79
|
+
if (is_final) DestroyAccumulator(invocation);
|
|
80
|
+
}
|
|
81
|
+
|
|
82
|
+
struct Accumulator { public:
|
|
83
|
+
v8::Global<v8::Value> value;
|
|
84
|
+
bool initialized;
|
|
85
|
+
bool is_window;
|
|
86
|
+
};
|
|
87
|
+
|
|
88
|
+
Accumulator* GetAccumulator(sqlite3_context* invocation) {
|
|
89
|
+
Accumulator* acc = static_cast<Accumulator*>(sqlite3_aggregate_context(invocation, sizeof(Accumulator)));
|
|
90
|
+
if (!acc->initialized) {
|
|
91
|
+
assert(acc->value.IsEmpty());
|
|
92
|
+
acc->initialized = true;
|
|
93
|
+
if (invoke_start) {
|
|
94
|
+
v8::MaybeLocal<v8::Value> maybeSeed = start.Get(isolate).As<v8::Function>()->Call(OnlyContext, v8::Undefined(isolate), 0, NULL);
|
|
95
|
+
if (maybeSeed.IsEmpty()) PropagateJSError(invocation);
|
|
96
|
+
else acc->value.Reset(isolate, maybeSeed.ToLocalChecked());
|
|
97
|
+
} else {
|
|
98
|
+
assert(!start.IsEmpty());
|
|
99
|
+
acc->value.Reset(isolate, start);
|
|
100
|
+
}
|
|
101
|
+
}
|
|
102
|
+
return acc;
|
|
103
|
+
}
|
|
104
|
+
|
|
105
|
+
static void DestroyAccumulator(sqlite3_context* invocation) {
|
|
106
|
+
Accumulator* acc = static_cast<Accumulator*>(sqlite3_aggregate_context(invocation, sizeof(Accumulator)));
|
|
107
|
+
assert(acc->initialized);
|
|
108
|
+
acc->value.Reset();
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
void PropagateJSError(sqlite3_context* invocation) {
|
|
112
|
+
DestroyAccumulator(invocation);
|
|
113
|
+
CustomFunction::PropagateJSError(invocation);
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
const bool invoke_result;
|
|
117
|
+
const bool invoke_start;
|
|
118
|
+
const v8::Global<v8::Function> inverse;
|
|
119
|
+
const v8::Global<v8::Function> result;
|
|
120
|
+
const v8::Global<v8::Value> start;
|
|
121
|
+
};
|
|
@@ -0,0 +1,59 @@
|
|
|
1
|
+
class CustomFunction : protected DataConverter {
|
|
2
|
+
public:
|
|
3
|
+
|
|
4
|
+
explicit CustomFunction(
|
|
5
|
+
v8::Isolate* isolate,
|
|
6
|
+
Database* db,
|
|
7
|
+
const char* name,
|
|
8
|
+
v8::Local<v8::Function> fn,
|
|
9
|
+
bool safe_ints
|
|
10
|
+
) :
|
|
11
|
+
name(name),
|
|
12
|
+
db(db),
|
|
13
|
+
isolate(isolate),
|
|
14
|
+
fn(isolate, fn),
|
|
15
|
+
safe_ints(safe_ints) {}
|
|
16
|
+
|
|
17
|
+
virtual ~CustomFunction() {}
|
|
18
|
+
|
|
19
|
+
static void xDestroy(void* self) {
|
|
20
|
+
delete static_cast<CustomFunction*>(self);
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
static void xFunc(sqlite3_context* invocation, int argc, sqlite3_value** argv) {
|
|
24
|
+
FUNCTION_START();
|
|
25
|
+
|
|
26
|
+
v8::Local<v8::Value> args_fast[4];
|
|
27
|
+
v8::Local<v8::Value>* args = NULL;
|
|
28
|
+
if (argc != 0) {
|
|
29
|
+
args = argc <= 4 ? args_fast : ALLOC_ARRAY<v8::Local<v8::Value>>(argc);
|
|
30
|
+
Data::GetArgumentsJS(isolate, args, argv, argc, self->safe_ints);
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
v8::MaybeLocal<v8::Value> maybeReturnValue = self->fn.Get(isolate)->Call(OnlyContext, v8::Undefined(isolate), argc, args);
|
|
34
|
+
if (args != args_fast) delete[] args;
|
|
35
|
+
|
|
36
|
+
if (maybeReturnValue.IsEmpty()) self->PropagateJSError(invocation);
|
|
37
|
+
else Data::ResultValueFromJS(isolate, invocation, maybeReturnValue.ToLocalChecked(), self);
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
protected:
|
|
41
|
+
|
|
42
|
+
void PropagateJSError(sqlite3_context* invocation) {
|
|
43
|
+
assert(db->GetState()->was_js_error == false);
|
|
44
|
+
db->GetState()->was_js_error = true;
|
|
45
|
+
sqlite3_result_error(invocation, "", 0);
|
|
46
|
+
}
|
|
47
|
+
|
|
48
|
+
std::string GetDataErrorPrefix() {
|
|
49
|
+
return std::string("User-defined function ") + name + "() returned";
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
private:
|
|
53
|
+
const std::string name;
|
|
54
|
+
Database* const db;
|
|
55
|
+
protected:
|
|
56
|
+
v8::Isolate* const isolate;
|
|
57
|
+
const v8::Global<v8::Function> fn;
|
|
58
|
+
const bool safe_ints;
|
|
59
|
+
};
|