better-sqlite3-multiple-ciphers 12.4.1 → 12.5.0
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/README.md +4 -4
- package/deps/sqlite3/sqlite3.c +5403 -2655
- package/deps/sqlite3/sqlite3.h +306 -113
- package/deps/sqlite3/sqlite3ext.h +7 -0
- package/deps/update-sqlite3mc.sh +2 -2
- package/package.json +2 -2
- package/src/better_sqlite3.cpp +74 -69
- package/src/objects/statement.cpp +383 -383
- package/src/util/binder.cpp +193 -193
- package/src/util/data.cpp +194 -194
- package/src/util/macros.cpp +70 -63
- package/src/util/row-builder.cpp +49 -49
package/src/util/data.cpp
CHANGED
|
@@ -1,194 +1,194 @@
|
|
|
1
|
-
#define JS_VALUE_TO_SQLITE(to, value, isolate, ...) \
|
|
2
|
-
if (value->IsNumber()) { \
|
|
3
|
-
return sqlite3_##to##_double( \
|
|
4
|
-
__VA_ARGS__, \
|
|
5
|
-
value.As<v8::Number>()->Value() \
|
|
6
|
-
); \
|
|
7
|
-
} else if (value->IsBigInt()) { \
|
|
8
|
-
bool lossless; \
|
|
9
|
-
int64_t v = value.As<v8::BigInt>()->Int64Value(&lossless); \
|
|
10
|
-
if (lossless) { \
|
|
11
|
-
return sqlite3_##to##_int64(__VA_ARGS__, v); \
|
|
12
|
-
} \
|
|
13
|
-
} else if (value->IsString()) { \
|
|
14
|
-
v8::String::Utf8Value utf8(isolate, value.As<v8::String>()); \
|
|
15
|
-
return sqlite3_##to##_text( \
|
|
16
|
-
__VA_ARGS__, \
|
|
17
|
-
*utf8, \
|
|
18
|
-
utf8.length(), \
|
|
19
|
-
SQLITE_TRANSIENT \
|
|
20
|
-
); \
|
|
21
|
-
} else if (node::Buffer::HasInstance(value)) { \
|
|
22
|
-
const char* data = node::Buffer::Data(value); \
|
|
23
|
-
return sqlite3_##to##_blob( \
|
|
24
|
-
__VA_ARGS__, \
|
|
25
|
-
data ? data : "", \
|
|
26
|
-
node::Buffer::Length(value), \
|
|
27
|
-
SQLITE_TRANSIENT \
|
|
28
|
-
); \
|
|
29
|
-
} else if (value->IsNull() || value->IsUndefined()) { \
|
|
30
|
-
return sqlite3_##to##_null(__VA_ARGS__); \
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
#define SQLITE_VALUE_TO_JS(from, isolate, safe_ints, ...) \
|
|
34
|
-
switch (sqlite3_##from##_type(__VA_ARGS__)) { \
|
|
35
|
-
case SQLITE_INTEGER: \
|
|
36
|
-
if (safe_ints) { \
|
|
37
|
-
return v8::BigInt::New( \
|
|
38
|
-
isolate, \
|
|
39
|
-
sqlite3_##from##_int64(__VA_ARGS__) \
|
|
40
|
-
); \
|
|
41
|
-
} \
|
|
42
|
-
case SQLITE_FLOAT: \
|
|
43
|
-
return v8::Number::New( \
|
|
44
|
-
isolate, \
|
|
45
|
-
sqlite3_##from##_double(__VA_ARGS__) \
|
|
46
|
-
); \
|
|
47
|
-
case SQLITE_TEXT: \
|
|
48
|
-
return StringFromUtf8( \
|
|
49
|
-
isolate, \
|
|
50
|
-
reinterpret_cast<const char*>(sqlite3_##from##_text(__VA_ARGS__)), \
|
|
51
|
-
sqlite3_##from##_bytes(__VA_ARGS__) \
|
|
52
|
-
); \
|
|
53
|
-
case SQLITE_BLOB: \
|
|
54
|
-
return node::Buffer::Copy( \
|
|
55
|
-
isolate, \
|
|
56
|
-
static_cast<const char*>(sqlite3_##from##_blob(__VA_ARGS__)), \
|
|
57
|
-
sqlite3_##from##_bytes(__VA_ARGS__) \
|
|
58
|
-
).ToLocalChecked(); \
|
|
59
|
-
default: \
|
|
60
|
-
assert(sqlite3_##from##_type(__VA_ARGS__) == SQLITE_NULL); \
|
|
61
|
-
return v8::Null(isolate); \
|
|
62
|
-
} \
|
|
63
|
-
assert(false);
|
|
64
|
-
|
|
65
|
-
namespace Data {
|
|
66
|
-
|
|
67
|
-
static const char FLAT = 0;
|
|
68
|
-
static const char PLUCK = 1;
|
|
69
|
-
static const char EXPAND = 2;
|
|
70
|
-
static const char RAW = 3;
|
|
71
|
-
|
|
72
|
-
v8::Local<v8::Value> GetValueJS(v8::Isolate* isolate, sqlite3_stmt* handle, int column, bool safe_ints) {
|
|
73
|
-
SQLITE_VALUE_TO_JS(column, isolate, safe_ints, handle, column);
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
v8::Local<v8::Value> GetValueJS(v8::Isolate* isolate, sqlite3_value* value, bool safe_ints) {
|
|
77
|
-
SQLITE_VALUE_TO_JS(value, isolate, safe_ints, value);
|
|
78
|
-
}
|
|
79
|
-
|
|
80
|
-
v8::Local<v8::Value> GetExpandedRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints) {
|
|
81
|
-
v8::Local<v8::Object> row = v8::Object::New(isolate);
|
|
82
|
-
int column_count = sqlite3_column_count(handle);
|
|
83
|
-
for (int i = 0; i < column_count; ++i) {
|
|
84
|
-
const char* table_raw = sqlite3_column_table_name(handle, i);
|
|
85
|
-
v8::Local<v8::String> table = InternalizedFromUtf8(isolate, table_raw == NULL ? "$" : table_raw, -1);
|
|
86
|
-
v8::Local<v8::String> column = InternalizedFromUtf8(isolate, sqlite3_column_name(handle, i), -1);
|
|
87
|
-
v8::Local<v8::Value> value = Data::GetValueJS(isolate, handle, i, safe_ints);
|
|
88
|
-
if (row->HasOwnProperty(ctx, table).FromJust()) {
|
|
89
|
-
row->Get(ctx, table).ToLocalChecked().As<v8::Object>()->Set(ctx, column, value).FromJust();
|
|
90
|
-
} else {
|
|
91
|
-
v8::Local<v8::Object> nested = v8::Object::New(isolate);
|
|
92
|
-
row->Set(ctx, table, nested).FromJust();
|
|
93
|
-
nested->Set(ctx, column, value).FromJust();
|
|
94
|
-
}
|
|
95
|
-
}
|
|
96
|
-
return row;
|
|
97
|
-
}
|
|
98
|
-
|
|
99
|
-
#if !defined(NODE_MODULE_VERSION) || NODE_MODULE_VERSION < 127
|
|
100
|
-
|
|
101
|
-
v8::Local<v8::Value> GetFlatRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints) {
|
|
102
|
-
v8::Local<v8::Object> row = v8::Object::New(isolate);
|
|
103
|
-
int column_count = sqlite3_column_count(handle);
|
|
104
|
-
for (int i = 0; i < column_count; ++i) {
|
|
105
|
-
row->Set(ctx,
|
|
106
|
-
InternalizedFromUtf8(isolate, sqlite3_column_name(handle, i), -1),
|
|
107
|
-
Data::GetValueJS(isolate, handle, i, safe_ints)
|
|
108
|
-
).FromJust();
|
|
109
|
-
}
|
|
110
|
-
return row;
|
|
111
|
-
}
|
|
112
|
-
|
|
113
|
-
v8::Local<v8::Value> GetRawRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints) {
|
|
114
|
-
v8::Local<v8::Array> row = v8::Array::New(isolate);
|
|
115
|
-
int column_count = sqlite3_column_count(handle);
|
|
116
|
-
for (int i = 0; i < column_count; ++i) {
|
|
117
|
-
row->Set(ctx, i, Data::GetValueJS(isolate, handle, i, safe_ints)).FromJust();
|
|
118
|
-
}
|
|
119
|
-
return row;
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
v8::Local<v8::Value> GetRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints, char mode) {
|
|
123
|
-
if (mode == FLAT) return GetFlatRowJS(isolate, ctx, handle, safe_ints);
|
|
124
|
-
if (mode == PLUCK) return GetValueJS(isolate, handle, 0, safe_ints);
|
|
125
|
-
if (mode == EXPAND) return GetExpandedRowJS(isolate, ctx, handle, safe_ints);
|
|
126
|
-
if (mode == RAW) return GetRawRowJS(isolate, ctx, handle, safe_ints);
|
|
127
|
-
assert(false);
|
|
128
|
-
return v8::Local<v8::Value>();
|
|
129
|
-
}
|
|
130
|
-
|
|
131
|
-
#else
|
|
132
|
-
|
|
133
|
-
v8::Local<v8::Value> GetFlatRowJS(v8::Isolate* isolate, sqlite3_stmt* handle, bool safe_ints) {
|
|
134
|
-
int column_count = sqlite3_column_count(handle);
|
|
135
|
-
v8::LocalVector<v8::Name> keys(isolate);
|
|
136
|
-
v8::LocalVector<v8::Value> values(isolate);
|
|
137
|
-
keys.reserve(column_count);
|
|
138
|
-
values.reserve(column_count);
|
|
139
|
-
for (int i = 0; i < column_count; ++i) {
|
|
140
|
-
keys.emplace_back(
|
|
141
|
-
InternalizedFromUtf8(isolate, sqlite3_column_name(handle, i), -1).As<v8::Name>()
|
|
142
|
-
);
|
|
143
|
-
values.emplace_back(
|
|
144
|
-
Data::GetValueJS(isolate, handle, i, safe_ints)
|
|
145
|
-
);
|
|
146
|
-
}
|
|
147
|
-
return v8::Object::New(
|
|
148
|
-
isolate,
|
|
149
|
-
v8::Object::New(isolate)
|
|
150
|
-
keys.data(),
|
|
151
|
-
values.data(),
|
|
152
|
-
column_count
|
|
153
|
-
);
|
|
154
|
-
}
|
|
155
|
-
|
|
156
|
-
v8::Local<v8::Value> GetRawRowJS(v8::Isolate* isolate, sqlite3_stmt* handle, bool safe_ints) {
|
|
157
|
-
int column_count = sqlite3_column_count(handle);
|
|
158
|
-
v8::LocalVector<v8::Value> row(isolate);
|
|
159
|
-
row.reserve(column_count);
|
|
160
|
-
for (int i = 0; i < column_count; ++i) {
|
|
161
|
-
row.emplace_back(Data::GetValueJS(isolate, handle, i, safe_ints));
|
|
162
|
-
}
|
|
163
|
-
return v8::Array::New(isolate, row.data(), row.size());
|
|
164
|
-
}
|
|
165
|
-
|
|
166
|
-
v8::Local<v8::Value> GetRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints, char mode) {
|
|
167
|
-
if (mode == FLAT) return GetFlatRowJS(isolate, handle, safe_ints);
|
|
168
|
-
if (mode == PLUCK) return GetValueJS(isolate, handle, 0, safe_ints);
|
|
169
|
-
if (mode == EXPAND) return GetExpandedRowJS(isolate, ctx, handle, safe_ints);
|
|
170
|
-
if (mode == RAW) return GetRawRowJS(isolate, handle, safe_ints);
|
|
171
|
-
assert(false);
|
|
172
|
-
return v8::Local<v8::Value>();
|
|
173
|
-
}
|
|
174
|
-
|
|
175
|
-
#endif
|
|
176
|
-
|
|
177
|
-
void GetArgumentsJS(v8::Isolate* isolate, v8::Local<v8::Value>* out, sqlite3_value** values, int argument_count, bool safe_ints) {
|
|
178
|
-
assert(argument_count > 0);
|
|
179
|
-
for (int i = 0; i < argument_count; ++i) {
|
|
180
|
-
out[i] = Data::GetValueJS(isolate, values[i], safe_ints);
|
|
181
|
-
}
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
int BindValueFromJS(v8::Isolate* isolate, sqlite3_stmt* handle, int index, v8::Local<v8::Value> value) {
|
|
185
|
-
JS_VALUE_TO_SQLITE(bind, value, isolate, handle, index);
|
|
186
|
-
return value->IsBigInt() ? SQLITE_TOOBIG : -1;
|
|
187
|
-
}
|
|
188
|
-
|
|
189
|
-
void ResultValueFromJS(v8::Isolate* isolate, sqlite3_context* invocation, v8::Local<v8::Value> value, DataConverter* converter) {
|
|
190
|
-
JS_VALUE_TO_SQLITE(result, value, isolate, invocation);
|
|
191
|
-
converter->ThrowDataConversionError(invocation, value->IsBigInt());
|
|
192
|
-
}
|
|
193
|
-
|
|
194
|
-
}
|
|
1
|
+
#define JS_VALUE_TO_SQLITE(to, value, isolate, ...) \
|
|
2
|
+
if (value->IsNumber()) { \
|
|
3
|
+
return sqlite3_##to##_double( \
|
|
4
|
+
__VA_ARGS__, \
|
|
5
|
+
value.As<v8::Number>()->Value() \
|
|
6
|
+
); \
|
|
7
|
+
} else if (value->IsBigInt()) { \
|
|
8
|
+
bool lossless; \
|
|
9
|
+
int64_t v = value.As<v8::BigInt>()->Int64Value(&lossless); \
|
|
10
|
+
if (lossless) { \
|
|
11
|
+
return sqlite3_##to##_int64(__VA_ARGS__, v); \
|
|
12
|
+
} \
|
|
13
|
+
} else if (value->IsString()) { \
|
|
14
|
+
v8::String::Utf8Value utf8(isolate, value.As<v8::String>()); \
|
|
15
|
+
return sqlite3_##to##_text( \
|
|
16
|
+
__VA_ARGS__, \
|
|
17
|
+
*utf8, \
|
|
18
|
+
utf8.length(), \
|
|
19
|
+
SQLITE_TRANSIENT \
|
|
20
|
+
); \
|
|
21
|
+
} else if (node::Buffer::HasInstance(value)) { \
|
|
22
|
+
const char* data = node::Buffer::Data(value); \
|
|
23
|
+
return sqlite3_##to##_blob( \
|
|
24
|
+
__VA_ARGS__, \
|
|
25
|
+
data ? data : "", \
|
|
26
|
+
node::Buffer::Length(value), \
|
|
27
|
+
SQLITE_TRANSIENT \
|
|
28
|
+
); \
|
|
29
|
+
} else if (value->IsNull() || value->IsUndefined()) { \
|
|
30
|
+
return sqlite3_##to##_null(__VA_ARGS__); \
|
|
31
|
+
}
|
|
32
|
+
|
|
33
|
+
#define SQLITE_VALUE_TO_JS(from, isolate, safe_ints, ...) \
|
|
34
|
+
switch (sqlite3_##from##_type(__VA_ARGS__)) { \
|
|
35
|
+
case SQLITE_INTEGER: \
|
|
36
|
+
if (safe_ints) { \
|
|
37
|
+
return v8::BigInt::New( \
|
|
38
|
+
isolate, \
|
|
39
|
+
sqlite3_##from##_int64(__VA_ARGS__) \
|
|
40
|
+
); \
|
|
41
|
+
} \
|
|
42
|
+
case SQLITE_FLOAT: \
|
|
43
|
+
return v8::Number::New( \
|
|
44
|
+
isolate, \
|
|
45
|
+
sqlite3_##from##_double(__VA_ARGS__) \
|
|
46
|
+
); \
|
|
47
|
+
case SQLITE_TEXT: \
|
|
48
|
+
return StringFromUtf8( \
|
|
49
|
+
isolate, \
|
|
50
|
+
reinterpret_cast<const char*>(sqlite3_##from##_text(__VA_ARGS__)), \
|
|
51
|
+
sqlite3_##from##_bytes(__VA_ARGS__) \
|
|
52
|
+
); \
|
|
53
|
+
case SQLITE_BLOB: \
|
|
54
|
+
return node::Buffer::Copy( \
|
|
55
|
+
isolate, \
|
|
56
|
+
static_cast<const char*>(sqlite3_##from##_blob(__VA_ARGS__)), \
|
|
57
|
+
sqlite3_##from##_bytes(__VA_ARGS__) \
|
|
58
|
+
).ToLocalChecked(); \
|
|
59
|
+
default: \
|
|
60
|
+
assert(sqlite3_##from##_type(__VA_ARGS__) == SQLITE_NULL); \
|
|
61
|
+
return v8::Null(isolate); \
|
|
62
|
+
} \
|
|
63
|
+
assert(false);
|
|
64
|
+
|
|
65
|
+
namespace Data {
|
|
66
|
+
|
|
67
|
+
static const char FLAT = 0;
|
|
68
|
+
static const char PLUCK = 1;
|
|
69
|
+
static const char EXPAND = 2;
|
|
70
|
+
static const char RAW = 3;
|
|
71
|
+
|
|
72
|
+
v8::Local<v8::Value> GetValueJS(v8::Isolate* isolate, sqlite3_stmt* handle, int column, bool safe_ints) {
|
|
73
|
+
SQLITE_VALUE_TO_JS(column, isolate, safe_ints, handle, column);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
v8::Local<v8::Value> GetValueJS(v8::Isolate* isolate, sqlite3_value* value, bool safe_ints) {
|
|
77
|
+
SQLITE_VALUE_TO_JS(value, isolate, safe_ints, value);
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
v8::Local<v8::Value> GetExpandedRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints) {
|
|
81
|
+
v8::Local<v8::Object> row = v8::Object::New(isolate);
|
|
82
|
+
int column_count = sqlite3_column_count(handle);
|
|
83
|
+
for (int i = 0; i < column_count; ++i) {
|
|
84
|
+
const char* table_raw = sqlite3_column_table_name(handle, i);
|
|
85
|
+
v8::Local<v8::String> table = InternalizedFromUtf8(isolate, table_raw == NULL ? "$" : table_raw, -1);
|
|
86
|
+
v8::Local<v8::String> column = InternalizedFromUtf8(isolate, sqlite3_column_name(handle, i), -1);
|
|
87
|
+
v8::Local<v8::Value> value = Data::GetValueJS(isolate, handle, i, safe_ints);
|
|
88
|
+
if (row->HasOwnProperty(ctx, table).FromJust()) {
|
|
89
|
+
row->Get(ctx, table).ToLocalChecked().As<v8::Object>()->Set(ctx, column, value).FromJust();
|
|
90
|
+
} else {
|
|
91
|
+
v8::Local<v8::Object> nested = v8::Object::New(isolate);
|
|
92
|
+
row->Set(ctx, table, nested).FromJust();
|
|
93
|
+
nested->Set(ctx, column, value).FromJust();
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
return row;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
#if !defined(NODE_MODULE_VERSION) || NODE_MODULE_VERSION < 127
|
|
100
|
+
|
|
101
|
+
v8::Local<v8::Value> GetFlatRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints) {
|
|
102
|
+
v8::Local<v8::Object> row = v8::Object::New(isolate);
|
|
103
|
+
int column_count = sqlite3_column_count(handle);
|
|
104
|
+
for (int i = 0; i < column_count; ++i) {
|
|
105
|
+
row->Set(ctx,
|
|
106
|
+
InternalizedFromUtf8(isolate, sqlite3_column_name(handle, i), -1),
|
|
107
|
+
Data::GetValueJS(isolate, handle, i, safe_ints)
|
|
108
|
+
).FromJust();
|
|
109
|
+
}
|
|
110
|
+
return row;
|
|
111
|
+
}
|
|
112
|
+
|
|
113
|
+
v8::Local<v8::Value> GetRawRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints) {
|
|
114
|
+
v8::Local<v8::Array> row = v8::Array::New(isolate);
|
|
115
|
+
int column_count = sqlite3_column_count(handle);
|
|
116
|
+
for (int i = 0; i < column_count; ++i) {
|
|
117
|
+
row->Set(ctx, i, Data::GetValueJS(isolate, handle, i, safe_ints)).FromJust();
|
|
118
|
+
}
|
|
119
|
+
return row;
|
|
120
|
+
}
|
|
121
|
+
|
|
122
|
+
v8::Local<v8::Value> GetRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints, char mode) {
|
|
123
|
+
if (mode == FLAT) return GetFlatRowJS(isolate, ctx, handle, safe_ints);
|
|
124
|
+
if (mode == PLUCK) return GetValueJS(isolate, handle, 0, safe_ints);
|
|
125
|
+
if (mode == EXPAND) return GetExpandedRowJS(isolate, ctx, handle, safe_ints);
|
|
126
|
+
if (mode == RAW) return GetRawRowJS(isolate, ctx, handle, safe_ints);
|
|
127
|
+
assert(false);
|
|
128
|
+
return v8::Local<v8::Value>();
|
|
129
|
+
}
|
|
130
|
+
|
|
131
|
+
#else
|
|
132
|
+
|
|
133
|
+
v8::Local<v8::Value> GetFlatRowJS(v8::Isolate* isolate, sqlite3_stmt* handle, bool safe_ints) {
|
|
134
|
+
int column_count = sqlite3_column_count(handle);
|
|
135
|
+
v8::LocalVector<v8::Name> keys(isolate);
|
|
136
|
+
v8::LocalVector<v8::Value> values(isolate);
|
|
137
|
+
keys.reserve(column_count);
|
|
138
|
+
values.reserve(column_count);
|
|
139
|
+
for (int i = 0; i < column_count; ++i) {
|
|
140
|
+
keys.emplace_back(
|
|
141
|
+
InternalizedFromUtf8(isolate, sqlite3_column_name(handle, i), -1).As<v8::Name>()
|
|
142
|
+
);
|
|
143
|
+
values.emplace_back(
|
|
144
|
+
Data::GetValueJS(isolate, handle, i, safe_ints)
|
|
145
|
+
);
|
|
146
|
+
}
|
|
147
|
+
return v8::Object::New(
|
|
148
|
+
isolate,
|
|
149
|
+
GET_PROTOTYPE(v8::Object::New(isolate)),
|
|
150
|
+
keys.data(),
|
|
151
|
+
values.data(),
|
|
152
|
+
column_count
|
|
153
|
+
);
|
|
154
|
+
}
|
|
155
|
+
|
|
156
|
+
v8::Local<v8::Value> GetRawRowJS(v8::Isolate* isolate, sqlite3_stmt* handle, bool safe_ints) {
|
|
157
|
+
int column_count = sqlite3_column_count(handle);
|
|
158
|
+
v8::LocalVector<v8::Value> row(isolate);
|
|
159
|
+
row.reserve(column_count);
|
|
160
|
+
for (int i = 0; i < column_count; ++i) {
|
|
161
|
+
row.emplace_back(Data::GetValueJS(isolate, handle, i, safe_ints));
|
|
162
|
+
}
|
|
163
|
+
return v8::Array::New(isolate, row.data(), row.size());
|
|
164
|
+
}
|
|
165
|
+
|
|
166
|
+
v8::Local<v8::Value> GetRowJS(v8::Isolate* isolate, v8::Local<v8::Context> ctx, sqlite3_stmt* handle, bool safe_ints, char mode) {
|
|
167
|
+
if (mode == FLAT) return GetFlatRowJS(isolate, handle, safe_ints);
|
|
168
|
+
if (mode == PLUCK) return GetValueJS(isolate, handle, 0, safe_ints);
|
|
169
|
+
if (mode == EXPAND) return GetExpandedRowJS(isolate, ctx, handle, safe_ints);
|
|
170
|
+
if (mode == RAW) return GetRawRowJS(isolate, handle, safe_ints);
|
|
171
|
+
assert(false);
|
|
172
|
+
return v8::Local<v8::Value>();
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
#endif
|
|
176
|
+
|
|
177
|
+
void GetArgumentsJS(v8::Isolate* isolate, v8::Local<v8::Value>* out, sqlite3_value** values, int argument_count, bool safe_ints) {
|
|
178
|
+
assert(argument_count > 0);
|
|
179
|
+
for (int i = 0; i < argument_count; ++i) {
|
|
180
|
+
out[i] = Data::GetValueJS(isolate, values[i], safe_ints);
|
|
181
|
+
}
|
|
182
|
+
}
|
|
183
|
+
|
|
184
|
+
int BindValueFromJS(v8::Isolate* isolate, sqlite3_stmt* handle, int index, v8::Local<v8::Value> value) {
|
|
185
|
+
JS_VALUE_TO_SQLITE(bind, value, isolate, handle, index);
|
|
186
|
+
return value->IsBigInt() ? SQLITE_TOOBIG : -1;
|
|
187
|
+
}
|
|
188
|
+
|
|
189
|
+
void ResultValueFromJS(v8::Isolate* isolate, sqlite3_context* invocation, v8::Local<v8::Value> value, DataConverter* converter) {
|
|
190
|
+
JS_VALUE_TO_SQLITE(result, value, isolate, invocation);
|
|
191
|
+
converter->ThrowDataConversionError(invocation, value->IsBigInt());
|
|
192
|
+
}
|
|
193
|
+
|
|
194
|
+
}
|
package/src/util/macros.cpp
CHANGED
|
@@ -1,63 +1,70 @@
|
|
|
1
|
-
#define NODE_ARGUMENTS const v8::FunctionCallbackInfo<v8::Value>&
|
|
2
|
-
#define NODE_ARGUMENTS_POINTER const v8::FunctionCallbackInfo<v8::Value>*
|
|
3
|
-
#define NODE_METHOD(name) void name(NODE_ARGUMENTS info)
|
|
4
|
-
#define NODE_GETTER(name) void name(v8::Local<v8::Name> _, const v8::PropertyCallbackInfo<v8::Value>& info)
|
|
5
|
-
#define INIT(name) v8::Local<v8::Function> name(v8::Isolate* isolate, v8::Local<v8::External> data)
|
|
6
|
-
|
|
7
|
-
#
|
|
8
|
-
|
|
9
|
-
#define
|
|
10
|
-
#
|
|
11
|
-
#define
|
|
12
|
-
#
|
|
13
|
-
|
|
14
|
-
#define
|
|
15
|
-
|
|
16
|
-
#define
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
|
|
20
|
-
|
|
21
|
-
#define
|
|
22
|
-
|
|
23
|
-
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
#define
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
#define
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
42
|
-
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
#define
|
|
51
|
-
if (
|
|
52
|
-
return ThrowTypeError("This
|
|
53
|
-
|
|
54
|
-
|
|
55
|
-
|
|
56
|
-
|
|
57
|
-
#define
|
|
58
|
-
|
|
59
|
-
|
|
60
|
-
|
|
61
|
-
#define
|
|
62
|
-
#define
|
|
63
|
-
#define
|
|
1
|
+
#define NODE_ARGUMENTS const v8::FunctionCallbackInfo<v8::Value>&
|
|
2
|
+
#define NODE_ARGUMENTS_POINTER const v8::FunctionCallbackInfo<v8::Value>*
|
|
3
|
+
#define NODE_METHOD(name) void name(NODE_ARGUMENTS info)
|
|
4
|
+
#define NODE_GETTER(name) void name(v8::Local<v8::Name> _, const v8::PropertyCallbackInfo<v8::Value>& info)
|
|
5
|
+
#define INIT(name) v8::Local<v8::Function> name(v8::Isolate* isolate, v8::Local<v8::External> data)
|
|
6
|
+
|
|
7
|
+
#if defined(V8_MAJOR_VERSION) && V8_MAJOR_VERSION >= 13
|
|
8
|
+
// v8::Object::GetPrototype has been deprecated. See http://crbug.com/333672197
|
|
9
|
+
#define GET_PROTOTYPE(obj) ((obj)->GetPrototypeV2())
|
|
10
|
+
#else
|
|
11
|
+
#define GET_PROTOTYPE(obj) ((obj)->GetPrototype())
|
|
12
|
+
#endif
|
|
13
|
+
|
|
14
|
+
#define EasyIsolate v8::Isolate* isolate = v8::Isolate::GetCurrent()
|
|
15
|
+
#define OnlyIsolate info.GetIsolate()
|
|
16
|
+
#define OnlyContext isolate->GetCurrentContext()
|
|
17
|
+
#define OnlyAddon static_cast<Addon*>(info.Data().As<v8::External>()->Value())
|
|
18
|
+
#define UseIsolate v8::Isolate* isolate = OnlyIsolate
|
|
19
|
+
#define UseContext v8::Local<v8::Context> ctx = OnlyContext
|
|
20
|
+
#define UseAddon Addon* addon = OnlyAddon
|
|
21
|
+
#define Unwrap node::ObjectWrap::Unwrap
|
|
22
|
+
|
|
23
|
+
#define REQUIRE_ARGUMENT_ANY(at, var) \
|
|
24
|
+
if (info.Length() <= (at())) \
|
|
25
|
+
return ThrowTypeError("Expected a "#at" argument"); \
|
|
26
|
+
var = info[at()]
|
|
27
|
+
|
|
28
|
+
#define _REQUIRE_ARGUMENT(at, var, Type, message, ...) \
|
|
29
|
+
if (info.Length() <= (at()) || !info[at()]->Is##Type()) \
|
|
30
|
+
return ThrowTypeError("Expected "#at" argument to be "#message); \
|
|
31
|
+
var = (info[at()].As<v8::Type>())__VA_ARGS__
|
|
32
|
+
|
|
33
|
+
#define REQUIRE_ARGUMENT_INT32(at, var) \
|
|
34
|
+
_REQUIRE_ARGUMENT(at, var, Int32, a 32-bit signed integer, ->Value())
|
|
35
|
+
#define REQUIRE_ARGUMENT_BOOLEAN(at, var) \
|
|
36
|
+
_REQUIRE_ARGUMENT(at, var, Boolean, a boolean, ->Value())
|
|
37
|
+
#define REQUIRE_ARGUMENT_STRING(at, var) \
|
|
38
|
+
_REQUIRE_ARGUMENT(at, var, String, a string)
|
|
39
|
+
#define REQUIRE_ARGUMENT_OBJECT(at, var) \
|
|
40
|
+
_REQUIRE_ARGUMENT(at, var, Object, an object)
|
|
41
|
+
#define REQUIRE_ARGUMENT_FUNCTION(at, var) \
|
|
42
|
+
_REQUIRE_ARGUMENT(at, var, Function, a function)
|
|
43
|
+
|
|
44
|
+
#define REQUIRE_DATABASE_OPEN(db) \
|
|
45
|
+
if (!db->open) \
|
|
46
|
+
return ThrowTypeError("The database connection is not open")
|
|
47
|
+
#define REQUIRE_DATABASE_NOT_BUSY(db) \
|
|
48
|
+
if (db->busy) \
|
|
49
|
+
return ThrowTypeError("This database connection is busy executing a query")
|
|
50
|
+
#define REQUIRE_DATABASE_NO_ITERATORS(db) \
|
|
51
|
+
if (db->iterators) \
|
|
52
|
+
return ThrowTypeError("This database connection is busy executing a query")
|
|
53
|
+
#define REQUIRE_DATABASE_NO_ITERATORS_UNLESS_UNSAFE(db) \
|
|
54
|
+
if (!db->unsafe_mode) { \
|
|
55
|
+
REQUIRE_DATABASE_NO_ITERATORS(db); \
|
|
56
|
+
} ((void)0)
|
|
57
|
+
#define REQUIRE_STATEMENT_NOT_LOCKED(stmt) \
|
|
58
|
+
if (stmt->locked) \
|
|
59
|
+
return ThrowTypeError("This statement is busy executing a query")
|
|
60
|
+
|
|
61
|
+
#define first() 0
|
|
62
|
+
#define second() 1
|
|
63
|
+
#define third() 2
|
|
64
|
+
#define fourth() 3
|
|
65
|
+
#define fifth() 4
|
|
66
|
+
#define sixth() 5
|
|
67
|
+
#define seventh() 6
|
|
68
|
+
#define eighth() 7
|
|
69
|
+
#define ninth() 8
|
|
70
|
+
#define tenth() 9
|
package/src/util/row-builder.cpp
CHANGED
|
@@ -1,49 +1,49 @@
|
|
|
1
|
-
class RowBuilder {
|
|
2
|
-
public:
|
|
3
|
-
|
|
4
|
-
explicit RowBuilder(
|
|
5
|
-
v8::Isolate* isolate,
|
|
6
|
-
sqlite3_stmt* handle,
|
|
7
|
-
bool safe_ints
|
|
8
|
-
) :
|
|
9
|
-
isolate(isolate),
|
|
10
|
-
handle(handle),
|
|
11
|
-
column_count(-1),
|
|
12
|
-
safe_ints(safe_ints),
|
|
13
|
-
keys(isolate) {}
|
|
14
|
-
|
|
15
|
-
v8::Local<v8::Value> GetRowJS() {
|
|
16
|
-
if (column_count < 0) {
|
|
17
|
-
column_count = sqlite3_column_count(handle);
|
|
18
|
-
keys.reserve(column_count);
|
|
19
|
-
for (int i = 0; i < column_count; ++i) {
|
|
20
|
-
keys.emplace_back(
|
|
21
|
-
InternalizedFromUtf8(isolate, sqlite3_column_name(handle, i), -1)
|
|
22
|
-
.As<v8::Name>()
|
|
23
|
-
);
|
|
24
|
-
}
|
|
25
|
-
}
|
|
26
|
-
|
|
27
|
-
v8::LocalVector<v8::Value> values(isolate);
|
|
28
|
-
values.reserve(column_count);
|
|
29
|
-
for (int i = 0; i < column_count; ++i) {
|
|
30
|
-
values.emplace_back(
|
|
31
|
-
Data::GetValueJS(isolate, handle, i, safe_ints)
|
|
32
|
-
);
|
|
33
|
-
}
|
|
34
|
-
|
|
35
|
-
return v8::Object::New(isolate,
|
|
36
|
-
v8::Object::New(isolate)
|
|
37
|
-
keys.data(),
|
|
38
|
-
values.data(),
|
|
39
|
-
column_count
|
|
40
|
-
);
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
private:
|
|
44
|
-
v8::Isolate* isolate;
|
|
45
|
-
sqlite3_stmt* handle;
|
|
46
|
-
int column_count;
|
|
47
|
-
const bool safe_ints;
|
|
48
|
-
v8::LocalVector<v8::Name> keys;
|
|
49
|
-
};
|
|
1
|
+
class RowBuilder {
|
|
2
|
+
public:
|
|
3
|
+
|
|
4
|
+
explicit RowBuilder(
|
|
5
|
+
v8::Isolate* isolate,
|
|
6
|
+
sqlite3_stmt* handle,
|
|
7
|
+
bool safe_ints
|
|
8
|
+
) :
|
|
9
|
+
isolate(isolate),
|
|
10
|
+
handle(handle),
|
|
11
|
+
column_count(-1),
|
|
12
|
+
safe_ints(safe_ints),
|
|
13
|
+
keys(isolate) {}
|
|
14
|
+
|
|
15
|
+
v8::Local<v8::Value> GetRowJS() {
|
|
16
|
+
if (column_count < 0) {
|
|
17
|
+
column_count = sqlite3_column_count(handle);
|
|
18
|
+
keys.reserve(column_count);
|
|
19
|
+
for (int i = 0; i < column_count; ++i) {
|
|
20
|
+
keys.emplace_back(
|
|
21
|
+
InternalizedFromUtf8(isolate, sqlite3_column_name(handle, i), -1)
|
|
22
|
+
.As<v8::Name>()
|
|
23
|
+
);
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
|
|
27
|
+
v8::LocalVector<v8::Value> values(isolate);
|
|
28
|
+
values.reserve(column_count);
|
|
29
|
+
for (int i = 0; i < column_count; ++i) {
|
|
30
|
+
values.emplace_back(
|
|
31
|
+
Data::GetValueJS(isolate, handle, i, safe_ints)
|
|
32
|
+
);
|
|
33
|
+
}
|
|
34
|
+
|
|
35
|
+
return v8::Object::New(isolate,
|
|
36
|
+
GET_PROTOTYPE(v8::Object::New(isolate)),
|
|
37
|
+
keys.data(),
|
|
38
|
+
values.data(),
|
|
39
|
+
column_count
|
|
40
|
+
);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
private:
|
|
44
|
+
v8::Isolate* isolate;
|
|
45
|
+
sqlite3_stmt* handle;
|
|
46
|
+
int column_count;
|
|
47
|
+
const bool safe_ints;
|
|
48
|
+
v8::LocalVector<v8::Name> keys;
|
|
49
|
+
};
|