2022-08-10 14:26:08 +03:00
|
|
|
/*
|
|
|
|
2022-05-22
|
|
|
|
|
|
|
|
The author disclaims copyright to this source code. In place of a
|
|
|
|
legal notice, here is a blessing:
|
|
|
|
|
|
|
|
* May you do good and not evil.
|
|
|
|
* May you find forgiveness for yourself and forgive others.
|
|
|
|
* May you share freely, never taking more than you give.
|
|
|
|
|
|
|
|
***********************************************************************
|
|
|
|
|
|
|
|
This file is intended to be combined at build-time with other
|
|
|
|
related code, most notably a header and footer which wraps this whole
|
|
|
|
file into an Emscripten Module.postRun() handler which has a parameter
|
|
|
|
named "Module" (the Emscripten Module object). The exact requirements,
|
|
|
|
conventions, and build process are very much under construction and
|
|
|
|
will be (re)documented once they've stopped fluctuating so much.
|
|
|
|
|
|
|
|
Specific goals of this project:
|
|
|
|
|
|
|
|
- Except where noted in the non-goals, provide a more-or-less
|
|
|
|
feature-complete wrapper to the sqlite3 C API, insofar as WASM
|
|
|
|
feature parity with C allows for. In fact, provide at least 3
|
|
|
|
APIs...
|
|
|
|
|
|
|
|
1) Bind a low-level sqlite3 API which is as close to the native
|
|
|
|
one as feasible in terms of usage.
|
|
|
|
|
|
|
|
2) A higher-level API, more akin to sql.js and node.js-style
|
|
|
|
implementations. This one speaks directly to the low-level
|
|
|
|
API. This API must be used from the same thread as the
|
|
|
|
low-level API.
|
|
|
|
|
|
|
|
3) A second higher-level API which speaks to the previous APIs via
|
|
|
|
worker messages. This one is intended for use in the main
|
|
|
|
thread, with the lower-level APIs installed in a Worker thread,
|
|
|
|
and talking to them via Worker messages. Because Workers are
|
|
|
|
asynchronouns and have only a single message channel, some
|
|
|
|
acrobatics are needed here to feed async work results back to
|
|
|
|
the client (as we cannot simply pass around callbacks between
|
|
|
|
the main and Worker threads).
|
|
|
|
|
|
|
|
- Insofar as possible, support client-side storage using JS
|
|
|
|
filesystem APIs. As of this writing, such things are still very
|
2022-09-13 22:27:03 +03:00
|
|
|
much under development.
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
Specific non-goals of this project:
|
|
|
|
|
|
|
|
- As WASM is a web-centric technology and UTF-8 is the King of
|
|
|
|
Encodings in that realm, there are no currently plans to support
|
|
|
|
the UTF16-related sqlite3 APIs. They would add a complication to
|
|
|
|
the bindings for no appreciable benefit. Though web-related
|
2022-09-13 22:27:03 +03:00
|
|
|
implementation details take priority, and the JavaScript
|
|
|
|
components of the API specifically focus on browser clients, the
|
|
|
|
lower-level WASM module "should" work in non-web WASM
|
|
|
|
environments.
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
- Supporting old or niche-market platforms. WASM is built for a
|
|
|
|
modern web and requires modern platforms.
|
|
|
|
|
|
|
|
- Though scalar User-Defined Functions (UDFs) may be created in
|
|
|
|
JavaScript, there are currently no plans to add support for
|
|
|
|
aggregate and window functions.
|
|
|
|
|
|
|
|
Attribution:
|
|
|
|
|
|
|
|
This project is endebted to the work of sql.js:
|
|
|
|
|
|
|
|
https://github.com/sql-js/sql.js
|
|
|
|
|
|
|
|
sql.js was an essential stepping stone in this code's development as
|
|
|
|
it demonstrated how to handle some of the WASM-related voodoo (like
|
|
|
|
handling pointers-to-pointers and adding JS implementations of
|
|
|
|
C-bound callback functions). These APIs have a considerably
|
|
|
|
different shape than sql.js's, however.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/**
|
2022-09-13 22:27:03 +03:00
|
|
|
sqlite3ApiBootstrap() is the only global symbol persistently
|
|
|
|
exposed by this API. It is intended to be called one time at the
|
|
|
|
end of the API amalgamation process, passed configuration details
|
|
|
|
for the current environment, and then optionally be removed from
|
|
|
|
the global object using `delete self.sqlite3ApiBootstrap`.
|
2022-08-10 14:26:08 +03:00
|
|
|
|
2022-08-22 16:34:13 +03:00
|
|
|
This function expects a configuration object, intended to abstract
|
2022-08-10 14:26:08 +03:00
|
|
|
away details specific to any given WASM environment, primarily so
|
2022-08-22 16:34:13 +03:00
|
|
|
that it can be used without any _direct_ dependency on
|
2022-09-13 22:27:03 +03:00
|
|
|
Emscripten. (Note the default values for the config object!) The
|
|
|
|
config object is only honored the first time this is
|
|
|
|
called. Subsequent calls ignore the argument and return the same
|
2022-08-22 16:34:13 +03:00
|
|
|
(configured) object which gets initialized by the first call.
|
|
|
|
|
|
|
|
The config object properties include:
|
|
|
|
|
2022-08-24 08:59:23 +03:00
|
|
|
- `exports`[^1]: the "exports" object for the current WASM
|
2022-08-22 16:34:13 +03:00
|
|
|
environment. In an Emscripten build, this should be set to
|
|
|
|
`Module['asm']`.
|
|
|
|
|
2022-08-24 08:59:23 +03:00
|
|
|
- `memory`[^1]: optional WebAssembly.Memory object, defaulting to
|
|
|
|
`exports.memory`. In Emscripten environments this should be set
|
2022-08-22 16:34:13 +03:00
|
|
|
to `Module.wasmMemory` if the build uses `-sIMPORT_MEMORY`, or be
|
|
|
|
left undefined/falsy to default to `exports.memory` when using
|
|
|
|
WASM-exported memory.
|
|
|
|
|
|
|
|
- `bigIntEnabled`: true if BigInt support is enabled. Defaults to
|
|
|
|
true if self.BigInt64Array is available, else false. Some APIs
|
|
|
|
will throw exceptions if called without BigInt support, as BigInt
|
|
|
|
is required for marshalling C-side int64 into and out of JS.
|
|
|
|
|
|
|
|
- `allocExportName`: the name of the function, in `exports`, of the
|
|
|
|
`malloc(3)`-compatible routine for the WASM environment. Defaults
|
|
|
|
to `"malloc"`.
|
|
|
|
|
|
|
|
- `deallocExportName`: the name of the function, in `exports`, of
|
|
|
|
the `free(3)`-compatible routine for the WASM
|
|
|
|
environment. Defaults to `"free"`.
|
|
|
|
|
2022-09-27 12:17:37 +03:00
|
|
|
- `wasmfsOpfsDir`[^1]: if the environment supports persistent storage, this
|
2022-08-22 16:34:13 +03:00
|
|
|
directory names the "mount point" for that directory. It must be prefixed
|
|
|
|
by `/` and may currently contain only a single directory-name part. Using
|
|
|
|
the root directory name is not supported by any current persistent backend.
|
2022-08-24 08:59:23 +03:00
|
|
|
|
|
|
|
|
|
|
|
[^1] = This property may optionally be a function, in which case this
|
|
|
|
function re-assigns it to the value returned from that function,
|
|
|
|
enabling delayed evaluation.
|
|
|
|
|
2022-08-10 14:26:08 +03:00
|
|
|
*/
|
2022-08-24 08:59:23 +03:00
|
|
|
'use strict';
|
|
|
|
self.sqlite3ApiBootstrap = function sqlite3ApiBootstrap(
|
2022-09-13 22:27:03 +03:00
|
|
|
apiConfig = (self.sqlite3ApiConfig || sqlite3ApiBootstrap.defaultConfig)
|
2022-08-24 08:59:23 +03:00
|
|
|
){
|
2022-08-22 16:34:13 +03:00
|
|
|
if(sqlite3ApiBootstrap.sqlite3){ /* already initalized */
|
|
|
|
console.warn("sqlite3ApiBootstrap() called multiple times.",
|
|
|
|
"Config and external initializers are ignored on calls after the first.");
|
|
|
|
return sqlite3ApiBootstrap.sqlite3;
|
|
|
|
}
|
2022-10-02 23:08:53 +03:00
|
|
|
const config = Object.assign(Object.create(null),{
|
|
|
|
exports: undefined,
|
|
|
|
memory: undefined,
|
|
|
|
bigIntEnabled: (()=>{
|
|
|
|
if('undefined'!==typeof Module){
|
|
|
|
/* Emscripten module will contain HEAPU64 when built with
|
|
|
|
-sWASM_BIGINT=1, else it will not. */
|
|
|
|
return !!Module.HEAPU64;
|
2022-08-23 00:37:17 +03:00
|
|
|
}
|
2022-10-02 23:08:53 +03:00
|
|
|
return !!self.BigInt64Array;
|
|
|
|
})(),
|
|
|
|
allocExportName: 'malloc',
|
|
|
|
deallocExportName: 'free',
|
|
|
|
wasmfsOpfsDir: '/opfs'
|
|
|
|
}, apiConfig || {});
|
2022-08-10 14:26:08 +03:00
|
|
|
|
2022-08-24 08:59:23 +03:00
|
|
|
[
|
|
|
|
// If any of these config options are functions, replace them with
|
|
|
|
// the result of calling that function...
|
2022-09-29 19:54:23 +03:00
|
|
|
'exports', 'memory', 'wasmfsOpfsDir'
|
2022-08-24 08:59:23 +03:00
|
|
|
].forEach((k)=>{
|
|
|
|
if('function' === typeof config[k]){
|
|
|
|
config[k] = config[k]();
|
|
|
|
}
|
|
|
|
});
|
2022-08-24 17:50:10 +03:00
|
|
|
|
2022-08-10 14:26:08 +03:00
|
|
|
/** Throws a new Error, the message of which is the concatenation
|
|
|
|
all args with a space between each. */
|
|
|
|
const toss = (...args)=>{throw new Error(args.join(' '))};
|
|
|
|
|
2022-09-27 12:17:37 +03:00
|
|
|
if(config.wasmfsOpfsDir && !/^\/[^/]+$/.test(config.wasmfsOpfsDir)){
|
|
|
|
toss("config.wasmfsOpfsDir must be falsy or in the form '/dir-name'.");
|
2022-08-22 16:34:13 +03:00
|
|
|
}
|
|
|
|
|
2022-08-10 14:26:08 +03:00
|
|
|
/**
|
|
|
|
Returns true if n is a 32-bit (signed) integer, else
|
|
|
|
false. This is used for determining when we need to switch to
|
|
|
|
double-type DB operations for integer values in order to keep
|
|
|
|
more precision.
|
|
|
|
*/
|
|
|
|
const isInt32 = function(n){
|
|
|
|
return ('bigint'!==typeof n /*TypeError: can't convert BigInt to number*/)
|
|
|
|
&& !!(n===(n|0) && n<=2147483647 && n>=-2147483648);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Returns v if v appears to be a TypedArray, else false. */
|
|
|
|
const isTypedArray = (v)=>{
|
|
|
|
return (v && v.constructor && isInt32(v.constructor.BYTES_PER_ELEMENT)) ? v : false;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
Returns true if v appears to be one of our bind()-able
|
|
|
|
TypedArray types: Uint8Array or Int8Array. Support for
|
|
|
|
TypedArrays with element sizes >1 is TODO.
|
|
|
|
*/
|
|
|
|
const isBindableTypedArray = (v)=>{
|
|
|
|
return v && v.constructor && (1===v.constructor.BYTES_PER_ELEMENT);
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
Returns true if v appears to be one of the TypedArray types
|
|
|
|
which is legal for holding SQL code (as opposed to binary blobs).
|
|
|
|
|
|
|
|
Currently this is the same as isBindableTypedArray() but it
|
|
|
|
seems likely that we'll eventually want to add Uint32Array
|
|
|
|
and friends to the isBindableTypedArray() list but not to the
|
|
|
|
isSQLableTypedArray() list.
|
|
|
|
*/
|
|
|
|
const isSQLableTypedArray = (v)=>{
|
|
|
|
return v && v.constructor && (1===v.constructor.BYTES_PER_ELEMENT);
|
|
|
|
};
|
|
|
|
|
|
|
|
/** Returns true if isBindableTypedArray(v) does, else throws with a message
|
|
|
|
that v is not a supported TypedArray value. */
|
|
|
|
const affirmBindableTypedArray = (v)=>{
|
|
|
|
return isBindableTypedArray(v)
|
|
|
|
|| toss("Value is not of a supported TypedArray type.");
|
|
|
|
};
|
|
|
|
|
|
|
|
const utf8Decoder = new TextDecoder('utf-8');
|
2022-08-22 16:34:13 +03:00
|
|
|
|
|
|
|
/** Internal helper to use in operations which need to distinguish
|
|
|
|
between SharedArrayBuffer heap memory and non-shared heap. */
|
|
|
|
const __SAB = ('undefined'===typeof SharedArrayBuffer)
|
|
|
|
? function(){} : SharedArrayBuffer;
|
|
|
|
const typedArrayToString = function(arrayBuffer, begin, end){
|
|
|
|
return utf8Decoder.decode(
|
|
|
|
(arrayBuffer.buffer instanceof __SAB)
|
|
|
|
? arrayBuffer.slice(begin, end)
|
|
|
|
: arrayBuffer.subarray(begin, end)
|
|
|
|
);
|
|
|
|
};
|
2022-08-10 14:26:08 +03:00
|
|
|
|
2022-09-30 23:35:37 +03:00
|
|
|
/**
|
|
|
|
If v is-a Array, its join('') result is returned. If
|
|
|
|
isSQLableTypedArray(v) then typedArrayToString(v) is
|
|
|
|
returned. Else v is returned as-is.
|
|
|
|
*/
|
|
|
|
const arrayToString = function(v){
|
|
|
|
if(isSQLableTypedArray(v)) return typedArrayToString(v);
|
|
|
|
else if(Array.isArray(v)) return v.join('');
|
|
|
|
return v;
|
|
|
|
};
|
|
|
|
|
2022-08-10 14:26:08 +03:00
|
|
|
/**
|
|
|
|
An Error subclass specifically for reporting Wasm-level malloc()
|
|
|
|
failure and enabling clients to unambiguously identify such
|
|
|
|
exceptions.
|
|
|
|
*/
|
|
|
|
class WasmAllocError extends Error {
|
|
|
|
constructor(...args){
|
|
|
|
super(...args);
|
|
|
|
this.name = 'WasmAllocError';
|
|
|
|
}
|
|
|
|
};
|
2022-10-02 06:11:13 +03:00
|
|
|
WasmAllocError.toss = (...args)=>{
|
|
|
|
throw new WasmAllocError(args.join(' '));
|
|
|
|
};
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
The main sqlite3 binding API gets installed into this object,
|
|
|
|
mimicking the C API as closely as we can. The numerous members
|
|
|
|
names with prefixes 'sqlite3_' and 'SQLITE_' behave, insofar as
|
|
|
|
possible, identically to the C-native counterparts, as documented at:
|
|
|
|
|
|
|
|
https://www.sqlite.org/c3ref/intro.html
|
|
|
|
|
|
|
|
A very few exceptions require an additional level of proxy
|
|
|
|
function or may otherwise require special attention in the WASM
|
|
|
|
environment, and all such cases are document here. Those not
|
|
|
|
documented here are installed as 1-to-1 proxies for their C-side
|
|
|
|
counterparts.
|
|
|
|
*/
|
|
|
|
const capi = {
|
|
|
|
/**
|
2022-10-02 21:47:39 +03:00
|
|
|
sqlite3_create_function_v2() differs from its native
|
|
|
|
counterpart only in the following ways:
|
|
|
|
|
|
|
|
1) The fourth argument (`eTextRep`) argument must not specify
|
|
|
|
any encoding other than sqlite.SQLITE_UTF8. The JS API does not
|
|
|
|
currently support any other encoding and likely never
|
|
|
|
will. This function does not replace that argument on its own
|
|
|
|
because it may contain other flags.
|
|
|
|
|
|
|
|
2) Any of the four final arguments may be either WASM pointers
|
|
|
|
(assumed to be function pointers) or JS Functions. In the
|
|
|
|
latter case, each gets bound to WASM using
|
|
|
|
sqlite3.capi.wasm.installFunction() and that wrapper is passed
|
|
|
|
on to the native implementation.
|
|
|
|
|
|
|
|
The semantics of JS functions are:
|
|
|
|
|
2022-10-02 23:08:53 +03:00
|
|
|
xFunc: is passed `(arrayOfValues)`. Its return value becomes
|
2022-10-02 21:47:39 +03:00
|
|
|
the new SQL function's result.
|
|
|
|
|
2022-10-02 23:08:53 +03:00
|
|
|
xStep: is passed `(arrayOfValues)`. Its return value is
|
2022-10-02 21:47:39 +03:00
|
|
|
ignored.
|
|
|
|
|
|
|
|
xFinal: is passed no arguments. Its return value becomes the
|
|
|
|
new aggragate SQL function's result.
|
|
|
|
|
|
|
|
xDestroy: is passed `(void*)`. Its return value is ignored. The
|
|
|
|
pointer passed to it is the one from the 5th argument to
|
|
|
|
sqlite3_create_function_v2().
|
|
|
|
|
|
|
|
Note that JS callback implementations have different call
|
|
|
|
signatures than their native counterparts (namely, they do not
|
|
|
|
get passed an `sqlite3_context*` argument) because practice has
|
|
|
|
shown that this is almost always more convenient and desirable
|
|
|
|
in JS code. Clients which need access to the full range of
|
|
|
|
native arguments will have to create a WASM-bound function
|
|
|
|
themselves (using wasm.installFunction() or equivalent) and
|
|
|
|
pass that function's WASM pointer to this function, rather than
|
|
|
|
passing a JS function. Be warned, however, that working with
|
|
|
|
UDFs at that level from JS is quite tedious.
|
|
|
|
|
|
|
|
For xFunc(), xStep(), and xFinal():
|
|
|
|
|
|
|
|
- When called from SQL, arguments to the UDF, and its result,
|
|
|
|
will be converted between JS and SQL with as much fidelity as
|
|
|
|
is feasible, triggering an exception if a type conversion
|
|
|
|
cannot be determined. Some freedom is afforded to numeric
|
|
|
|
conversions due to friction between the JS and C worlds:
|
|
|
|
integers which are larger than 32 bits will be treated as
|
|
|
|
doubles. TODO: use BigInt support if enabled. That feature
|
|
|
|
was added after this functionality was implemented.
|
|
|
|
|
|
|
|
If any JS-side functions throw, those exceptions are
|
|
|
|
intercepted and converted to database-side errors with
|
|
|
|
the exception of xFinal(): any exception from it is
|
|
|
|
ignored, possibly generating a console.error() message.
|
|
|
|
Destructors must not throw.
|
|
|
|
|
|
|
|
Once installed, there is currently no way to uninstall the
|
|
|
|
bound methods from WASM. They can be uninstalled from the
|
|
|
|
database as documented in the C API, but this wrapper currently
|
|
|
|
has no infrastructure in place to also free the WASM-bound JS
|
|
|
|
wrappers, effectively resulting in a memory leak if the client
|
|
|
|
uninstalls the UDF. Improving that is a potential TODO, but
|
|
|
|
removing client-installed UDFs is rare in practice.
|
|
|
|
|
|
|
|
Maintenance reminder: the ability to add new
|
|
|
|
WASM-accessible functions to the runtime requires that the
|
|
|
|
WASM build is compiled with emcc's `-sALLOW_TABLE_GROWTH`
|
|
|
|
flag.
|
2022-08-10 14:26:08 +03:00
|
|
|
*/
|
2022-10-02 21:47:39 +03:00
|
|
|
sqlite3_create_function_v2: function(
|
|
|
|
pDb, funcName, nArg, eTextRep, pApp,
|
|
|
|
xFunc, //function(arrayOfValues)
|
|
|
|
xStep, //function(arrayOfValues)
|
|
|
|
xFinal, //function()
|
|
|
|
xDestroy //function(void*)
|
|
|
|
){/*installed later*/},
|
|
|
|
/**
|
|
|
|
Equivalent to passing the same arguments to
|
|
|
|
sqlite3_create_function_v2(), with 0 as the final argument.
|
|
|
|
*/
|
|
|
|
sqlite3_create_function:function(
|
|
|
|
pDb, funcName, nArg, eTextRep, pApp,
|
|
|
|
xFunc, //function(arrayOfValues)
|
|
|
|
xStep, //function(arrayOfValues)
|
|
|
|
xFinal //function()
|
|
|
|
){/*installed later*/},
|
2022-10-02 23:08:53 +03:00
|
|
|
/**
|
|
|
|
The sqlite3_create_window_function() JS wrapper differs from
|
|
|
|
its native implementation in the exact same way that
|
|
|
|
sqlite3_create_function_v2() does. The additional function,
|
|
|
|
xInverse(), is treated identically to xStep() by the wrapping
|
|
|
|
layer.
|
|
|
|
*/
|
|
|
|
sqlite3_create_window_function: function(
|
|
|
|
pDb, funcName, nArg, eTextRep, pApp,
|
|
|
|
xStep, //function(arrayOfValues)
|
|
|
|
xFinal, //function()
|
|
|
|
xValue, //function()
|
|
|
|
xInverse,//function(arrayOfValues)
|
|
|
|
xDestroy //function(void*)
|
|
|
|
){/*installed later*/},
|
2022-08-10 14:26:08 +03:00
|
|
|
/**
|
|
|
|
The sqlite3_prepare_v3() binding handles two different uses
|
|
|
|
with differing JS/WASM semantics:
|
|
|
|
|
2022-09-28 10:53:47 +03:00
|
|
|
1) sqlite3_prepare_v3(pDb, sqlString, -1, prepFlags, ppStmt , null)
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
2) sqlite3_prepare_v3(pDb, sqlPointer, sqlByteLen, prepFlags, ppStmt, sqlPointerToPointer)
|
|
|
|
|
|
|
|
Note that the SQL length argument (the 3rd argument) must, for
|
|
|
|
usage (1), always be negative because it must be a byte length
|
|
|
|
and that value is expensive to calculate from JS (where only
|
|
|
|
the character length of strings is readily available). It is
|
|
|
|
retained in this API's interface for code/documentation
|
|
|
|
compatibility reasons but is currently _always_ ignored. With
|
|
|
|
usage (2), the 3rd argument is used as-is but is is still
|
|
|
|
critical that the C-style input string (2nd argument) be
|
|
|
|
terminated with a 0 byte.
|
|
|
|
|
|
|
|
In usage (1), the 2nd argument must be of type string,
|
|
|
|
Uint8Array, or Int8Array (either of which is assumed to
|
|
|
|
hold SQL). If it is, this function assumes case (1) and
|
|
|
|
calls the underyling C function with the equivalent of:
|
|
|
|
|
|
|
|
(pDb, sqlAsString, -1, prepFlags, ppStmt, null)
|
|
|
|
|
2022-09-28 10:53:47 +03:00
|
|
|
The `pzTail` argument is ignored in this case because its
|
|
|
|
result is meaningless when a string-type value is passed
|
|
|
|
through: the string goes through another level of internal
|
2022-08-10 14:26:08 +03:00
|
|
|
conversion for WASM's sake and the result pointer would refer
|
|
|
|
to that transient conversion's memory, not the passed-in
|
2022-09-28 10:53:47 +03:00
|
|
|
string.
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
If the sql argument is not a string, it must be a _pointer_ to
|
|
|
|
a NUL-terminated string which was allocated in the WASM memory
|
2022-09-28 10:53:47 +03:00
|
|
|
(e.g. using capi.wasm.alloc() or equivalent). In that case,
|
2022-08-10 14:26:08 +03:00
|
|
|
the final argument may be 0/null/undefined or must be a pointer
|
|
|
|
to which the "tail" of the compiled SQL is written, as
|
|
|
|
documented for the C-side sqlite3_prepare_v3(). In case (2),
|
|
|
|
the underlying C function is called with the equivalent of:
|
|
|
|
|
2022-09-28 10:53:47 +03:00
|
|
|
(pDb, sqlAsPointer, sqlByteLen, prepFlags, ppStmt, pzTail)
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
It returns its result and compiled statement as documented in
|
|
|
|
the C API. Fetching the output pointers (5th and 6th
|
2022-09-28 10:53:47 +03:00
|
|
|
parameters) requires using `capi.wasm.getMemValue()` (or
|
|
|
|
equivalent) and the `pzTail` will point to an address relative to
|
|
|
|
the `sqlAsPointer` value.
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
If passed an invalid 2nd argument type, this function will
|
2022-09-28 10:53:47 +03:00
|
|
|
return SQLITE_MISUSE and sqlite3_errmsg() will contain a string
|
|
|
|
describing the problem.
|
2022-08-10 14:26:08 +03:00
|
|
|
|
2022-09-28 10:53:47 +03:00
|
|
|
Side-note: if given an empty string, or one which contains only
|
|
|
|
comments or an empty SQL expression, 0 is returned but the result
|
|
|
|
output pointer will be NULL.
|
2022-08-10 14:26:08 +03:00
|
|
|
*/
|
2022-09-30 23:35:37 +03:00
|
|
|
sqlite3_prepare_v3: (dbPtr, sql, sqlByteLen, prepFlags,
|
|
|
|
stmtPtrPtr, strPtrPtr)=>{}/*installed later*/,
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
Equivalent to calling sqlite3_prapare_v3() with 0 as its 4th argument.
|
|
|
|
*/
|
2022-09-30 23:35:37 +03:00
|
|
|
sqlite3_prepare_v2: (dbPtr, sql, sqlByteLen,
|
|
|
|
stmtPtrPtr,strPtrPtr)=>{}/*installed later*/,
|
2022-08-10 14:26:08 +03:00
|
|
|
|
2022-09-30 23:35:37 +03:00
|
|
|
/**
|
|
|
|
This binding enables the callback argument to be a JavaScript.
|
|
|
|
|
|
|
|
If the callback is a function, then for the duration of the
|
|
|
|
sqlite3_exec() call, it installs a WASM-bound function which
|
|
|
|
acts as a proxy for the given callback. That proxy will
|
|
|
|
also perform a conversion of the callback's arguments from
|
|
|
|
`(char**)` to JS arrays of strings. However, for API
|
|
|
|
consistency's sake it will still honor the C-level
|
|
|
|
callback parameter order and will call it like:
|
|
|
|
|
|
|
|
`callback(pVoid, colCount, listOfValues, listOfColNames)`
|
|
|
|
|
|
|
|
If the callback is not a JS function then this binding performs
|
|
|
|
no translation of the callback, but the sql argument is still
|
|
|
|
converted to a WASM string for the call using the
|
|
|
|
"flexible-string" argument converter.
|
|
|
|
*/
|
|
|
|
sqlite3_exec: (pDb, sql, callback, pVoid, pErrMsg)=>{}/*installed later*/,
|
2022-08-10 14:26:08 +03:00
|
|
|
/**
|
|
|
|
Various internal-use utilities are added here as needed. They
|
|
|
|
are bound to an object only so that we have access to them in
|
|
|
|
the differently-scoped steps of the API bootstrapping
|
|
|
|
process. At the end of the API setup process, this object gets
|
|
|
|
removed.
|
|
|
|
*/
|
|
|
|
util:{
|
2022-09-30 23:35:37 +03:00
|
|
|
affirmBindableTypedArray, arrayToString, isBindableTypedArray,
|
|
|
|
isInt32, isSQLableTypedArray, isTypedArray,
|
|
|
|
typedArrayToString,
|
2022-09-30 13:55:28 +03:00
|
|
|
isMainWindow: ()=>{
|
|
|
|
return self.window===self && self.document;
|
|
|
|
}
|
2022-08-10 14:26:08 +03:00
|
|
|
},
|
|
|
|
|
|
|
|
/**
|
|
|
|
Holds state which are specific to the WASM-related
|
|
|
|
infrastructure and glue code. It is not expected that client
|
|
|
|
code will normally need these, but they're exposed here in case
|
|
|
|
it does. These APIs are _not_ to be considered an
|
|
|
|
official/stable part of the sqlite3 WASM API. They may change
|
|
|
|
as the developers' experience suggests appropriate changes.
|
|
|
|
|
|
|
|
Note that a number of members of this object are injected
|
|
|
|
dynamically after the api object is fully constructed, so
|
|
|
|
not all are documented inline here.
|
|
|
|
*/
|
|
|
|
wasm: {
|
|
|
|
//^^^ TODO?: move wasm from sqlite3.capi.wasm to sqlite3.wasm
|
|
|
|
/**
|
|
|
|
Emscripten APIs have a deep-seated assumption that all pointers
|
|
|
|
are 32 bits. We'll remain optimistic that that won't always be
|
|
|
|
the case and will use this constant in places where we might
|
|
|
|
otherwise use a hard-coded 4.
|
|
|
|
*/
|
|
|
|
ptrSizeof: config.wasmPtrSizeof || 4,
|
|
|
|
/**
|
|
|
|
The WASM IR (Intermediate Representation) value for
|
|
|
|
pointer-type values. It MUST refer to a value type of the
|
|
|
|
size described by this.ptrSizeof _or_ it may be any value
|
|
|
|
which ends in '*', which Emscripten's glue code internally
|
|
|
|
translates to i32.
|
|
|
|
*/
|
|
|
|
ptrIR: config.wasmPtrIR || "i32",
|
|
|
|
/**
|
|
|
|
True if BigInt support was enabled via (e.g.) the
|
|
|
|
Emscripten -sWASM_BIGINT flag, else false. When
|
|
|
|
enabled, certain 64-bit sqlite3 APIs are enabled which
|
|
|
|
are not otherwise enabled due to JS/WASM int64
|
|
|
|
impedence mismatches.
|
|
|
|
*/
|
|
|
|
bigIntEnabled: !!config.bigIntEnabled,
|
|
|
|
/**
|
|
|
|
The symbols exported by the WASM environment.
|
|
|
|
*/
|
|
|
|
exports: config.exports
|
|
|
|
|| toss("Missing API config.exports (WASM module exports)."),
|
|
|
|
|
|
|
|
/**
|
|
|
|
When Emscripten compiles with `-sIMPORT_MEMORY`, it
|
|
|
|
initalizes the heap and imports it into wasm, as opposed to
|
|
|
|
the other way around. In this case, the memory is not
|
|
|
|
available via this.exports.memory.
|
|
|
|
*/
|
|
|
|
memory: config.memory || config.exports['memory']
|
|
|
|
|| toss("API config object requires a WebAssembly.Memory object",
|
|
|
|
"in either config.exports.memory (exported)",
|
|
|
|
"or config.memory (imported)."),
|
2022-08-17 19:44:05 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
The API's one single point of access to the WASM-side memory
|
|
|
|
allocator. Works like malloc(3) (and is likely bound to
|
|
|
|
malloc()) but throws an WasmAllocError if allocation fails. It is
|
|
|
|
important that any code which might pass through the sqlite3 C
|
|
|
|
API NOT throw and must instead return SQLITE_NOMEM (or
|
|
|
|
equivalent, depending on the context).
|
|
|
|
|
|
|
|
That said, very few cases in the API can result in
|
|
|
|
client-defined functions propagating exceptions via the C-style
|
|
|
|
API. Most notably, this applies ot User-defined SQL Functions
|
|
|
|
(UDFs) registered via sqlite3_create_function_v2(). For that
|
|
|
|
specific case it is recommended that all UDF creation be
|
|
|
|
funneled through a utility function and that a wrapper function
|
|
|
|
be added around the UDF which catches any exception and sets
|
|
|
|
the error state to OOM. (The overall complexity of registering
|
|
|
|
UDFs essentially requires a helper for doing so!)
|
|
|
|
*/
|
|
|
|
alloc: undefined/*installed later*/,
|
|
|
|
/**
|
|
|
|
The API's one single point of access to the WASM-side memory
|
|
|
|
deallocator. Works like free(3) (and is likely bound to
|
|
|
|
free()).
|
|
|
|
*/
|
|
|
|
dealloc: undefined/*installed later*/
|
|
|
|
|
2022-08-10 14:26:08 +03:00
|
|
|
/* Many more wasm-related APIs get installed later on. */
|
|
|
|
}/*wasm*/
|
|
|
|
}/*capi*/;
|
|
|
|
|
|
|
|
/**
|
|
|
|
capi.wasm.alloc()'s srcTypedArray.byteLength bytes,
|
|
|
|
populates them with the values from the source
|
|
|
|
TypedArray, and returns the pointer to that memory. The
|
|
|
|
returned pointer must eventually be passed to
|
|
|
|
capi.wasm.dealloc() to clean it up.
|
|
|
|
|
|
|
|
As a special case, to avoid further special cases where
|
|
|
|
this is used, if srcTypedArray.byteLength is 0, it
|
|
|
|
allocates a single byte and sets it to the value
|
|
|
|
0. Even in such cases, calls must behave as if the
|
|
|
|
allocated memory has exactly srcTypedArray.byteLength
|
|
|
|
bytes.
|
|
|
|
|
|
|
|
ACHTUNG: this currently only works for Uint8Array and
|
|
|
|
Int8Array types and will throw if srcTypedArray is of
|
|
|
|
any other type.
|
|
|
|
*/
|
2022-08-17 19:44:05 +03:00
|
|
|
capi.wasm.allocFromTypedArray = function(srcTypedArray){
|
2022-08-10 14:26:08 +03:00
|
|
|
affirmBindableTypedArray(srcTypedArray);
|
2022-10-02 23:08:53 +03:00
|
|
|
const pRet = capi.wasm.alloc(srcTypedArray.byteLength || 1);
|
|
|
|
capi.wasm.heapForSize(srcTypedArray.constructor).set(srcTypedArray.byteLength ? srcTypedArray : [0], pRet);
|
2022-08-10 14:26:08 +03:00
|
|
|
return pRet;
|
2022-10-02 23:08:53 +03:00
|
|
|
};
|
2022-08-10 14:26:08 +03:00
|
|
|
|
|
|
|
const keyAlloc = config.allocExportName || 'malloc',
|
|
|
|
keyDealloc = config.deallocExportName || 'free';
|
|
|
|
for(const key of [keyAlloc, keyDealloc]){
|
|
|
|
const f = capi.wasm.exports[key];
|
|
|
|
if(!(f instanceof Function)) toss("Missing required exports[",key,"] function.");
|
|
|
|
}
|
2022-08-17 19:44:05 +03:00
|
|
|
|
2022-08-10 14:26:08 +03:00
|
|
|
capi.wasm.alloc = function(n){
|
2022-10-02 23:08:53 +03:00
|
|
|
const m = capi.wasm.exports[keyAlloc](n);
|
2022-08-10 14:26:08 +03:00
|
|
|
if(!m) throw new WasmAllocError("Failed to allocate "+n+" bytes.");
|
|
|
|
return m;
|
2022-10-02 23:08:53 +03:00
|
|
|
};
|
2022-08-17 19:44:05 +03:00
|
|
|
|
2022-08-10 14:26:08 +03:00
|
|
|
capi.wasm.dealloc = (m)=>capi.wasm.exports[keyDealloc](m);
|
|
|
|
|
|
|
|
/**
|
|
|
|
Reports info about compile-time options using
|
|
|
|
sqlite_compileoption_get() and sqlite3_compileoption_used(). It
|
|
|
|
has several distinct uses:
|
|
|
|
|
|
|
|
If optName is an array then it is expected to be a list of
|
|
|
|
compilation options and this function returns an object
|
|
|
|
which maps each such option to true or false, indicating
|
|
|
|
whether or not the given option was included in this
|
|
|
|
build. That object is returned.
|
|
|
|
|
|
|
|
If optName is an object, its keys are expected to be compilation
|
|
|
|
options and this function sets each entry to true or false,
|
|
|
|
indicating whether the compilation option was used or not. That
|
|
|
|
object is returned.
|
|
|
|
|
|
|
|
If passed no arguments then it returns an object mapping
|
|
|
|
all known compilation options to their compile-time values,
|
|
|
|
or boolean true if they are defined with no value. This
|
|
|
|
result, which is relatively expensive to compute, is cached
|
|
|
|
and returned for future no-argument calls.
|
|
|
|
|
|
|
|
In all other cases it returns true if the given option was
|
|
|
|
active when when compiling the sqlite3 module, else false.
|
|
|
|
|
|
|
|
Compile-time option names may optionally include their
|
|
|
|
"SQLITE_" prefix. When it returns an object of all options,
|
|
|
|
the prefix is elided.
|
|
|
|
*/
|
|
|
|
capi.wasm.compileOptionUsed = function f(optName){
|
|
|
|
if(!arguments.length){
|
|
|
|
if(f._result) return f._result;
|
|
|
|
else if(!f._opt){
|
|
|
|
f._rx = /^([^=]+)=(.+)/;
|
|
|
|
f._rxInt = /^-?\d+$/;
|
|
|
|
f._opt = function(opt, rv){
|
|
|
|
const m = f._rx.exec(opt);
|
|
|
|
rv[0] = (m ? m[1] : opt);
|
|
|
|
rv[1] = m ? (f._rxInt.test(m[2]) ? +m[2] : m[2]) : true;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
const rc = {}, ov = [0,0];
|
|
|
|
let i = 0, k;
|
|
|
|
while((k = capi.sqlite3_compileoption_get(i++))){
|
|
|
|
f._opt(k,ov);
|
|
|
|
rc[ov[0]] = ov[1];
|
|
|
|
}
|
|
|
|
return f._result = rc;
|
|
|
|
}else if(Array.isArray(optName)){
|
|
|
|
const rc = {};
|
|
|
|
optName.forEach((v)=>{
|
|
|
|
rc[v] = capi.sqlite3_compileoption_used(v);
|
|
|
|
});
|
|
|
|
return rc;
|
|
|
|
}else if('object' === typeof optName){
|
|
|
|
Object.keys(optName).forEach((k)=> {
|
|
|
|
optName[k] = capi.sqlite3_compileoption_used(k);
|
|
|
|
});
|
|
|
|
return optName;
|
|
|
|
}
|
|
|
|
return (
|
|
|
|
'string'===typeof optName
|
|
|
|
) ? !!capi.sqlite3_compileoption_used(optName) : false;
|
|
|
|
}/*compileOptionUsed()*/;
|
|
|
|
|
2022-09-13 22:27:03 +03:00
|
|
|
/**
|
|
|
|
Signatures for the WASM-exported C-side functions. Each entry
|
|
|
|
is an array with 2+ elements:
|
|
|
|
|
|
|
|
[ "c-side name",
|
|
|
|
"result type" (capi.wasm.xWrap() syntax),
|
|
|
|
[arg types in xWrap() syntax]
|
|
|
|
// ^^^ this needn't strictly be an array: it can be subsequent
|
|
|
|
// elements instead: [x,y,z] is equivalent to x,y,z
|
|
|
|
]
|
|
|
|
|
|
|
|
Note that support for the API-specific data types in the
|
|
|
|
result/argument type strings gets plugged in at a later phase in
|
|
|
|
the API initialization process.
|
|
|
|
*/
|
2022-08-10 14:26:08 +03:00
|
|
|
capi.wasm.bindingSignatures = [
|
|
|
|
// Please keep these sorted by function name!
|
2022-10-01 15:04:49 +03:00
|
|
|
["sqlite3_bind_blob","int", "sqlite3_stmt*", "int", "*", "int", "*"
|
|
|
|
/* We should arguably write a custom wrapper which knows how
|
|
|
|
to handle Blob, TypedArrays, and JS strings. */
|
|
|
|
],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_bind_double","int", "sqlite3_stmt*", "int", "f64"],
|
|
|
|
["sqlite3_bind_int","int", "sqlite3_stmt*", "int", "int"],
|
|
|
|
["sqlite3_bind_null",undefined, "sqlite3_stmt*", "int"],
|
|
|
|
["sqlite3_bind_parameter_count", "int", "sqlite3_stmt*"],
|
|
|
|
["sqlite3_bind_parameter_index","int", "sqlite3_stmt*", "string"],
|
2022-10-01 15:04:49 +03:00
|
|
|
["sqlite3_bind_text","int", "sqlite3_stmt*", "int", "string", "int", "int"
|
|
|
|
/* We should arguably create a hand-written binding
|
|
|
|
which does more flexible text conversion, along the lines of
|
|
|
|
sqlite3_prepare_v3(). The slightly problematic part is the
|
|
|
|
final argument (text destructor). */
|
|
|
|
],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_close_v2", "int", "sqlite3*"],
|
|
|
|
["sqlite3_changes", "int", "sqlite3*"],
|
|
|
|
["sqlite3_clear_bindings","int", "sqlite3_stmt*"],
|
|
|
|
["sqlite3_column_blob","*", "sqlite3_stmt*", "int"],
|
|
|
|
["sqlite3_column_bytes","int", "sqlite3_stmt*", "int"],
|
|
|
|
["sqlite3_column_count", "int", "sqlite3_stmt*"],
|
|
|
|
["sqlite3_column_double","f64", "sqlite3_stmt*", "int"],
|
|
|
|
["sqlite3_column_int","int", "sqlite3_stmt*", "int"],
|
|
|
|
["sqlite3_column_name","string", "sqlite3_stmt*", "int"],
|
|
|
|
["sqlite3_column_text","string", "sqlite3_stmt*", "int"],
|
|
|
|
["sqlite3_column_type","int", "sqlite3_stmt*", "int"],
|
|
|
|
["sqlite3_compileoption_get", "string", "int"],
|
|
|
|
["sqlite3_compileoption_used", "int", "string"],
|
2022-10-02 21:47:39 +03:00
|
|
|
/* sqlite3_create_function_v2() is handled separate to simplify conversion
|
|
|
|
of its callback argument */
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_data_count", "int", "sqlite3_stmt*"],
|
|
|
|
["sqlite3_db_filename", "string", "sqlite3*", "string"],
|
2022-09-13 22:27:03 +03:00
|
|
|
["sqlite3_db_handle", "sqlite3*", "sqlite3_stmt*"],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_db_name", "string", "sqlite3*", "int"],
|
2022-09-30 13:55:28 +03:00
|
|
|
["sqlite3_deserialize", "int", "sqlite3*", "string", "*", "i64", "i64", "int"]
|
|
|
|
/* Careful! Short version: de/serialize() are problematic because they
|
|
|
|
might use a different allocator that the user for managing the
|
|
|
|
deserialized block. de/serialize() are ONLY safe to use with
|
|
|
|
sqlite3_malloc(), sqlite3_free(), and its 64-bit variants. */,
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_errmsg", "string", "sqlite3*"],
|
|
|
|
["sqlite3_error_offset", "int", "sqlite3*"],
|
|
|
|
["sqlite3_errstr", "string", "int"],
|
2022-09-30 23:35:37 +03:00
|
|
|
/*["sqlite3_exec", "int", "sqlite3*", "string", "*", "*", "**"
|
|
|
|
Handled seperately to perform translation of the callback
|
|
|
|
into a WASM-usable one. ],*/
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_expanded_sql", "string", "sqlite3_stmt*"],
|
|
|
|
["sqlite3_extended_errcode", "int", "sqlite3*"],
|
|
|
|
["sqlite3_extended_result_codes", "int", "sqlite3*", "int"],
|
2022-09-13 22:27:03 +03:00
|
|
|
["sqlite3_file_control", "int", "sqlite3*", "string", "int", "*"],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_finalize", "int", "sqlite3_stmt*"],
|
2022-09-30 13:55:28 +03:00
|
|
|
["sqlite3_free", undefined,"*"],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_initialize", undefined],
|
|
|
|
["sqlite3_interrupt", undefined, "sqlite3*"
|
|
|
|
/* ^^^ we cannot actually currently support this because JS is
|
|
|
|
single-threaded and we don't have a portable way to access a DB
|
|
|
|
from 2 SharedWorkers concurrently. */],
|
|
|
|
["sqlite3_libversion", "string"],
|
|
|
|
["sqlite3_libversion_number", "int"],
|
2022-09-30 13:55:28 +03:00
|
|
|
["sqlite3_malloc", "*","int"],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_open", "int", "string", "*"],
|
|
|
|
["sqlite3_open_v2", "int", "string", "*", "int", "string"],
|
|
|
|
/* sqlite3_prepare_v2() and sqlite3_prepare_v3() are handled
|
|
|
|
separately due to us requiring two different sets of semantics
|
|
|
|
for those, depending on how their SQL argument is provided. */
|
2022-09-30 13:55:28 +03:00
|
|
|
["sqlite3_realloc", "*","*","int"],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_reset", "int", "sqlite3_stmt*"],
|
|
|
|
["sqlite3_result_blob",undefined, "*", "*", "int", "*"],
|
|
|
|
["sqlite3_result_double",undefined, "*", "f64"],
|
|
|
|
["sqlite3_result_error",undefined, "*", "string", "int"],
|
|
|
|
["sqlite3_result_error_code", undefined, "*", "int"],
|
|
|
|
["sqlite3_result_error_nomem", undefined, "*"],
|
|
|
|
["sqlite3_result_error_toobig", undefined, "*"],
|
|
|
|
["sqlite3_result_int",undefined, "*", "int"],
|
|
|
|
["sqlite3_result_null",undefined, "*"],
|
|
|
|
["sqlite3_result_text",undefined, "*", "string", "int", "*"],
|
2022-09-30 13:55:28 +03:00
|
|
|
["sqlite3_serialize","*", "sqlite3*", "string", "*", "int"],
|
2022-09-24 13:15:08 +03:00
|
|
|
["sqlite3_shutdown", undefined],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_sourceid", "string"],
|
|
|
|
["sqlite3_sql", "string", "sqlite3_stmt*"],
|
|
|
|
["sqlite3_step", "int", "sqlite3_stmt*"],
|
|
|
|
["sqlite3_strglob", "int", "string","string"],
|
|
|
|
["sqlite3_strlike", "int", "string","string","int"],
|
|
|
|
["sqlite3_total_changes", "int", "sqlite3*"],
|
2022-09-20 17:52:26 +03:00
|
|
|
["sqlite3_uri_boolean", "int", "string", "string", "int"],
|
|
|
|
["sqlite3_uri_key", "string", "string", "int"],
|
|
|
|
["sqlite3_uri_parameter", "string", "string", "string"],
|
2022-08-10 14:26:08 +03:00
|
|
|
["sqlite3_value_blob", "*", "*"],
|
|
|
|
["sqlite3_value_bytes","int", "*"],
|
|
|
|
["sqlite3_value_double","f64", "*"],
|
|
|
|
["sqlite3_value_text", "string", "*"],
|
|
|
|
["sqlite3_value_type", "int", "*"],
|
|
|
|
["sqlite3_vfs_find", "*", "string"],
|
2022-09-11 19:59:40 +03:00
|
|
|
["sqlite3_vfs_register", "int", "*", "int"]
|
2022-08-10 14:26:08 +03:00
|
|
|
]/*capi.wasm.bindingSignatures*/;
|
|
|
|
|
|
|
|
if(false && capi.wasm.compileOptionUsed('SQLITE_ENABLE_NORMALIZE')){
|
|
|
|
/* ^^^ "the problem" is that this is an option feature and the
|
|
|
|
build-time function-export list does not currently take
|
|
|
|
optional features into account. */
|
|
|
|
capi.wasm.bindingSignatures.push(["sqlite3_normalized_sql", "string", "sqlite3_stmt*"]);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
Functions which require BigInt (int64) support are separated from
|
|
|
|
the others because we need to conditionally bind them or apply
|
|
|
|
dummy impls, depending on the capabilities of the environment.
|
|
|
|
*/
|
|
|
|
capi.wasm.bindingSignatures.int64 = [
|
2022-09-20 17:52:26 +03:00
|
|
|
["sqlite3_bind_int64","int", ["sqlite3_stmt*", "int", "i64"]],
|
|
|
|
["sqlite3_changes64","i64", ["sqlite3*"]],
|
|
|
|
["sqlite3_column_int64","i64", ["sqlite3_stmt*", "int"]],
|
2022-09-30 13:55:28 +03:00
|
|
|
["sqlite3_malloc64", "*","i64"],
|
2022-09-30 15:10:36 +03:00
|
|
|
["sqlite3_msize", "i64", "*"],
|
2022-09-30 13:55:28 +03:00
|
|
|
["sqlite3_realloc64", "*","*", "i64"],
|
2022-09-20 17:52:26 +03:00
|
|
|
["sqlite3_total_changes64", "i64", ["sqlite3*"]],
|
|
|
|
["sqlite3_uri_int64", "i64", ["string", "string", "i64"]]
|
2022-08-10 14:26:08 +03:00
|
|
|
];
|
|
|
|
|
2022-09-11 19:59:40 +03:00
|
|
|
/**
|
|
|
|
Functions which are intended solely for API-internal use by the
|
|
|
|
WASM components, not client code. These get installed into
|
|
|
|
capi.wasm.
|
2022-10-02 06:11:13 +03:00
|
|
|
|
|
|
|
TODO: get rid of sqlite3_wasm_vfs_unlink(). It is ill-conceived
|
|
|
|
and only rarely actually useful.
|
2022-09-11 19:59:40 +03:00
|
|
|
*/
|
|
|
|
capi.wasm.bindingSignatures.wasm = [
|
|
|
|
["sqlite3_wasm_vfs_unlink", "int", "string"]
|
|
|
|
];
|
|
|
|
|
2022-10-01 21:47:42 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
sqlite3.capi.wasm.pstack (pseudo-stack) holds a special-case
|
|
|
|
stack-style allocator intended only for use with _small_ data of
|
|
|
|
not more than (in total) a few kb in size, managed as if it were
|
|
|
|
stack-based.
|
|
|
|
|
|
|
|
It has only a single intended usage:
|
|
|
|
|
|
|
|
```
|
|
|
|
const stackPos = pstack.pointer;
|
|
|
|
try{
|
|
|
|
const ptr = pstack.alloc(8);
|
|
|
|
// ==> pstack.pointer === ptr
|
|
|
|
const otherPtr = pstack.alloc(8);
|
|
|
|
// ==> pstack.pointer === otherPtr
|
|
|
|
...
|
|
|
|
}finally{
|
|
|
|
pstack.restore(stackPos);
|
|
|
|
// ==> pstack.pointer === stackPos
|
|
|
|
}
|
|
|
|
```
|
|
|
|
|
|
|
|
This allocator is much faster than a general-purpose one but is
|
|
|
|
limited to usage patterns like the one shown above.
|
|
|
|
|
|
|
|
It operates from a static range of memory which lives outside of
|
|
|
|
space managed by Emscripten's stack-management, so does not
|
|
|
|
collide with Emscripten-provided stack allocation APIs. The
|
|
|
|
memory lives in the WASM heap and can be used with routines such
|
|
|
|
as wasm.setMemValue() and any wasm.heap8u().slice().
|
|
|
|
*/
|
|
|
|
capi.wasm.pstack = Object.assign(Object.create(null),{
|
|
|
|
/**
|
|
|
|
Sets the current ppstack position to the given pointer.
|
|
|
|
Results are undefined if the passed-in value did not come from
|
|
|
|
this.pointer.
|
|
|
|
*/
|
|
|
|
restore: capi.wasm.exports.sqlite3_wasm_pstack_restore,
|
|
|
|
/**
|
|
|
|
Attempts to allocate the given number of bytes from the
|
|
|
|
pstack. On success, it zeroes out a block of memory of the
|
|
|
|
given size, adjusts the pstack pointer, and returns a pointer
|
2022-10-02 06:11:13 +03:00
|
|
|
to the memory. On error, returns throws a WasmAllocError. The
|
|
|
|
memory must eventually be released using restore().
|
2022-10-01 21:47:42 +03:00
|
|
|
|
|
|
|
This method always adjusts the given value to be a multiple
|
2022-10-02 03:09:40 +03:00
|
|
|
of 8 bytes because failing to do so can lead to incorrect
|
|
|
|
results when reading and writing 64-bit values from/to the WASM
|
|
|
|
heap.
|
2022-10-01 21:47:42 +03:00
|
|
|
*/
|
2022-10-02 06:11:13 +03:00
|
|
|
alloc: (n)=>{
|
|
|
|
return capi.wasm.exports.sqlite3_wasm_pstack_alloc(n)
|
|
|
|
|| WasmAllocError.toss("Could not allocate",n,
|
|
|
|
"bytes from the pstack.");
|
|
|
|
}
|
|
|
|
// More methods get added after the capi.wasm object is populated
|
|
|
|
// by WhWasmUtilInstaller.
|
2022-10-01 21:47:42 +03:00
|
|
|
});
|
|
|
|
/**
|
|
|
|
sqlite3.capi.wasm.pstack.pointer resolves to the current pstack
|
|
|
|
position pointer. This value is intended _only_ to be passed to restore().
|
|
|
|
*/
|
|
|
|
Object.defineProperty(capi.wasm.pstack, 'pointer', {
|
|
|
|
configurable: false, iterable: true, writeable: false,
|
|
|
|
get: capi.wasm.exports.sqlite3_wasm_pstack_ptr
|
|
|
|
//Whether or not a setter as an alternative to restore() is
|
|
|
|
//clearer or would just lead to confusion is unclear.
|
|
|
|
//set: capi.wasm.exports.sqlite3_wasm_pstack_restore
|
|
|
|
});
|
|
|
|
/**
|
|
|
|
sqlite3.capi.wasm.pstack.remaining resolves to the amount of
|
|
|
|
space remaining in the pstack.
|
|
|
|
*/
|
|
|
|
Object.defineProperty(capi.wasm.pstack, 'remaining', {
|
|
|
|
configurable: false, iterable: true, writeable: false,
|
|
|
|
get: capi.wasm.exports.sqlite3_wasm_pstack_remaining
|
|
|
|
});
|
|
|
|
|
2022-10-02 03:09:40 +03:00
|
|
|
/**
|
|
|
|
An Error subclass specifically for reporting DB-level errors and
|
|
|
|
enabling clients to unambiguously identify such exceptions.
|
|
|
|
The C-level APIs never throw, but some of the higher-level
|
|
|
|
C-style APIs do and the object-oriented APIs use exceptions
|
|
|
|
exclusively to report errors.
|
|
|
|
*/
|
|
|
|
class SQLite3Error extends Error {
|
|
|
|
/**
|
|
|
|
Constructs this object with a message equal to all arguments
|
|
|
|
concatenated with a space between each one.
|
|
|
|
*/
|
|
|
|
constructor(...args){
|
|
|
|
super(args.join(' '));
|
|
|
|
this.name = 'SQLite3Error';
|
|
|
|
}
|
|
|
|
};
|
2022-10-02 06:11:13 +03:00
|
|
|
SQLite3Error.toss = (...args)=>{
|
|
|
|
throw new SQLite3Error(args.join(' '));
|
|
|
|
};
|
2022-10-01 21:47:42 +03:00
|
|
|
|
2022-09-27 16:40:12 +03:00
|
|
|
/** State for sqlite3_wasmfs_opfs_dir(). */
|
|
|
|
let __persistentDir = undefined;
|
2022-08-13 16:56:00 +03:00
|
|
|
/**
|
2022-10-01 21:47:42 +03:00
|
|
|
If the wasm environment has a WASMFS/OPFS-backed persistent
|
|
|
|
storage directory, its path is returned by this function. If it
|
|
|
|
does not then it returns "" (noting that "" is a falsy value).
|
2022-08-13 16:56:00 +03:00
|
|
|
|
2022-08-17 19:44:05 +03:00
|
|
|
The first time this is called, this function inspects the current
|
2022-10-01 21:47:42 +03:00
|
|
|
environment to determine whether persistence support is available
|
|
|
|
and, if it is, enables it (if needed).
|
2022-08-13 16:56:00 +03:00
|
|
|
|
2022-09-18 20:32:35 +03:00
|
|
|
This function currently only recognizes the WASMFS/OPFS storage
|
2022-10-01 21:47:42 +03:00
|
|
|
combination and its path refers to storage rooted in the
|
|
|
|
Emscripten-managed virtual filesystem.
|
2022-08-13 16:56:00 +03:00
|
|
|
*/
|
2022-09-27 16:40:12 +03:00
|
|
|
capi.sqlite3_wasmfs_opfs_dir = function(){
|
2022-08-13 16:56:00 +03:00
|
|
|
if(undefined !== __persistentDir) return __persistentDir;
|
|
|
|
// If we have no OPFS, there is no persistent dir
|
2022-09-27 12:17:37 +03:00
|
|
|
const pdir = config.wasmfsOpfsDir;
|
2022-08-22 16:34:13 +03:00
|
|
|
if(!pdir
|
|
|
|
|| !self.FileSystemHandle
|
|
|
|
|| !self.FileSystemDirectoryHandle
|
2022-08-13 16:56:00 +03:00
|
|
|
|| !self.FileSystemFileHandle){
|
|
|
|
return __persistentDir = "";
|
|
|
|
}
|
|
|
|
try{
|
2022-09-12 19:09:50 +03:00
|
|
|
if(pdir && 0===capi.wasm.xCallWrapped(
|
2022-09-15 09:42:41 +03:00
|
|
|
'sqlite3_wasm_init_wasmfs', 'i32', ['string'], pdir
|
2022-08-22 16:34:13 +03:00
|
|
|
)){
|
|
|
|
return __persistentDir = pdir;
|
2022-08-13 16:56:00 +03:00
|
|
|
}else{
|
|
|
|
return __persistentDir = "";
|
|
|
|
}
|
|
|
|
}catch(e){
|
2022-09-15 09:42:41 +03:00
|
|
|
// sqlite3_wasm_init_wasmfs() is not available
|
2022-08-13 16:56:00 +03:00
|
|
|
return __persistentDir = "";
|
|
|
|
}
|
2022-09-12 19:09:50 +03:00
|
|
|
};
|
2022-08-13 16:56:00 +03:00
|
|
|
|
2022-08-17 19:44:05 +03:00
|
|
|
/**
|
2022-09-19 16:16:35 +03:00
|
|
|
Experimental and subject to change or removal.
|
|
|
|
|
2022-09-27 16:40:12 +03:00
|
|
|
Returns true if sqlite3.capi.sqlite3_wasmfs_opfs_dir() is a
|
2022-09-13 22:27:03 +03:00
|
|
|
non-empty string and the given name starts with (that string +
|
|
|
|
'/'), else returns false.
|
|
|
|
|
|
|
|
Potential (but arguable) TODO: return true if the name is one of
|
|
|
|
(":localStorage:", "local", ":sessionStorage:", "session") and
|
|
|
|
kvvfs is available.
|
2022-08-17 19:44:05 +03:00
|
|
|
*/
|
|
|
|
capi.sqlite3_web_filename_is_persistent = function(name){
|
2022-09-27 16:40:12 +03:00
|
|
|
const p = capi.sqlite3_wasmfs_opfs_dir();
|
2022-09-13 22:27:03 +03:00
|
|
|
return (p && name) ? name.startsWith(p+'/') : false;
|
2022-09-12 19:09:50 +03:00
|
|
|
};
|
2022-09-13 22:27:03 +03:00
|
|
|
|
2022-09-24 10:36:45 +03:00
|
|
|
// This bit is highly arguable and is incompatible with the fiddle shell.
|
|
|
|
if(false && 0===capi.wasm.exports.sqlite3_vfs_find(0)){
|
2022-09-12 19:09:50 +03:00
|
|
|
/* Assume that sqlite3_initialize() has not yet been called.
|
|
|
|
This will be the case in an SQLITE_OS_KV build. */
|
|
|
|
capi.wasm.exports.sqlite3_initialize();
|
|
|
|
}
|
2022-08-17 19:44:05 +03:00
|
|
|
|
2022-09-18 20:32:35 +03:00
|
|
|
/**
|
2022-09-26 16:55:10 +03:00
|
|
|
Given an `sqlite3*`, an sqlite3_vfs name, and an optional db
|
|
|
|
name, returns a truthy value (see below) if that db handle uses
|
|
|
|
that VFS, else returns false. If pDb is falsy then the 3rd
|
|
|
|
argument is ignored and this function returns a truthy value if
|
|
|
|
the default VFS name matches that of the 2nd argument. Results
|
|
|
|
are undefined if pDb is truthy but refers to an invalid
|
|
|
|
pointer. The 3rd argument specifies the database name of the
|
|
|
|
given database connection to check, defaulting to the main db.
|
|
|
|
|
|
|
|
The 2nd and 3rd arguments may either be a JS string or a C-string
|
2022-09-18 20:32:35 +03:00
|
|
|
allocated from the wasm environment.
|
|
|
|
|
|
|
|
The truthy value it returns is a pointer to the `sqlite3_vfs`
|
|
|
|
object.
|
|
|
|
|
|
|
|
To permit safe use of this function from APIs which may be called
|
|
|
|
via the C stack (like SQL UDFs), this function does not throw: if
|
|
|
|
bad arguments cause a conversion error when passing into
|
|
|
|
wasm-space, false is returned.
|
|
|
|
*/
|
2022-09-26 16:55:10 +03:00
|
|
|
capi.sqlite3_web_db_uses_vfs = function(pDb,vfsName,dbName="main"){
|
2022-09-18 20:32:35 +03:00
|
|
|
try{
|
2022-09-26 16:55:10 +03:00
|
|
|
const pK = capi.sqlite3_vfs_find(vfsName);
|
2022-09-18 20:32:35 +03:00
|
|
|
if(!pK) return false;
|
|
|
|
else if(!pDb){
|
|
|
|
return capi.sqlite3_vfs_find(0)===pK ? pK : false;
|
|
|
|
}
|
|
|
|
const ppVfs = capi.wasm.allocPtr();
|
|
|
|
try{
|
|
|
|
return (
|
|
|
|
(0===capi.sqlite3_file_control(
|
2022-09-26 16:55:10 +03:00
|
|
|
pDb, dbName, capi.SQLITE_FCNTL_VFS_POINTER, ppVfs
|
2022-09-18 20:32:35 +03:00
|
|
|
)) && (capi.wasm.getPtrValue(ppVfs) === pK)
|
|
|
|
) ? pK : false;
|
|
|
|
}finally{
|
|
|
|
capi.wasm.dealloc(ppVfs);
|
|
|
|
}
|
|
|
|
}catch(e){
|
|
|
|
/* Ignore - probably bad args to a wasm-bound function. */
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-09-19 16:44:23 +03:00
|
|
|
/**
|
|
|
|
Returns an array of the names of all currently-registered sqlite3
|
|
|
|
VFSes.
|
|
|
|
*/
|
|
|
|
capi.sqlite3_web_vfs_list = function(){
|
|
|
|
const rc = [];
|
|
|
|
let pVfs = capi.sqlite3_vfs_find(0);
|
|
|
|
while(pVfs){
|
|
|
|
const oVfs = new capi.sqlite3_vfs(pVfs);
|
|
|
|
rc.push(capi.wasm.cstringToJs(oVfs.$zName));
|
|
|
|
pVfs = oVfs.$pNext;
|
|
|
|
oVfs.dispose();
|
|
|
|
}
|
|
|
|
return rc;
|
|
|
|
};
|
|
|
|
|
2022-10-01 19:01:41 +03:00
|
|
|
/**
|
|
|
|
Serializes the given `sqlite3*` pointer to a Uint8Array, as per
|
|
|
|
sqlite3_serialize(). On success it returns a Uint8Array. On
|
|
|
|
error it throws with a description of the problem.
|
|
|
|
*/
|
|
|
|
capi.sqlite3_web_db_export = function(pDb){
|
|
|
|
if(!pDb) toss('Invalid sqlite3* argument.');
|
|
|
|
const wasm = capi.wasm;
|
|
|
|
if(!wasm.bigIntEnabled) toss('BigInt64 support is not enabled.');
|
2022-10-01 22:21:52 +03:00
|
|
|
const stack = wasm.pstack.pointer;
|
2022-10-01 19:01:41 +03:00
|
|
|
let pOut;
|
|
|
|
try{
|
2022-10-01 21:47:42 +03:00
|
|
|
const pSize = wasm.pstack.alloc(8/*i64*/ + wasm.ptrSizeof);
|
2022-10-01 19:01:41 +03:00
|
|
|
const ppOut = pSize + 8;
|
|
|
|
/**
|
|
|
|
Maintenance reminder, since this cost a full hour of grief
|
|
|
|
and confusion: if the order of pSize/ppOut are reversed in
|
|
|
|
that memory block, fetching the value of pSize after the
|
|
|
|
export reads a garbage size because it's not on an 8-byte
|
|
|
|
memory boundary!
|
|
|
|
*/
|
|
|
|
let rc = wasm.exports.sqlite3_wasm_db_serialize(
|
|
|
|
pDb, ppOut, pSize, 0
|
|
|
|
);
|
|
|
|
if(rc){
|
|
|
|
toss("Database serialization failed with code",
|
|
|
|
sqlite3.capi.sqlite3_web_rc_str(rc));
|
|
|
|
}
|
2022-10-01 21:47:42 +03:00
|
|
|
pOut = wasm.getPtrValue(ppOut);
|
2022-10-01 19:01:41 +03:00
|
|
|
const nOut = wasm.getMemValue(pSize, 'i64');
|
|
|
|
rc = nOut
|
|
|
|
? wasm.heap8u().slice(pOut, pOut + Number(nOut))
|
|
|
|
: new Uint8Array();
|
|
|
|
return rc;
|
|
|
|
}catch(e){
|
|
|
|
console.error('internal error?',e);
|
|
|
|
throw w;
|
|
|
|
}finally{
|
|
|
|
if(pOut) wasm.exports.sqlite3_free(pOut);
|
2022-10-01 21:47:42 +03:00
|
|
|
wasm.pstack.restore(stack);
|
2022-10-01 19:01:41 +03:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2022-09-30 13:55:28 +03:00
|
|
|
if( capi.util.isMainWindow() ){
|
2022-09-13 22:27:03 +03:00
|
|
|
/* Features specific to the main window thread... */
|
|
|
|
|
|
|
|
/**
|
|
|
|
Internal helper for sqlite3_web_kvvfs_clear() and friends.
|
|
|
|
Its argument should be one of ('local','session','').
|
|
|
|
*/
|
|
|
|
const __kvvfsInfo = function(which){
|
|
|
|
const rc = Object.create(null);
|
|
|
|
rc.prefix = 'kvvfs-'+which;
|
|
|
|
rc.stores = [];
|
|
|
|
if('session'===which || ''===which) rc.stores.push(self.sessionStorage);
|
|
|
|
if('local'===which || ''===which) rc.stores.push(self.localStorage);
|
|
|
|
return rc;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
Clears all storage used by the kvvfs DB backend, deleting any
|
|
|
|
DB(s) stored there. Its argument must be either 'session',
|
|
|
|
'local', or ''. In the first two cases, only sessionStorage
|
|
|
|
resp. localStorage is cleared. If it's an empty string (the
|
|
|
|
default) then both are cleared. Only storage keys which match
|
|
|
|
the pattern used by kvvfs are cleared: any other client-side
|
|
|
|
data are retained.
|
|
|
|
|
|
|
|
This function is only available in the main window thread.
|
|
|
|
|
|
|
|
Returns the number of entries cleared.
|
|
|
|
*/
|
|
|
|
capi.sqlite3_web_kvvfs_clear = function(which=''){
|
|
|
|
let rc = 0;
|
|
|
|
const kvinfo = __kvvfsInfo(which);
|
|
|
|
kvinfo.stores.forEach((s)=>{
|
|
|
|
const toRm = [] /* keys to remove */;
|
|
|
|
let i;
|
|
|
|
for( i = 0; i < s.length; ++i ){
|
|
|
|
const k = s.key(i);
|
|
|
|
if(k.startsWith(kvinfo.prefix)) toRm.push(k);
|
|
|
|
}
|
|
|
|
toRm.forEach((kk)=>s.removeItem(kk));
|
|
|
|
rc += toRm.length;
|
|
|
|
});
|
|
|
|
return rc;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
This routine guesses the approximate amount of
|
|
|
|
window.localStorage and/or window.sessionStorage in use by the
|
2022-09-18 20:32:35 +03:00
|
|
|
kvvfs database backend. Its argument must be one of
|
2022-09-13 22:27:03 +03:00
|
|
|
('session', 'local', ''). In the first two cases, only
|
|
|
|
sessionStorage resp. localStorage is counted. If it's an empty
|
|
|
|
string (the default) then both are counted. Only storage keys
|
|
|
|
which match the pattern used by kvvfs are counted. The returned
|
|
|
|
value is the "length" value of every matching key and value,
|
2022-09-16 15:49:45 +03:00
|
|
|
noting that JavaScript stores each character in 2 bytes.
|
2022-09-13 22:27:03 +03:00
|
|
|
|
|
|
|
Note that the returned size is not authoritative from the
|
|
|
|
perspective of how much data can fit into localStorage and
|
|
|
|
sessionStorage, as the precise algorithms for determining
|
|
|
|
those limits are unspecified and may include per-entry
|
|
|
|
overhead invisible to clients.
|
|
|
|
*/
|
|
|
|
capi.sqlite3_web_kvvfs_size = function(which=''){
|
|
|
|
let sz = 0;
|
|
|
|
const kvinfo = __kvvfsInfo(which);
|
|
|
|
kvinfo.stores.forEach((s)=>{
|
|
|
|
let i;
|
|
|
|
for(i = 0; i < s.length; ++i){
|
|
|
|
const k = s.key(i);
|
|
|
|
if(k.startsWith(kvinfo.prefix)){
|
|
|
|
sz += k.length;
|
|
|
|
sz += s.getItem(k).length;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2022-10-01 19:01:41 +03:00
|
|
|
return sz * 2 /* because JS uses 2-byte char encoding */;
|
2022-09-13 22:27:03 +03:00
|
|
|
};
|
|
|
|
|
|
|
|
}/* main-window-only bits */
|
|
|
|
|
2022-09-27 16:40:12 +03:00
|
|
|
|
2022-08-10 14:26:08 +03:00
|
|
|
/* The remainder of the API will be set up in later steps. */
|
2022-08-22 16:34:13 +03:00
|
|
|
const sqlite3 = {
|
2022-08-17 19:44:05 +03:00
|
|
|
WasmAllocError: WasmAllocError,
|
2022-10-02 03:09:40 +03:00
|
|
|
SQLite3Error: SQLite3Error,
|
2022-08-10 14:26:08 +03:00
|
|
|
capi,
|
2022-09-27 16:40:12 +03:00
|
|
|
config,
|
|
|
|
/**
|
|
|
|
Performs any optional asynchronous library-level initialization
|
|
|
|
which might be required. This function returns a Promise which
|
|
|
|
resolves to the sqlite3 namespace object. It _ignores any
|
|
|
|
errors_ in the asynchronous init process, as such components
|
|
|
|
are all optional. If called more than once, the second and
|
|
|
|
subsequent calls are no-ops which return a pre-resolved
|
|
|
|
Promise.
|
|
|
|
|
2022-09-30 19:49:03 +03:00
|
|
|
Ideally this function is called as part of the Promise chain
|
|
|
|
which handles the loading and bootstrapping of the API. If not
|
|
|
|
then it must be called by client-level code, which must not use
|
|
|
|
the library until the returned promise resolves.
|
2022-09-27 16:40:12 +03:00
|
|
|
|
|
|
|
Bug: if called while a prior call is still resolving, the 2nd
|
|
|
|
call will resolve prematurely, before the 1st call has finished
|
2022-09-30 19:49:03 +03:00
|
|
|
resolving. The current build setup precludes that possibility,
|
|
|
|
so it's only a hypothetical problem if/when this function
|
|
|
|
ever needs to be invoked by clients.
|
2022-09-29 16:17:50 +03:00
|
|
|
|
|
|
|
In Emscripten-based builds, this function is called
|
|
|
|
automatically and deleted from this object.
|
2022-09-27 16:40:12 +03:00
|
|
|
*/
|
|
|
|
asyncPostInit: async function(){
|
|
|
|
let lip = sqlite3ApiBootstrap.initializersAsync;
|
|
|
|
delete sqlite3ApiBootstrap.initializersAsync;
|
|
|
|
if(!lip || !lip.length) return Promise.resolve(sqlite3);
|
|
|
|
// Is it okay to resolve these in parallel or do we need them
|
|
|
|
// to resolve in order? We currently only have 1, so it
|
|
|
|
// makes no difference.
|
|
|
|
lip = lip.map((f)=>f(sqlite3).catch(()=>{}));
|
|
|
|
//let p = lip.shift();
|
|
|
|
//while(lip.length) p = p.then(lip.shift());
|
|
|
|
//return p.then(()=>sqlite3);
|
|
|
|
return Promise.all(lip).then(()=>sqlite3);
|
|
|
|
}
|
2022-08-10 14:26:08 +03:00
|
|
|
};
|
2022-08-22 16:34:13 +03:00
|
|
|
sqlite3ApiBootstrap.initializers.forEach((f)=>f(sqlite3));
|
|
|
|
delete sqlite3ApiBootstrap.initializers;
|
|
|
|
sqlite3ApiBootstrap.sqlite3 = sqlite3;
|
|
|
|
return sqlite3;
|
2022-08-10 14:26:08 +03:00
|
|
|
}/*sqlite3ApiBootstrap()*/;
|
2022-08-22 16:34:13 +03:00
|
|
|
/**
|
|
|
|
self.sqlite3ApiBootstrap.initializers is an internal detail used by
|
|
|
|
the various pieces of the sqlite3 API's amalgamation process. It
|
|
|
|
must not be modified by client code except when plugging such code
|
|
|
|
into the amalgamation process.
|
|
|
|
|
|
|
|
Each component of the amalgamation is expected to append a function
|
|
|
|
to this array. When sqlite3ApiBootstrap() is called for the first
|
|
|
|
time, each such function will be called (in their appended order)
|
|
|
|
and passed the sqlite3 namespace object, into which they can install
|
|
|
|
their features (noting that most will also require that certain
|
|
|
|
features alread have been installed). At the end of that process,
|
|
|
|
this array is deleted.
|
2022-09-27 16:40:12 +03:00
|
|
|
|
|
|
|
Note that the order of insertion into this array is significant for
|
|
|
|
some pieces. e.g. sqlite3.capi.wasm cannot be fully utilized until
|
|
|
|
the whwasmutil.js part is plugged in.
|
2022-08-22 16:34:13 +03:00
|
|
|
*/
|
|
|
|
self.sqlite3ApiBootstrap.initializers = [];
|
2022-09-27 16:40:12 +03:00
|
|
|
/**
|
|
|
|
self.sqlite3ApiBootstrap.initializersAsync is an internal detail
|
|
|
|
used by the sqlite3 API's amalgamation process. It must not be
|
|
|
|
modified by client code except when plugging such code into the
|
|
|
|
amalgamation process.
|
|
|
|
|
|
|
|
Counterpart of self.sqlite3ApiBootstrap.initializers, specifically
|
|
|
|
for initializers which are asynchronous. All functions in this list
|
|
|
|
take the sqlite3 object as their argument and MUST return a
|
|
|
|
Promise. Both the resolved value and rejection cases are ignored.
|
|
|
|
|
|
|
|
This list is not processed until the client calls
|
|
|
|
sqlite3.asyncPostInit(). This means, for example, that intializers
|
|
|
|
added to self.sqlite3ApiBootstrap.initializers may push entries to
|
|
|
|
this list.
|
|
|
|
*/
|
|
|
|
self.sqlite3ApiBootstrap.initializersAsync = [];
|
2022-08-24 08:59:23 +03:00
|
|
|
/**
|
|
|
|
Client code may assign sqlite3ApiBootstrap.defaultConfig an
|
|
|
|
object-type value before calling sqlite3ApiBootstrap() (without
|
|
|
|
arguments) in order to tell that call to use this object as its
|
|
|
|
default config value. The intention of this is to provide
|
|
|
|
downstream clients with a reasonably flexible approach for plugging in
|
|
|
|
an environment-suitable configuration without having to define a new
|
|
|
|
global-scope symbol.
|
|
|
|
*/
|
|
|
|
self.sqlite3ApiBootstrap.defaultConfig = Object.create(null);
|
2022-09-13 22:27:03 +03:00
|
|
|
/**
|
|
|
|
Placeholder: gets installed by the first call to
|
|
|
|
self.sqlite3ApiBootstrap(). However, it is recommended that the
|
|
|
|
caller of sqlite3ApiBootstrap() capture its return value and delete
|
|
|
|
self.sqlite3ApiBootstrap after calling it. It returns the same
|
|
|
|
value which will be stored here.
|
|
|
|
*/
|
2022-08-24 08:59:23 +03:00
|
|
|
self.sqlite3ApiBootstrap.sqlite3 = undefined;
|