@rocicorp/zero-sqlite3 1.0.9 → 1.0.11

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.
@@ -0,0 +1,109 @@
1
+ inline v8::Local<v8::String> StringFromUtf8(v8::Isolate* isolate, const char* data, int length) {
2
+ return v8::String::NewFromUtf8(isolate, data, v8::NewStringType::kNormal, length).ToLocalChecked();
3
+ }
4
+
5
+ inline v8::Local<v8::String> InternalizedFromUtf8(v8::Isolate* isolate, const char* data, int length) {
6
+ return v8::String::NewFromUtf8(isolate, data, v8::NewStringType::kInternalized, length).ToLocalChecked();
7
+ }
8
+
9
+ inline v8::Local<v8::Value> InternalizedFromUtf8OrNull(v8::Isolate* isolate, const char* data, int length) {
10
+ if (data == NULL) return v8::Null(isolate);
11
+ return InternalizedFromUtf8(isolate, data, length);
12
+ }
13
+
14
+ inline v8::Local<v8::String> InternalizedFromLatin1(v8::Isolate* isolate, const char* str) {
15
+ return v8::String::NewFromOneByte(isolate, reinterpret_cast<const uint8_t*>(str), v8::NewStringType::kInternalized).ToLocalChecked();
16
+ }
17
+
18
+ inline void SetFrozen(v8::Isolate* isolate, v8::Local<v8::Context> ctx, v8::Local<v8::Object> obj, v8::Global<v8::String>& key, v8::Local<v8::Value> value) {
19
+ obj->DefineOwnProperty(ctx, key.Get(isolate), value, static_cast<v8::PropertyAttribute>(v8::DontDelete | v8::ReadOnly)).FromJust();
20
+ }
21
+
22
+ void ThrowError(const char* message) { EasyIsolate; isolate->ThrowException(v8::Exception::Error(StringFromUtf8(isolate, message, -1))); }
23
+ void ThrowTypeError(const char* message) { EasyIsolate; isolate->ThrowException(v8::Exception::TypeError(StringFromUtf8(isolate, message, -1))); }
24
+ void ThrowRangeError(const char* message) { EasyIsolate; isolate->ThrowException(v8::Exception::RangeError(StringFromUtf8(isolate, message, -1))); }
25
+
26
+ // Determines whether to skip the given character at the start of an SQL string.
27
+ inline bool IS_SKIPPED(char c) {
28
+ return c == ' ' || c == ';' || (c >= '\t' && c <= '\r');
29
+ }
30
+
31
+ // Allocates an empty array, without calling constructors/initializers.
32
+ template<class T> inline T* ALLOC_ARRAY(size_t count) {
33
+ return static_cast<T*>(::operator new[](count * sizeof(T)));
34
+ }
35
+
36
+ // Deallocates an array, without calling destructors.
37
+ template<class T> inline void FREE_ARRAY(T* array_pointer) {
38
+ ::operator delete[](array_pointer);
39
+ }
40
+
41
+ v8::Local<v8::FunctionTemplate> NewConstructorTemplate(
42
+ v8::Isolate* isolate,
43
+ v8::Local<v8::External> data,
44
+ v8::FunctionCallback func,
45
+ const char* name
46
+ ) {
47
+ v8::Local<v8::FunctionTemplate> t = v8::FunctionTemplate::New(isolate, func, data);
48
+ t->InstanceTemplate()->SetInternalFieldCount(1);
49
+ t->SetClassName(InternalizedFromLatin1(isolate, name));
50
+ return t;
51
+ }
52
+
53
+ void SetPrototypeMethod(
54
+ v8::Isolate* isolate,
55
+ v8::Local<v8::External> data,
56
+ v8::Local<v8::FunctionTemplate> recv,
57
+ const char* name,
58
+ v8::FunctionCallback func
59
+ ) {
60
+ v8::HandleScope scope(isolate);
61
+ recv->PrototypeTemplate()->Set(
62
+ InternalizedFromLatin1(isolate, name),
63
+ v8::FunctionTemplate::New(isolate, func, data, v8::Signature::New(isolate, recv))
64
+ );
65
+ }
66
+
67
+ void SetPrototypeSymbolMethod(
68
+ v8::Isolate* isolate,
69
+ v8::Local<v8::External> data,
70
+ v8::Local<v8::FunctionTemplate> recv,
71
+ v8::Local<v8::Symbol> symbol,
72
+ v8::FunctionCallback func
73
+ ) {
74
+ v8::HandleScope scope(isolate);
75
+ recv->PrototypeTemplate()->Set(
76
+ symbol,
77
+ v8::FunctionTemplate::New(isolate, func, data, v8::Signature::New(isolate, recv))
78
+ );
79
+ }
80
+
81
+ void SetPrototypeGetter(
82
+ v8::Isolate* isolate,
83
+ v8::Local<v8::External> data,
84
+ v8::Local<v8::FunctionTemplate> recv,
85
+ const char* name,
86
+ v8::AccessorNameGetterCallback func
87
+ ) {
88
+ v8::HandleScope scope(isolate);
89
+ recv->InstanceTemplate()->SetNativeDataProperty(
90
+ InternalizedFromLatin1(isolate, name),
91
+ func,
92
+ 0,
93
+ data
94
+ );
95
+ }
96
+
97
+ #if defined(V8_ENABLE_SANDBOX)
98
+ // When V8 Sandbox is enabled (in newer Electron versions), we need to use Buffer::Copy
99
+ // instead of Buffer::New to ensure the ArrayBuffer backing store is allocated inside the sandbox
100
+ static inline v8::MaybeLocal<v8::Object> BufferSandboxNew(v8::Isolate* isolate, char* data, size_t length, void (*finalizeCallback)(char*, void*), void* finalizeHint) {
101
+ v8::MaybeLocal<v8::Object> buffer = node::Buffer::Copy(isolate, data, length);
102
+ finalizeCallback(data, finalizeHint);
103
+ return buffer;
104
+ }
105
+ #define SAFE_NEW_BUFFER(env, data, length, finalizeCallback, finalizeHint) BufferSandboxNew(env, data, length, finalizeCallback, finalizeHint)
106
+ #else
107
+ // When V8 Sandbox is not enabled, we can use the more efficient Buffer::New
108
+ #define SAFE_NEW_BUFFER(env, data, length, finalizeCallback, finalizeHint) node::Buffer::New(env, data, length, finalizeCallback, finalizeHint)
109
+ #endif
@@ -0,0 +1,63 @@
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
+ #define EasyIsolate v8::Isolate* isolate = v8::Isolate::GetCurrent()
8
+ #define OnlyIsolate info.GetIsolate()
9
+ #define OnlyContext isolate->GetCurrentContext()
10
+ #define OnlyAddon static_cast<Addon*>(info.Data().As<v8::External>()->Value())
11
+ #define UseIsolate v8::Isolate* isolate = OnlyIsolate
12
+ #define UseContext v8::Local<v8::Context> ctx = OnlyContext
13
+ #define UseAddon Addon* addon = OnlyAddon
14
+ #define Unwrap node::ObjectWrap::Unwrap
15
+
16
+ #define REQUIRE_ARGUMENT_ANY(at, var) \
17
+ if (info.Length() <= (at())) \
18
+ return ThrowTypeError("Expected a "#at" argument"); \
19
+ var = info[at()]
20
+
21
+ #define _REQUIRE_ARGUMENT(at, var, Type, message, ...) \
22
+ if (info.Length() <= (at()) || !info[at()]->Is##Type()) \
23
+ return ThrowTypeError("Expected "#at" argument to be "#message); \
24
+ var = (info[at()].As<v8::Type>())__VA_ARGS__
25
+
26
+ #define REQUIRE_ARGUMENT_INT32(at, var) \
27
+ _REQUIRE_ARGUMENT(at, var, Int32, a 32-bit signed integer, ->Value())
28
+ #define REQUIRE_ARGUMENT_BOOLEAN(at, var) \
29
+ _REQUIRE_ARGUMENT(at, var, Boolean, a boolean, ->Value())
30
+ #define REQUIRE_ARGUMENT_STRING(at, var) \
31
+ _REQUIRE_ARGUMENT(at, var, String, a string)
32
+ #define REQUIRE_ARGUMENT_OBJECT(at, var) \
33
+ _REQUIRE_ARGUMENT(at, var, Object, an object)
34
+ #define REQUIRE_ARGUMENT_FUNCTION(at, var) \
35
+ _REQUIRE_ARGUMENT(at, var, Function, a function)
36
+
37
+ #define REQUIRE_DATABASE_OPEN(db) \
38
+ if (!db->open) \
39
+ return ThrowTypeError("The database connection is not open")
40
+ #define REQUIRE_DATABASE_NOT_BUSY(db) \
41
+ if (db->busy) \
42
+ return ThrowTypeError("This database connection is busy executing a query")
43
+ #define REQUIRE_DATABASE_NO_ITERATORS(db) \
44
+ if (db->iterators) \
45
+ return ThrowTypeError("This database connection is busy executing a query")
46
+ #define REQUIRE_DATABASE_NO_ITERATORS_UNLESS_UNSAFE(db) \
47
+ if (!db->unsafe_mode) { \
48
+ REQUIRE_DATABASE_NO_ITERATORS(db); \
49
+ } ((void)0)
50
+ #define REQUIRE_STATEMENT_NOT_LOCKED(stmt) \
51
+ if (stmt->locked) \
52
+ return ThrowTypeError("This statement is busy executing a query")
53
+
54
+ #define first() 0
55
+ #define second() 1
56
+ #define third() 2
57
+ #define fourth() 3
58
+ #define fifth() 4
59
+ #define sixth() 5
60
+ #define seventh() 6
61
+ #define eighth() 7
62
+ #define ninth() 8
63
+ #define tenth() 9
@@ -0,0 +1,71 @@
1
+ #define STATEMENT_BIND(handle) \
2
+ Binder binder(handle); \
3
+ if (!binder.Bind(info, info.Length(), stmt)) { \
4
+ sqlite3_clear_bindings(handle); \
5
+ return; \
6
+ } ((void)0)
7
+
8
+ #define STATEMENT_THROW_LOGIC() \
9
+ db->ThrowDatabaseError(); \
10
+ if (!bound) { sqlite3_clear_bindings(handle); } \
11
+ return
12
+
13
+ #define STATEMENT_RETURN_LOGIC(return_value) \
14
+ info.GetReturnValue().Set(return_value); \
15
+ if (!bound) { sqlite3_clear_bindings(handle); } \
16
+ return
17
+
18
+ #define STATEMENT_START_LOGIC(RETURNS_DATA_CHECK, MUTATE_CHECK) \
19
+ Statement* stmt = Unwrap<Statement>(info.This()); \
20
+ RETURNS_DATA_CHECK(); \
21
+ sqlite3_stmt* handle = stmt->handle; \
22
+ Database* db = stmt->db; \
23
+ REQUIRE_DATABASE_OPEN(db->GetState()); \
24
+ REQUIRE_DATABASE_NOT_BUSY(db->GetState()); \
25
+ MUTATE_CHECK(); \
26
+ const bool bound = stmt->bound; \
27
+ if (!bound) { \
28
+ STATEMENT_BIND(handle); \
29
+ } else if (info.Length() > 0) { \
30
+ return ThrowTypeError("This statement already has bound parameters"); \
31
+ } ((void)0)
32
+
33
+
34
+ #define STATEMENT_THROW() db->GetState()->busy = false; STATEMENT_THROW_LOGIC()
35
+ #define STATEMENT_RETURN(x) db->GetState()->busy = false; STATEMENT_RETURN_LOGIC(x)
36
+ #define STATEMENT_START(x, y) \
37
+ STATEMENT_START_LOGIC(x, y); \
38
+ db->GetState()->busy = true; \
39
+ UseIsolate; \
40
+ if (db->Log(isolate, handle)) { \
41
+ STATEMENT_THROW(); \
42
+ } ((void)0)
43
+
44
+
45
+ #define DOES_NOT_MUTATE() REQUIRE_STATEMENT_NOT_LOCKED(stmt)
46
+ #define DOES_MUTATE() \
47
+ REQUIRE_STATEMENT_NOT_LOCKED(stmt); \
48
+ REQUIRE_DATABASE_NO_ITERATORS_UNLESS_UNSAFE(db->GetState())
49
+ #define DOES_ADD_ITERATOR() \
50
+ DOES_NOT_MUTATE(); \
51
+ if (db->GetState()->iterators == USHRT_MAX) \
52
+ return ThrowRangeError("Too many active database iterators")
53
+ #define REQUIRE_STATEMENT_RETURNS_DATA() \
54
+ if (!stmt->returns_data) \
55
+ return ThrowTypeError("This statement does not return data. Use run() instead")
56
+ #define ALLOW_ANY_STATEMENT() \
57
+ ((void)0)
58
+
59
+
60
+ #define _FUNCTION_START(type) \
61
+ type* self = static_cast<type*>(sqlite3_user_data(invocation)); \
62
+ v8::Isolate* isolate = self->isolate; \
63
+ v8::HandleScope scope(isolate)
64
+
65
+ #define FUNCTION_START() \
66
+ _FUNCTION_START(CustomFunction)
67
+
68
+ #define AGGREGATE_START() \
69
+ _FUNCTION_START(CustomAggregate); \
70
+ Accumulator* acc = self->GetAccumulator(invocation); \
71
+ if (acc->value.IsEmpty()) return
@@ -0,0 +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)->GetPrototype(),
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
+ };