autostart server, return wrapped
This commit is contained in:
parent
4e3b9b88f7
commit
696ac1ed1c
|
@ -23,7 +23,7 @@ var sqlite = require('sqlite3-server');
|
|||
var opts = {
|
||||
key: '1892d335081d8d346e556c9c3c8ff2c3'
|
||||
, bits: 128
|
||||
, storage: path.join('/tmp/authn.sqlcipher')
|
||||
, filename: path.join('/tmp/authn.sqlcipher')
|
||||
, verbose: false
|
||||
, port: 3232 // default random
|
||||
, forceServer: true // default false
|
||||
|
|
138
client.js
138
client.js
|
@ -1,3 +1,141 @@
|
|||
'use strict';
|
||||
|
||||
/*global Promise*/
|
||||
|
||||
// TODO iterate over the prototype
|
||||
// translate request / response
|
||||
var sqlite3real = require('sqlite3');
|
||||
|
||||
/*
|
||||
function createConnection(opts) {
|
||||
var server = ;
|
||||
|
||||
return server.create(opts).then(function () {
|
||||
// created and listening
|
||||
});
|
||||
}
|
||||
*/
|
||||
|
||||
function startServer(opts) {
|
||||
return require('./server').create(opts).then(function (server) {
|
||||
// this process doesn't need to connect to itself
|
||||
// through a socket
|
||||
return server.masterClient;
|
||||
});
|
||||
}
|
||||
|
||||
function getConnection(opts) {
|
||||
if (!opts.sock) {
|
||||
opts.sock = opts.filename + '.sock';
|
||||
}
|
||||
|
||||
return new Promise(function (resolve) {
|
||||
setTimeout(function () {
|
||||
var WebSocket = require('ws');
|
||||
var ws = new WebSocket('ws+unix:' + opts.sock);
|
||||
|
||||
if (opts.server) {
|
||||
return startServer(opts);
|
||||
}
|
||||
|
||||
ws.on('error', function (err) {
|
||||
console.error('[ERROR] ws connection failed, retrying');
|
||||
console.error(err);
|
||||
|
||||
function retry() {
|
||||
setTimeout(function () {
|
||||
getConnection(opts).then(resolve);
|
||||
}, 100 + (Math.random() * 250));
|
||||
}
|
||||
|
||||
if ('ENOENT' === err.code || 'ECONNREFUSED' === err.code) {
|
||||
return startServer(opts).then(function (client) {
|
||||
// ws.masterClient = client;
|
||||
resolve({ masterClient: client });
|
||||
}, function () {
|
||||
retry();
|
||||
});
|
||||
}
|
||||
|
||||
retry();
|
||||
});
|
||||
|
||||
ws.on('open', function () {
|
||||
resolve(ws);
|
||||
});
|
||||
});
|
||||
}, 100 + (Math.random() * 250));
|
||||
}
|
||||
|
||||
function create(opts) {
|
||||
// TODO maybe use HTTP POST instead?
|
||||
return getConnection(opts).then(function (ws) {
|
||||
if (ws.masterClient) {
|
||||
return ws.masterClient;
|
||||
}
|
||||
|
||||
var db = {};
|
||||
var proto = sqlite3real.Database.prototype;
|
||||
|
||||
function rpc(fname, args) {
|
||||
var id;
|
||||
var cb;
|
||||
|
||||
if ('function' === typeof args[args.length - 1]) {
|
||||
id = Math.random();
|
||||
cb = args.pop();
|
||||
}
|
||||
|
||||
ws.send({
|
||||
type: 'rpc'
|
||||
, func: fname
|
||||
, args: args
|
||||
, filename: opts.filename
|
||||
, id: id
|
||||
});
|
||||
|
||||
if (!cb) {
|
||||
return;
|
||||
}
|
||||
|
||||
function onMessage(data) {
|
||||
if (!data || 'object' !== typeof data) {
|
||||
return;
|
||||
}
|
||||
|
||||
if (data.id !== id) {
|
||||
return;
|
||||
}
|
||||
|
||||
cb.apply(data.this, data.args);
|
||||
}
|
||||
|
||||
ws.on('message', onMessage);
|
||||
}
|
||||
|
||||
db.sanitize = require('./wrapper').sanitize;
|
||||
|
||||
Object.keys(sqlite3real.Database.prototype).forEach(function (key) {
|
||||
|
||||
if ('function' === typeof proto[key]) {
|
||||
db[key] = function () {
|
||||
rpc(key, Array.prototype.slice.call(arguments));
|
||||
};
|
||||
}
|
||||
|
||||
});
|
||||
|
||||
|
||||
// serialize
|
||||
// parallel
|
||||
db.serialize = db.parallel = function () {
|
||||
throw new Error('NOT IMPLEMENTED in SQLITE3-remote');
|
||||
};
|
||||
|
||||
return db;
|
||||
});
|
||||
}
|
||||
|
||||
|
||||
module.exports.sanitize = require('./wrapper').sanitize;
|
||||
module.exports.create = create;
|
||||
|
|
2
index.js
2
index.js
|
@ -3,7 +3,7 @@
|
|||
var numcpus = require('os').cpus().length;
|
||||
var sqlite3;
|
||||
|
||||
if (false && numcpus >= 2) {
|
||||
if (numcpus >= 2) {
|
||||
sqlite3 = require('./client');
|
||||
} else {
|
||||
sqlite3 = require('./wrapper');
|
||||
|
|
100
server.js
100
server.js
|
@ -1,54 +1,84 @@
|
|||
'use strict';
|
||||
/*global Promise*/
|
||||
|
||||
function create(options) {
|
||||
var url = require('url');
|
||||
var express = require('express');
|
||||
var app = express();
|
||||
var wss = options.wss;
|
||||
var wsses = {};
|
||||
|
||||
wss.on('connection', function (ws) {
|
||||
var location = url.parse(ws.upgradeReq.url, true);
|
||||
// you might use location.query.access_token to authenticate or share sessions
|
||||
// or ws.upgradeReq.headers.cookie (see http://stackoverflow.com/a/16395220/151312
|
||||
function createApp(server, options) {
|
||||
console.log('Create App');
|
||||
|
||||
ws.__session_id = location.query.session_id || Math.random();
|
||||
if (wsses[options.filename]) {
|
||||
return Promise.resolve(wsses[options.filename]);
|
||||
}
|
||||
|
||||
ws.on('message', function (buffer) {
|
||||
var cmd;
|
||||
return require('./wrapper').create(options).then(function (db) {
|
||||
var url = require('url');
|
||||
var express = require('express');
|
||||
var app = express();
|
||||
var wss = server.wss;
|
||||
|
||||
try {
|
||||
cmd = JSON.parse(buffer.toString('utf8'));
|
||||
} catch(e) {
|
||||
ws.send(JSON.stringify({ type: 'error', value: { message: e.message, code: "E_PARSE_JSON" } }));
|
||||
}
|
||||
wss.on('connection', function (ws) {
|
||||
var location = url.parse(ws.upgradeReq.url, true);
|
||||
// you might use location.query.access_token to authenticate or share sessions
|
||||
// or ws.upgradeReq.headers.cookie (see http://stackoverflow.com/a/16395220/151312
|
||||
|
||||
switch(cmd.type) {
|
||||
case 'init':
|
||||
break;
|
||||
ws.__session_id = location.query.session_id || Math.random();
|
||||
|
||||
case 'rpc':
|
||||
break;
|
||||
ws.on('message', function (buffer) {
|
||||
var cmd;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
try {
|
||||
cmd = JSON.parse(buffer.toString('utf8'));
|
||||
} catch(e) {
|
||||
ws.send(JSON.stringify({ type: 'error', value: { message: e.message, code: "E_PARSE_JSON" } }));
|
||||
}
|
||||
|
||||
switch(cmd.type) {
|
||||
case 'init':
|
||||
break;
|
||||
|
||||
case 'rpc':
|
||||
break;
|
||||
|
||||
default:
|
||||
break;
|
||||
}
|
||||
|
||||
});
|
||||
|
||||
ws.send(JSON.stringify({ type: 'session', value: ws.__session_id }));
|
||||
});
|
||||
|
||||
ws.send(JSON.stringify({ type: 'session', value: ws.__session_id }));
|
||||
app.masterClient = db;
|
||||
wsses[options.filename] = app;
|
||||
|
||||
return app;
|
||||
});
|
||||
}
|
||||
|
||||
/*
|
||||
var tablename = 'authn';
|
||||
if (tablename) {
|
||||
setup.push(db.runAsync("CREATE TABLE IF NOT EXISTS '" + sanitize(tablename)
|
||||
+ "' (id TEXT, secret TEXT, json TEXT, PRIMARY KEY(id))"));
|
||||
}
|
||||
*/
|
||||
function create(options) {
|
||||
console.log('Create Server');
|
||||
|
||||
/*global Promise*/
|
||||
return new Promise(function (resolve) {
|
||||
resolve(app);
|
||||
var server = require('http').createServer();
|
||||
var WebSocketServer = require('ws').Server;
|
||||
var wss = new WebSocketServer({ server: server });
|
||||
//var port = process.env.PORT || process.argv[0] || 4080;
|
||||
|
||||
console.log('options.sock');
|
||||
console.log(options.sock);
|
||||
var fs = require('fs');
|
||||
fs.unlink(options.sock, function () {
|
||||
// ignore error when socket doesn't exist
|
||||
|
||||
server.listen(options.sock, function () {
|
||||
console.log('Listening');
|
||||
});
|
||||
});
|
||||
|
||||
createApp({ server: server, wss: wss }, options).then(function (app) {
|
||||
server.on('request', app);
|
||||
resolve({ masterClient: app.masterClient });
|
||||
});
|
||||
});
|
||||
}
|
||||
|
||||
|
|
|
@ -19,11 +19,12 @@ function create(opts) {
|
|||
sqlite3.verbose();
|
||||
}
|
||||
|
||||
if (!dbs[opts.storage] || dbs[opts.storage].__key !== opts.key) {
|
||||
dbs[opts.storage] = new sqlite3.Database(opts.storage);
|
||||
if (!dbs[opts.filename] || dbs[opts.filename].__key !== opts.key) {
|
||||
dbs[opts.filename] = new sqlite3.Database(opts.filename);
|
||||
}
|
||||
|
||||
db = dbs[opts.storage];
|
||||
db = dbs[opts.filename];
|
||||
db.sanitize = sanitize;
|
||||
db.__key = opts.key;
|
||||
|
||||
return new Promise(function (resolve, reject) {
|
||||
|
|
Loading…
Reference in New Issue