| 
									
										
										
										
											2015-07-21 18:58:34 -06:00
										 |  |  | 'use strict'; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  | // I'm not a fan of singletons (as they are EVIL).
 | 
					
						
							|  |  |  | // However, this module is, by intent, meant to
 | 
					
						
							|  |  |  | // represent a worker process in a cluster.
 | 
					
						
							|  |  |  | // Thus, it is appropriately a singleton.
 | 
					
						
							|  |  |  | var processWebSocket; | 
					
						
							|  |  |  | var promiseWebSocket; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | function startServer(opts, verbs, myServer) { | 
					
						
							|  |  |  |   if (myServer) { | 
					
						
							|  |  |  |     return verbs.Promise.resolve(myServer); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  |   return require('./server').create(opts, verbs).then(function (server) { | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |     // this process doesn't need to connect to itself
 | 
					
						
							|  |  |  |     // through a socket
 | 
					
						
							|  |  |  |     return server.masterClient; | 
					
						
							|  |  |  |   }); | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  | // connection is scoped per-process, nothing more
 | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  | function getConnection(opts, verbs, mySocket, retry) { | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |   function incr(ws) { | 
					
						
							|  |  |  |     if (!ws.__count) { | 
					
						
							|  |  |  |       ws.__count = 0; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     ws.__count += 1; | 
					
						
							|  |  |  |     return ws; | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  |   if (!retry && (mySocket || processWebSocket)) { | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |     promiseWebSocket = verbs.Promise.resolve(mySocket || processWebSocket); | 
					
						
							|  |  |  |     return promiseWebSocket.then(incr); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  |   if (!retry && promiseWebSocket) { | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |     return promiseWebSocket.then(incr); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   promiseWebSocket = new verbs.Promise(function (resolve) { | 
					
						
							| 
									
										
										
										
											2015-07-24 16:11:10 -06:00
										 |  |  |     //setTimeout(function () {
 | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |       var WebSocket = require('ws'); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       // TODO how to include path and such?
 | 
					
						
							|  |  |  |       // http://unix:/absolute/path/to/unix.socket:/request/path
 | 
					
						
							|  |  |  |       // https://github.com/websockets/ws/issues/236
 | 
					
						
							|  |  |  |       var ws; | 
					
						
							| 
									
										
										
										
											2017-05-09 11:37:46 -06:00
										 |  |  |       var pathname = '/' + (require('cluster').worker||{}).id + '/' + opts.ipcKey; | 
					
						
							|  |  |  |       var queryparams = { | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |         ipcKey: opts.ipcKey | 
					
						
							|  |  |  |       , ipc_key: opts.ipcKey | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  |       , worker_id: (require('cluster').worker||{}).id | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       }; | 
					
						
							| 
									
										
										
										
											2017-05-09 11:37:46 -06:00
										 |  |  |       var search = require('querystring').stringify(queryparams); | 
					
						
							|  |  |  |       var wspath = 'ws+unix://' + opts.sock + ':' + pathname + '?' + search; | 
					
						
							|  |  |  |       // https://github.com/websockets/ws/blob/master/doc/ws.md#unix-domain-sockets
 | 
					
						
							|  |  |  |       // ws+unix:///absolule/path/to/uds_socket:/pathname?search_params
 | 
					
						
							|  |  |  |       ws = new WebSocket(wspath); | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |       ws.on('error', function (err) { | 
					
						
							|  |  |  |         console.error('[ERROR] ws connection failed, retrying'); | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |         console.error(err.stack || err); | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         function retry() { | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           // TODO eventually throw up
 | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |           setTimeout(function () { | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  |             getConnection(opts, verbs, mySocket, true).then(resolve, retry); | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |           }, 100 + (Math.random() * 250)); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-24 13:58:29 -06:00
										 |  |  |         if (!opts.connect && ('ENOENT' === err.code || 'ECONNREFUSED' === err.code)) { | 
					
						
							|  |  |  |           console.log('[NO SERVER] attempting to create a server #######################'); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           return startServer(opts, verbs).then(function (client) { | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |             // ws.masterClient = client;
 | 
					
						
							|  |  |  |             resolve({ masterClient: client }); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:04:54 -06:00
										 |  |  |           }, function (err) { | 
					
						
							|  |  |  |             console.error('[ERROR] failed to connect to sqlite3-cluster service. retrying...'); | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |             console.error(err.stack || err); | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |             retry(); | 
					
						
							|  |  |  |           }); | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         retry(); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       ws.on('open', function () { | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |         resolve(ws); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-07-24 16:11:10 -06:00
										 |  |  |     //}, 100 + (Math.random() * 250));
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |   }).then(function (ws) { | 
					
						
							|  |  |  |     if (!processWebSocket) { | 
					
						
							|  |  |  |       processWebSocket = ws; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return ws; | 
					
						
							| 
									
										
										
										
											2015-07-24 16:11:10 -06:00
										 |  |  |   }); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   return promiseWebSocket.then(incr); | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  | module.exports.createClientFactory = function (conf, verbs, _socket) { | 
					
						
							|  |  |  |   // TODO distinguish between defaults and mandates
 | 
					
						
							| 
									
										
										
										
											2015-07-24 13:58:29 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |   if (!conf.ipcKey) { | 
					
						
							|  |  |  |     throw new Error("[E_NO_IPCKEY] Your config must specify an ipcKey."); | 
					
						
							| 
									
										
										
										
											2015-07-24 13:58:29 -06:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |   return { | 
					
						
							|  |  |  |     create: function (opts, _s) { | 
					
						
							|  |  |  |       var copy = {}; | 
					
						
							| 
									
										
										
										
											2015-07-24 13:58:29 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       if (_socket && _s) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_SOCKET] Your parent has decided that you may not choose your own SOCKET. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |       if (opts.key && conf.key) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_KEY] Your parent has decided that you may not choose your own KEY. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (opts.algo && conf.algo) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_ALGO] Your parent has decided that you may not choose your own ALGO. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (opts.bits && conf.bits) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_BITS] Your parent has decided that you may not choose your own BITS. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       if (opts.dirname && conf.dirname) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_TENANT] Your parent has decided that you may not choose your own TENANT. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (opts.tenant && conf.tenant) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_TENANT] Your parent has decided that you may not choose your own TENANT. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (opts.prefix && conf.prefix) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_PREFIX] Your parent has decided that you may not choose your own PREFIX. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (opts.suffix && conf.suffix) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_SUFFIX] Your parent has decided that you may not choose your own SUFFIX. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (opts.ext && conf.ext) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_EXT] Your parent has decided that you may not choose your own EXT. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (opts.serve) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_SERVE] Your parent have forbidden you to SERVE. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (opts.sock && conf.sock) { | 
					
						
							|  |  |  |         throw new Error("[E_USR_SERVE] Your parent have forbidden you to choose your own SOCK. Don't get mad at me, take it up with them."); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       Object.keys(conf).forEach(function (key) { | 
					
						
							|  |  |  |         copy[key] = conf[key]; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |       Object.keys(opts).forEach(function (key) { | 
					
						
							|  |  |  |         copy[key] = opts[key]; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       if (!verbs) { | 
					
						
							|  |  |  |         verbs = { | 
					
						
							|  |  |  |           Promise: null | 
					
						
							|  |  |  |         }; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |       if (!verbs.Promise) { | 
					
						
							|  |  |  |         verbs.Promise = require('bluebird'); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  |       copy.connect = (require('cluster').worker && true); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       copy.sock = conf.sock; | 
					
						
							|  |  |  |       copy.tenant = conf.tenant; | 
					
						
							|  |  |  |       copy.ipcKey = conf.ipcKey; | 
					
						
							|  |  |  |       return module.exports.create(copy, verbs, _socket || _s || processWebSocket); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }; | 
					
						
							|  |  |  | }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | module.exports.create = function (opts, verbs, mySocket) { | 
					
						
							|  |  |  |   if (!verbs) { | 
					
						
							|  |  |  |     verbs = {}; | 
					
						
							| 
									
										
										
										
											2015-07-24 13:58:29 -06:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |   var PromiseA = verbs && verbs.Promise || require('bluebird'); | 
					
						
							|  |  |  |   verbs.Promise = PromiseA; | 
					
						
							|  |  |  |   // TODO iterate over the prototype
 | 
					
						
							|  |  |  |   // translate request / response
 | 
					
						
							|  |  |  |   var sqlite3real = require('sqlite3'); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   if (!mySocket) { | 
					
						
							|  |  |  |     mySocket = processWebSocket; | 
					
						
							| 
									
										
										
										
											2015-07-24 13:58:29 -06:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |   function create(opts) { | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |     function retryServe() { | 
					
						
							|  |  |  |       return startServer(opts, verbs).then(function (client) { | 
					
						
							|  |  |  |         // ws.masterClient = client;
 | 
					
						
							|  |  |  |         return { masterClient: client }; | 
					
						
							|  |  |  |       }, function (err) { | 
					
						
							|  |  |  |         console.error('[ERROR] retryServe()'); | 
					
						
							|  |  |  |         console.error(err.stack || err); | 
					
						
							|  |  |  |         retryServe(); | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (opts.serve) { | 
					
						
							|  |  |  |       return retryServe(opts).then(function (servers) { | 
					
						
							|  |  |  |         return servers.masterClient; | 
					
						
							|  |  |  |       }); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |     if (!opts.tenant) { | 
					
						
							|  |  |  |       opts.tenant = ""; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!opts.subtenant) { | 
					
						
							|  |  |  |       opts.subtenant = ""; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!opts.prefix) { | 
					
						
							|  |  |  |       opts.prefix = ''; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!opts.suffix) { | 
					
						
							|  |  |  |       opts.suffix = ''; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!opts.ext) { | 
					
						
							|  |  |  |       opts.ext = '.sqlite3'; // also .sqlcipher
 | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |     if (!opts.dbname) { | 
					
						
							|  |  |  |       throw new Error("Please specify opts.dbname as the name of the database"); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     var promise; | 
					
						
							|  |  |  |     var numcpus = require('os').cpus().length; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // if we're only on one core, use the lib directly, no socket
 | 
					
						
							|  |  |  |     if (opts.standalone || (1 === numcpus && !opts.serve && !opts.connect)) { | 
					
						
							|  |  |  |       return require('./wrapper').create(opts, verbs); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!opts.sock) { | 
					
						
							|  |  |  |       throw new Error("Please specify opts.sock as the path to the master socket. '/tmp/sqlite3-cluster' would do nicely."); | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  |     promise = getConnection(opts, verbs, mySocket, true).then(function (socket) { | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |       mySocket = socket; | 
					
						
							|  |  |  |       return mySocket; | 
					
						
							|  |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |     // TODO maybe use HTTP POST instead?
 | 
					
						
							|  |  |  |     return promise.then(function (ws) { | 
					
						
							|  |  |  |       var db = {}; | 
					
						
							|  |  |  |       var proto = sqlite3real.Database.prototype; | 
					
						
							|  |  |  |       var messages = []; | 
					
						
							|  |  |  |       var idprefix = require('crypto').randomBytes(12).toString('base64'); | 
					
						
							|  |  |  |       var idcount = 0; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       function genId() { | 
					
						
							|  |  |  |         idcount += 1; | 
					
						
							|  |  |  |         return idprefix + idcount; | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |       function init(iopts) { | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  |         //console.log('CLIENT INIT');
 | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |         if (db._initPromise) { | 
					
						
							|  |  |  |           return db._initPromise; | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |         db._initPromise = new PromiseA(function (resolve, reject) { | 
					
						
							|  |  |  |           // TODO needs to reject by a timeout
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           var id = genId(); | 
					
						
							|  |  |  |           ws.send(JSON.stringify({ | 
					
						
							|  |  |  |             type: 'init' | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |           , args: [{ | 
					
						
							|  |  |  |             // encryption
 | 
					
						
							|  |  |  |               bits: opts.bits || iopts.bits | 
					
						
							|  |  |  |             , algorithm: opts.algo || opts.algorithm || iopts.algorithm || iopts.algo | 
					
						
							|  |  |  |             , algo: opts.algo || opts.algorithm || iopts.algorithm || iopts.algo | 
					
						
							|  |  |  |             , encmode: opts.mode || iopts.mode | 
					
						
							|  |  |  |             }] | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           , func: 'init' | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |             // db
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           , dirname: opts.dirname | 
					
						
							|  |  |  |           , prefix: opts.prefix | 
					
						
							|  |  |  |           , subtenant: opts.subtenant | 
					
						
							|  |  |  |           , tenant: opts.tenant | 
					
						
							|  |  |  |           , dbname: opts.dbname | 
					
						
							|  |  |  |           , suffix: opts.suffix | 
					
						
							|  |  |  |           , ext: opts.ext | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |             // session
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           , id: id | 
					
						
							|  |  |  |           })); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           function onMessage(data) { | 
					
						
							|  |  |  |             var cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             try { | 
					
						
							|  |  |  |               cmd = JSON.parse(data.toString('utf8')); | 
					
						
							|  |  |  |             } catch(e) { | 
					
						
							|  |  |  |               console.error('[ERROR] in client, from sql server parse json'); | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |               console.error(e.stack || e); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |               console.error(data); | 
					
						
							|  |  |  |               console.error(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |               // ignore this message, it came out of order
 | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |               return reject(new Error("suxors to rejexors parse")); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (cmd.id !== id) { | 
					
						
							|  |  |  |               // ignore this message, it came out of order
 | 
					
						
							|  |  |  |               return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if (cmd.self) { | 
					
						
							|  |  |  |               cmd.args = [db]; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             messages.splice(messages.indexOf(onMessage), 1); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             if ('error' === cmd.type) { | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |               //console.log('ERROR ARGS');
 | 
					
						
							|  |  |  |               //console.log(cmd);
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |               reject(cmd.args[0]); | 
					
						
							|  |  |  |               return; | 
					
						
							|  |  |  |             } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-12-08 13:10:11 +00:00
										 |  |  |             //console.log('CLIENT RESOLVING INIT');
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |             resolve(cmd.args[0]); | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           messages.push(onMessage); | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  | 
 | 
					
						
							|  |  |  |         return db._initPromise; | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       function rpcThunk(fname, args) { | 
					
						
							|  |  |  |         var id; | 
					
						
							|  |  |  |         var cb; | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |         if ('function' === typeof args[args.length - 1]) { | 
					
						
							|  |  |  |           id = genId(); | 
					
						
							|  |  |  |           cb = args.pop(); | 
					
						
							|  |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-07-28 18:04:54 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  |         ws.send(JSON.stringify({ | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           type: 'rpc' | 
					
						
							|  |  |  |         , func: fname | 
					
						
							|  |  |  |         , args: args | 
					
						
							|  |  |  |         , dirname: opts.dirname | 
					
						
							|  |  |  |         , prefix: opts.prefix | 
					
						
							|  |  |  |         , subtenant: opts.subtenant | 
					
						
							|  |  |  |         , tenant: opts.tenant | 
					
						
							|  |  |  |         , dbname: opts.dbname | 
					
						
							|  |  |  |         , suffix: opts.suffix | 
					
						
							|  |  |  |         , ext: opts.ext | 
					
						
							| 
									
										
										
										
											2015-07-28 18:04:54 -06:00
										 |  |  |         , id: id | 
					
						
							|  |  |  |         })); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |         if (!cb) { | 
					
						
							|  |  |  |           return; | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:04:54 -06:00
										 |  |  |         function onMessage(data) { | 
					
						
							|  |  |  |           var cmd; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           try { | 
					
						
							|  |  |  |             cmd = JSON.parse(data.toString('utf8')); | 
					
						
							|  |  |  |           } catch(e) { | 
					
						
							|  |  |  |             console.error('[ERROR] in client, from sql server parse json'); | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |             console.error(e.stack || e); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:04:54 -06:00
										 |  |  |             console.error(data); | 
					
						
							|  |  |  |             console.error(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |             //ws.send(JSON.stringify({ type: 'error', value: { message: e.message, code: "E_PARSE_JSON" } }));
 | 
					
						
							|  |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (cmd.id !== id) { | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |             // ignore this message, it probably came out of order
 | 
					
						
							| 
									
										
										
										
											2015-07-28 18:04:54 -06:00
										 |  |  |             return; | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |           if (cmd.self) { | 
					
						
							|  |  |  |             cmd.args = [db]; | 
					
						
							|  |  |  |           } | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           //console.log('RESOLVING RPC', cmd.this, cmd.args);
 | 
					
						
							|  |  |  |           cb.apply(cmd.this, cmd.args); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:04:54 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |           if ('on' !== fname) { | 
					
						
							|  |  |  |             var index = messages.indexOf(onMessage); | 
					
						
							|  |  |  |             messages.splice(index, 1); | 
					
						
							| 
									
										
										
										
											2015-07-28 18:04:54 -06:00
										 |  |  |           } | 
					
						
							|  |  |  |         } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |         messages.push(onMessage); | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |       } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       db.sanitize = require('./wrapper').sanitize; | 
					
						
							|  |  |  |       db.escape = require('./wrapper').escape; | 
					
						
							| 
									
										
										
										
											2015-07-24 02:57:17 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       // TODO get methods from server (cluster-store does this)
 | 
					
						
							|  |  |  |       // instead of using the prototype
 | 
					
						
							|  |  |  |       Object.keys(sqlite3real.Database.prototype).forEach(function (key) { | 
					
						
							| 
									
										
										
										
											2015-07-24 02:57:17 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |         if ('function' === typeof proto[key]) { | 
					
						
							|  |  |  |           db[key] = function () { | 
					
						
							|  |  |  |             rpcThunk(key, Array.prototype.slice.call(arguments)); | 
					
						
							|  |  |  |           }; | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       db.init = init; | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       ws.on('message', function (data) { | 
					
						
							|  |  |  |         messages.forEach(function (fn) { | 
					
						
							|  |  |  |           try { | 
					
						
							|  |  |  |             fn(data); | 
					
						
							|  |  |  |           } catch(e) { | 
					
						
							|  |  |  |             console.error("[ERROR] ws.on('message', fn) (multi-callback)"); | 
					
						
							| 
									
										
										
										
											2015-11-10 12:16:50 +00:00
										 |  |  |             console.error(e.stack || e); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |             // ignore
 | 
					
						
							|  |  |  |           } | 
					
						
							|  |  |  |         }); | 
					
						
							|  |  |  |       }); | 
					
						
							| 
									
										
										
										
											2015-07-24 02:57:17 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       // serialize
 | 
					
						
							|  |  |  |       // parallel
 | 
					
						
							|  |  |  |       db.serialize = db.parallel = function () { | 
					
						
							|  |  |  |         throw new Error('NOT IMPLEMENTED in SQLITE3-remote'); | 
					
						
							|  |  |  |       }; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       db.close = function (fn) { | 
					
						
							|  |  |  |         ws.__count -= 1; | 
					
						
							|  |  |  |         if (!ws.__count) { | 
					
						
							|  |  |  |           // waiting for https://github.com/websockets/ws/issues/613 to land
 | 
					
						
							|  |  |  |           // 1000 means 'normal' https://github.com/websockets/ws/blob/master/lib/ErrorCodes.js
 | 
					
						
							|  |  |  |           ws.close(1000, null, fn); | 
					
						
							| 
									
										
										
										
											2015-07-24 02:57:17 -06:00
										 |  |  |         } | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       }; | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-17 08:33:54 +00:00
										 |  |  |       if (opts.init) { | 
					
						
							|  |  |  |         return db.init(opts); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |       return db; | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  |     }); | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-11-09 20:49:53 +00:00
										 |  |  |   return create(opts); | 
					
						
							|  |  |  | }; | 
					
						
							| 
									
										
										
										
											2015-07-22 20:58:18 -06:00
										 |  |  | 
 | 
					
						
							|  |  |  | module.exports.sanitize = require('./wrapper').sanitize; | 
					
						
							| 
									
										
										
										
											2015-07-24 14:47:11 -06:00
										 |  |  | module.exports.escape = require('./wrapper').escape; |