Mailspring/vendor/dblite-custom.js
Ben Gotow 1e8fd46342 fix(drafts): Various improvements and fixes to drafts, draft state management
Summary:
This diff contains a few major changes:

1. Scribe is no longer used for the text editor. It's just a plain contenteditable region. The toolbar items (bold, italic, underline) still work. Scribe was causing React inconcistency issues in the following scenario:
   - View thread with draft, edit draft
   - Move to another thread
   - Move back to thread with draft
   - Move to another thread. Notice that one or more messages from thread with draft are still there.

There may be a way to fix this, but I tried for hours and there are Github Issues open on it's repository asking for React compatibility, so it may be fixed soon. For now contenteditable is working great.

2. Action.saveDraft() is no longer debounced in the DraftStore. Instead, firing that action causes the save to happen immediately, and the DraftStoreProxy has a new "DraftChangeSet" class which is responsbile for batching saves as the user interacts with the ComposerView. There are a couple big wins here:

   - In the future, we may want to be able to call Action.saveDraft() in other situations and it should behave like a normal action. We may also want to expose the DraftStoreProxy as an easy way of backing interactive draft UI.

   - Previously, when you added a contact to To/CC/BCC, this happened:

     <input> -> Action.saveDraft -> (delay!!) -> Database -> DraftStore -> DraftStoreProxy -> View Updates

Increasing the delay to something reasonable like 200msec meant there was 200msec of lag before you saw the new view state.

To fix this, I created a new class called DraftChangeSet which is responsible for accumulating changes as they're made and firing Action.saveDraft. "Adding" a change to the change set also causes the Draft provided by the DraftStoreProxy to change immediately (the changes are a temporary layer on top of the database object). This means no delay while changes are being applied. There's a better explanation in the source!

This diff includes a few minor fixes as well:

1. Draft.state is gone—use Message.object = draft instead
2. String model attributes should never be null
3. Pre-send checks that can cancel draft send
4. Put the entire curl history and task queue into feedback reports
5. Cache localIds for extra speed
6. Move us up to latest React

Test Plan: No new tests - once we lock down this new design I'll write tests for the DraftChangeSet

Reviewers: evan

Reviewed By: evan

Differential Revision: https://review.inboxapp.com/D1125
2015-02-03 16:24:31 -08:00

906 lines
No EOL
29 KiB
JavaScript

/*!
Copyright (C) 2013 by WebReflection
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
/*!
Copyright (C) 2013 by WebReflection
Permission is hereby granted, free of charge, to any person obtaining a copy
of this software and associated documentation files (the "Software"), to deal
in the Software without restriction, including without limitation the rights
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
copies of the Software, and to permit persons to whom the Software is
furnished to do so, subject to the following conditions:
The above copyright notice and this permission notice shall be included in
all copies or substantial portions of the Software.
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
THE SOFTWARE.
*/
/*! a zero hassle wrapper for sqlite by Andrea Giammarchi !*/
var
isArray = Array.isArray,
// used to generate unique "end of the query" identifiers
crypto = require('crypto'),
// relative, absolute, and db paths are normalized anyway
path = require('path'),
// each dblite(fileName) instance is an EventEmitter
EventEmitter = require('events').EventEmitter,
// used to perform some fallback
WIN32 = process.platform === 'win32',
// what kind of Path Separator we have here ?
PATH_SEP = path.sep || (
WIN32 ? '\\' : '/'
),
// each dblite instance spawns a process once
// and interact with that shell for the whole session
// one spawn per database and no more (on avg 1 db is it)
spawn = require('child_process').spawn,
// use to re-generate Date objects
DECIMAL = /^[1-9][0-9]*$/,
// verify if it's a select or not
SELECT = /^(?:select|SELECT|pragma|PRAGMA) /,
// for simple query replacements: WHERE field = ?
REPLACE_QUESTIONMARKS = /\?/g,
// for named replacements: WHERE field = :data
REPLACE_PARAMS = /(?:\:|\@|\$)([a-zA-Z_$]+)/g,
// the way CSV threats double quotes
DOUBLE_DOUBLE_QUOTES = /""/g,
// to escape strings
SINGLE_QUOTES = /'/g,
// to use same escaping logic for double quotes
// except it makes escaping easier for JSON data
// which usually is full of "
SINGLE_QUOTES_DOUBLED = "''",
// to verify there are named fields/parametes
HAS_PARAMS = /(?:\?|(?:(?:\:|\@|\$)[a-zA-Z_$]+))/,
// shortcut used as deafault notifier
log = console.log.bind(console),
// the default binary as array of paths
bin = ['sqlite3'],
// private shared variables
// avoid creation of N functions
// keeps memory low and improves performance
paramsIndex, // which index is the current
paramsArray, // which value when Array
paramsObject, // which value when Object (named parameters)
IS_NODE_06 = false, // dirty things to do there ...
// defned later on
EOL, EOL_LENGTH,
SANITIZER, SANITIZER_REPLACER,
defineCSVEOL = function () {
defineCSVEOL = function () {};
var sqliteVersion = dblite.sqliteVersion ||
process.env.SQLITE_VERSION ||
'';
sqliteVersion = String(dblite.sqliteVersion || '')
.replace(/[^.\d]/g, '')
.split('.')
;
// what kind of End Of Line we have here ?
EOL = sqliteVersion.length && sqliteVersion.filter(function (n, i) {
n = parseInt(n, 10);
switch (i) {
case 0:
return n >= 3;
case 1:
return n >= 8;
case 2:
return n >= 6;
}
return false;
}).length === sqliteVersion.length ?
'\r\n' :
require('os').EOL || (
WIN32 ? '\r\n' : '\n'
)
;
// what's EOL length? Used to properly parse data
EOL_LENGTH = EOL.length;
// makes EOL safe for strings passed to the shell
SANITIZER = new RegExp("[;" + EOL.split('').map(function(c) {
return '\\x' + ('0' + c.charCodeAt(0).toString(16)).slice(-2);
}).join('') + "]+$");
// used to mark the end of each line passed to the shell
SANITIZER_REPLACER = ';' + EOL;
if (!sqliteVersion.length) {
console.warn([
'[WARNING] sqlite 3.8.6 changed CSV output',
'please specify your sqlite version',
'via `dblite.sqliteVersion = "3.8.5";`',
'or via SQLITE_VERSION=3.8.5'
].join(EOL));
}
}
;
/**
* var db = dblite('filename.sqlite'):EventEmitter;
*
* db.query( thismethod has **many** overloads where almost everything is optional
*
* SQL:string, only necessary field. Accepts a query or a command such `.databases`
*
* params:Array|Object, optional, if specified replaces SQL parts with this object
* db.query('INSERT INTO table VALUES(?, ?)', [null, 'content']);
* db.query('INSERT INTO table VALUES(:id, :value)', {id:null, value:'content'});
*
* fields:Array|Object, optional, if specified is used to normalize the query result with named fields.
* db.query('SELECT table.a, table.other FROM table', ['a', 'b']);
* [{a:'first value', b:'second'},{a:'row2 value', b:'row2'}]
*
*
* db.query('SELECT table.a, table.other FROM table', ['a', 'b']);
* [{a:'first value', b:'second'},{a:'row2 value', b:'row2'}]
* callback:Function
* );
*/
function dblite() {
defineCSVEOL();
var
// this is the delimiter of each sqlite3 shell command
SUPER_SECRET = '---' +
crypto.randomBytes(64).toString('base64') +
'---',
// ... I wish .print was introduced before SQLite 3.7.10 ...
// this is a weird way to get rid of the header, if enabled
SUPER_SECRET_SELECT = '"' + SUPER_SECRET + '" AS "' + SUPER_SECRET + '";' + EOL,
// used to check the end of a buffer
SUPER_SECRET_LENGTH = -(SUPER_SECRET.length + EOL_LENGTH),
// the incrementally concatenated buffer
// cleaned up as soon as the current command has been completed
selectResult = '',
// the current dblite "instance"
self = new EventEmitter(),
// usually the database file or ':memory:' only
// the "spawned once" program, will be used for the whole session
program = spawn(
// executable only, folder needs to be specified a part
bin.length === 1 ? bin[0] : ('.' + PATH_SEP + bin[bin.length - 1]),
// normalize file path if not :memory:
normalizeFirstArgument(
// it is possible to eventually send extra sqlite3 args
// so all arguments are passed
Array.prototype.slice.call(arguments)
).concat('-csv') // but the output MUST be csv
.reverse(), // see https://github.com/WebReflection/dblite/pull/12
// be sure the dir is the right one
{
// the right folder is important or sqlite3 won't work
cwd: bin.slice(0, -1).join(PATH_SEP) || process.cwd(),
env: process.env, // same env is OK
encoding: 'utf8', // utf8 is OK
detached: true, // asynchronous
stdio: ['pipe', 'pipe', 'pipe'] // handled here
}
),
// sqlite3 shell can produce one output per time
// evey operation performed through this wrapper
// should not bother the program until next
// available slot. This queue helps keeping
// requests ordered without stressing the system
// once things will be ready, callbacks will be notified
// accordingly. As simple as that ^_^
queue = [],
// set as true only once db.close() has been called
notWorking = false,
// marks the shell busy or not
busy = false,
// tells if current output needs to be processed
wasSelect = false,
wasNotSelect = false,
wasError = false,
// forces the output not to be processed
// might be handy in some case where it's passed around
// as string instread of needing to serialize/unserialize
// the list of already arrays or objects
dontParseCSV = false,
// one callback per time will be notified
$callback,
// recycled variable for fields operation
$fields
;
SUPER_SECRET += EOL;
// when program is killed or closed for some reason
// the dblite object needs to be notified too
function close(code) {
if (self.listeners('close').length) {
self.emit('close', code);
} else {
log('bye bye');
}
}
// as long as there's something else to do ...
function next() {
if (queue.length) {
// ... do that and wait for next check
self.query.apply(self, queue.shift());
}
}
// common error helper
function onerror(data) {
if($callback && 1 < $callback.length) {
// there is a callback waiting
// and there is more than an argument in there
// the callback is waiting for errors too
var callback = $callback;
wasSelect = wasNotSelect = dontParseCSV = false;
$callback = $fields = null;
wasError = true;
// should the next be called ? next();
callback.call(self, new Error(data.toString()), null);
} else if(self.listeners('error').length) {
// notify listeners
self.emit('error', '' + data);
} else {
// log the output avoiding exit 1
// if no listener was added
console.error('' + data);
}
}
// all IO handled here
program.stderr.on('data', onerror);
program.stdin.on('error', onerror);
program.stdout.on('error', onerror);
program.stderr.on('error', onerror);
// invoked each time the sqlite3 shell produces an output
program.stdout.on('data', function (data) {
/*jshint eqnull: true*/
// big output might require more than a call
var str, result, callback, fields, headers, wasSelectLocal, rows;
if (wasError) {
selectResult = '';
wasError = false;
if (self.ignoreErrors) {
busy = false;
next();
}
return;
}
// the whole output is converted into a string here
selectResult += data;
// if the end of the output is the serapator
if (selectResult.slice(SUPER_SECRET_LENGTH) === SUPER_SECRET) {
// time to move forward since sqlite3 has done
str = selectResult.slice(0, SUPER_SECRET_LENGTH);
// drop the secret header if present
headers = str.slice(SUPER_SECRET_LENGTH) === SUPER_SECRET;
if (headers) str = str.slice(0, SUPER_SECRET_LENGTH);
// clean up the outer variabls
selectResult = '';
// makes the spawned program not busy anymore
busy = false;
// if it was a select
if (wasSelect || wasNotSelect) {
wasSelectLocal = wasSelect;
// set as false all conditions
// only here dontParseCSV could have been true
// set to false that too
wasSelect = wasNotSelect = dontParseCSV = busy;
// which callback should be invoked?
// last expected one for this round
callback = $callback;
// same as fields
fields = $fields;
// parse only if it was a select/pragma
if (wasSelectLocal) {
// unless specified, process the string
// converting the CSV into an Array of rows
result = dontParseCSV ? str : parseCSV(str);
// if there were headers/fields and we have a result ...
if (headers && isArray(result) && result.length) {
// ... and fields is not defined
if (fields == null) {
// fields is the row 0
fields = result[0];
} else if(!isArray(fields)) {
// per each non present key, enrich the fields object
// it is then possible to have automatic headers
// with some known field validated/parsed
// e.g. {id:Number} will be {id:Number, value:String}
// if the query was SELECT id, value FROM table
// and the fields object was just {id:Number}
// but headers were active
result[0].forEach(enrichFields, fields);
}
// drop the first row with headers
result.shift();
}
}
// Record query duration
$lastQueryTime = Date.now() - $queryStart
// but next query, should not have
// previously set callbacks or fields
$callback = $fields = null;
// the spawned program can start a new job without current callback
// being able to push another job as soon as executed. This makes
// the queue fair for everyone without granting priority to anyone.
next();
// if there was actually a callback to call
if (callback) {
rows = fields ? (
// and if there was a need to parse each row
isArray(fields) ?
// as object with properties
result.map(row2object, fields) :
// or object with validated properties
result.map(row2parsed, parseFields(fields))
) :
// go for it ... otherwise returns the result as it is:
// an Array of Arrays
result
;
// if there was an error signature
if (1 < callback.length) {
callback.call(self, null, rows);
} else {
// invoke it with the db object as context
callback.call(self, rows);
}
}
} else {
// not a select, just a special command
// such .databases or .tables
next();
// if there is something to notify
if (str.length) {
// and if there was an 'info' listener
if (self.listeners('info').length) {
// notify
self.emit('info', EOL + str);
} else {
// otherwise log
log(EOL + str);
}
}
}
}
});
// detach the program from this one
// node 0.6 has not unref
if (program.unref) {
program.on('close', close);
program.unref();
} else {
IS_NODE_06 = true;
program.stdout.on('close', close);
}
// WARNING: this can be very unsafe !!!
// if there is an error and this
// property is explicitly set to false
// it keeps running queries no matter what
self.ignoreErrors = false;
// - - - - - - - - - - - - - - - - - - - -
// safely closes the process
// will emit 'close' once done
self.close = function() {
// close can happen only once
if (!notWorking) {
// this should gently terminate the program
// only once everything scheduled has been completed
self.query('.exit');
notWorking = true;
// the hardly killed version was like this:
// program.stdin.end();
// program.kill();
}
};
self.lastQueryTime = function () {
return $lastQueryTime;
};
// SELECT last_insert_rowid() FROM table might not work as expected
// This method makes the operation atomic and reliable
self.lastRowID = function(table, callback) {
self.query(
'SELECT ROWID FROM `' + table + '` ORDER BY ROWID DESC LIMIT 1',
function(result){
var row = result[0], k;
// if headers are switched on
if (!(row instanceof Array)) {
for (k in row) {
if (row.hasOwnProperty(k)) {
row = [row[k]];
break;
}
}
}
(callback || log).call(self, row[0]);
}
);
return self;
};
// Handy if for some reason data has to be passed around
// as string instead of being serialized and deserialized
// as Array of Arrays. Don't use if not needed.
self.plain = function() {
dontParseCSV = true;
return self.query.apply(self, arguments);
};
// main logic/method/entry point
self.query = function(string, params, fields, callback) {
// notWorking is set once .close() has been called
// it does not make sense to execute anything after
// the program is being closed
if (notWorking) return onerror('closing'), self;
// if something is still going on in the sqlite3 shell
// the progcess is flagged as busy. Just queue other operations
if (busy) return queue.push(arguments), self;
// if a SELECT or a PRAGMA ...
// Record start time
$queryStart = Date.now()
wasSelect = SELECT.test(string);
if (wasSelect) {
// SELECT and PRAGMA makes `dblite` busy
busy = true;
switch(arguments.length) {
// all arguments passed, nothing to do
case 4:
$callback = callback;
$fields = fields;
string = replaceString(string, params);
break;
// 3 arguments passed ...
case 3:
// is the last one the callback ?
if (typeof fields == 'function') {
// assign it
$callback = fields;
// has string parameters to repalce
// such ? or :id and others ?
if (HAS_PARAMS.test(string)) {
// no objectification and/or validation needed
$fields = null;
// string replaced wit parameters
string = replaceString(string, params);
} else {
// no replacement in the SQL needed
// objectification with validation
// if specified, will manage the result
$fields = params;
}
} else {
// no callback specified at all, probably in "dev mode"
$callback = log; // just log the result
$fields = fields; // use objectification
string = replaceString(string, params); // replace parameters
}
break;
// in this case ...
case 2:
// simple query with a callback
if (typeof params == 'function') {
// no objectification
$fields = null;
// callback is params argument
$callback = params;
} else {
// "dev mode", just log
$callback = log;
// if there's something to replace
if (HAS_PARAMS.test(string)) {
// no objectification
$fields = null;
string = replaceString(string, params);
} else {
// nothing to replace
// objectification with eventual validation
$fields = params;
}
}
break;
default:
// 1 argument, the SQL string and nothing else
// "dev mode" log will do
$callback = log;
$fields = null;
break;
}
// ask the sqlite3 shell ...
program.stdin.write(
// trick to always know when the console is not busy anymore
// specially for those cases where no result is shown
sanitize(string) + 'SELECT ' + SUPER_SECRET_SELECT
);
} else {
// if db.plain() was used but this is not a SELECT or PRAGMA
// something is wrong with the logic since no result
// was expected anyhow
if (dontParseCSV) {
dontParseCSV = false;
throw new Error('not a select');
} else if (string[0] === '.') {
// .commands are special queries .. so
// .commands make `dblite` busy
busy = true;
// same trick with the secret to emit('info', resultAsString)
// once everything is done
program.stdin.write(string + EOL + 'SELECT ' + SUPER_SECRET_SELECT);
} else {
switch(arguments.length) {
case 1:
/* falls through */
case 2:
if (typeof params !== 'function') {
// no need to make the shell busy
// since no output is shown at all (errors ... eventually)
// sqlite3 shell will take care of the order
// same as writing in a linux shell while something else is going on
// who cares, will show when possible, after current job ^_^
program.stdin.write(sanitize(HAS_PARAMS.test(string) ?
replaceString(string, params) :
string
));
// keep checking for possible following operations
process.nextTick(next);
break;
}
fields = params;
// not necessary but guards possible wrong replaceString
params = null;
/* falls through */
case 3:
// execute a non SELECT/PRAGMA statement
// and be notified once it's done.
// set state as busy
busy = wasNotSelect = true;
$callback = fields;
program.stdin.write(
(sanitize(
HAS_PARAMS.test(string) ?
replaceString(string, params) :
string
)) +
EOL + 'SELECT ' + SUPER_SECRET_SELECT
);
}
}
}
// chainability just useful here for multiple queries at once
return self;
};
return self;
}
// enrich possible fields object with extra headers
function enrichFields(key) {
var had = this.hasOwnProperty(key),
callback = had && this[key];
delete this[key];
this[key] = had ? callback : String;
}
// if not a memory database
// the file path should be resolved as absolute
function normalizeFirstArgument(args) {
var file = args[0];
if (file !== ':memory:') {
args[0] = path.resolve(args[0]);
}
return args;
}
// assuming generated CSV is always like
// 1,what,everEOL
// with double quotes when necessary
// 2,"what's up",everEOL
// this parser works like a charm
function parseCSV(output) {
defineCSVEOL();
for(var
fields = [],
rows = [],
index = 0,
rindex = 0,
length = output.length,
i = 0,
j, loop,
current,
endLine,
iNext,
str;
i < length; i++
) {
switch(output[i]) {
case '"':
loop = true;
j = i;
do {
iNext = output.indexOf('"', current = j + 1);
switch(output[j = iNext + 1]) {
case EOL[0]:
if (EOL_LENGTH === 2 && output[j + 1] !== EOL[1]) {
break;
}
/* falls through */
case ',':
loop = false;
}
} while(loop);
str = output.slice(i + 1, iNext++).replace(DOUBLE_DOUBLE_QUOTES, '"');
break;
default:
iNext = output.indexOf(',', i);
endLine = output.indexOf(EOL, i);
if (iNext < 0) iNext = length - EOL_LENGTH;
str = output.slice(i, endLine < iNext ? (iNext = endLine) : iNext);
break;
}
fields[index++] = str;
if (output[i = iNext] === EOL[0] && (
EOL_LENGTH === 1 || (
output[i + 1] === EOL[1] && ++i
)
)
) {
rows[rindex++] = fields;
fields = [];
index = 0;
}
}
return rows;
}
// create an object with right validation
// and right fields to simplify the parsing
// NOTE: this is based on ordered key
// which is not specified by old ES specs
// but it works like this in V8
function parseFields($fields) {
for (var
current,
fields = Object.keys($fields),
parsers = [],
length = fields.length,
i = 0; i < length; i++
) {
current = $fields[fields[i]];
parsers[i] = current === Boolean ?
$Boolean : (
current === Date ?
$Date :
current || String
)
;
}
return {f: fields, p: parsers};
}
// transform SQL strings using parameters
function replaceString(string, params) {
// if params is an array
if (isArray(params)) {
// replace all ? occurence ? with right
// incremental params[index++]
paramsIndex = 0;
paramsArray = params;
string = string.replace(REPLACE_QUESTIONMARKS, replaceQuestions);
} else {
// replace :all @fields with the right
// object.all or object.fields occurrences
paramsObject = params;
string = string.replace(REPLACE_PARAMS, replaceParams);
}
paramsArray = paramsObject = null;
return string;
}
// escape the property found in the SQL
function replaceParams(match, key) {
return escape(paramsObject[key]);
}
// escape the value found for that ? in the SQL
function replaceQuestions() {
return escape(paramsArray[paramsIndex++]);
}
// objectification: makes an Array an object
// assuming the context is an array of ordered fields
function row2object(row) {
for (var
out = {},
length = this.length,
i = 0; i < length; i++
) {
out[this[i]] = row[i];
}
return out;
}
// objectification with validation:
// makes an Array a validated object
// assuming the context is an object
// produced via parseFields() function
function row2parsed(row) {
for (var
out = {},
fields = this.f,
parsers = this.p,
length = fields.length,
i = 0; i < length; i++
) {
out[fields[i]] = parsers[i](row[i]);
}
return out;
}
// escape in a smart way generic values
// making them compatible with SQLite types
// or useful for JavaScript once retrieved back
function escape(what) {
defineCSVEOL();
/*jshint eqnull: true*/
switch (typeof what) {
case 'string':
return "'" + what.replace(
SINGLE_QUOTES, SINGLE_QUOTES_DOUBLED
) + "'";
case 'object':
return what == null ?
'null' :
("'" + JSON.stringify(what).replace(
SINGLE_QUOTES, SINGLE_QUOTES_DOUBLED
) + "'")
;
// SQLite has no Boolean type
case 'boolean':
return what ? '1' : '0'; // 1 => true, 0 => false
case 'number':
// only finite numbers can be stored
if (isFinite(what)) return '' + what;
case 'undefined':
return 'NULL'
}
// all other cases
throw new Error('unsupported data type');
}
// makes an SQL statement OK for dblite <=> sqlite communications
function sanitize(string) {
return string.replace(SANITIZER, '') + SANITIZER_REPLACER;
}
// no Boolean type in SQLite
// this will replace the possible Boolean validator
// returning the right expected value
function $Boolean(field) {
switch(field.toLowerCase()) {
case '0':
case 'false':
case 'null':
case '':
return false;
}
return true;
}
// no Date in SQLite, this will
// take care of validating/creating Dates
// when the field is retrieved with a Date validator
function $Date(field) {
return new Date(
DECIMAL.test(field) ? parseInt(field, 10) : field
);
}
// which sqlite3 executable ?
// it is possible to specify a different
// sqlite3 executable even in relative paths
// be sure the file exists and is usable as executable
Object.defineProperty(
dblite,
'bin',
{
get: function () {
// normalized string if was a path
return bin.join(PATH_SEP);
},
set: function (value) {
var isPath = -1 < value.indexOf(PATH_SEP);
if (isPath) {
// resolve the path
value = path.resolve(value);
// verify it exists
if (!require(IS_NODE_06 ? 'path' : 'fs').existsSync(value)) {
throw 'invalid executable: ' + value;
}
}
// assign as Array in any case
bin = value.split(PATH_SEP);
}
}
);
// starting from v0.6.0 sqlite version shuold be specified
// specially if SQLite version is 3.8.6 or greater
// var dblite = require('dblite').withSQLite('3.8.6')
dblite.withSQLite = function (sqliteVersion) {
dblite.sqliteVersion = sqliteVersion;
return dblite;
};
// to manually parse CSV data if necessary
// mainly to be able to use db.plain(SQL)
// without parsing it right away and pass the string
// around instead of serializing and de-serializing it
// all the time. Ideally this is a scenario for clusters
// no need to usually do manually anything otherwise.
dblite.parseCSV = parseCSV;
// how to manually escape data
// might be handy to write directly SQL strings
// instead of using handy paramters Array/Object
// usually you don't want to do this
dblite.escape = escape;
// that's it!
module.exports = dblite;
/** some simple example
var db =
require('./build/dblite.node.js')('./test/dblite.test.sqlite').
on('info', console.log.bind(console)).
on('error', console.error.bind(console)).
on('close', console.log.bind(console));
// CORE FUNCTIONS: http://www.sqlite.org/lang_corefunc.html
// PRAGMA: http://www.sqlite.org/pragma.html
db.query('PRAGMA table_info(kvp)');
// to test memory database
var db = require('./build/dblite.node.js')(':memory:');
db.query('CREATE TABLE test (key INTEGER PRIMARY KEY, value TEXT)') && undefined;
db.query('INSERT INTO test VALUES(null, "asd")') && undefined;
db.query('SELECT * FROM test') && undefined;
// db.close();
*/