node-postgres/lib/native/index.js

236 lines
6.7 KiB
JavaScript
Raw Normal View History

2011-02-21 06:12:06 +08:00
//require the c++ bindings & export to javascript
var EventEmitter = require('events').EventEmitter;
var ConnectionParameters = require(__dirname + '/../connection-parameters');
var CopyFromStream = require(__dirname + '/../copystream').CopyFromStream;
var CopyToStream = require(__dirname + '/../copystream').CopyToStream;
2011-10-11 11:03:27 +08:00
var binding;
//TODO remove on v1.0.0
2013-01-24 07:52:55 +08:00
try {
2011-10-11 11:03:27 +08:00
//v0.5.x
2013-01-24 07:52:55 +08:00
binding = require(__dirname + '/../../build/Release/binding.node');
2011-10-11 11:03:27 +08:00
} catch(e) {
//v0.4.x
2013-01-24 07:52:55 +08:00
binding = require(__dirname + '/../../build/default/binding');
2011-10-11 11:03:27 +08:00
}
2011-08-30 11:53:38 +08:00
var Connection = binding.Connection;
2011-08-30 12:06:07 +08:00
var NativeQuery = require(__dirname + '/query');
2011-08-30 11:53:38 +08:00
2011-10-11 11:03:27 +08:00
for(var k in EventEmitter.prototype) {
Connection.prototype[k] = EventEmitter.prototype[k];
2011-10-11 11:03:27 +08:00
}
2011-02-23 13:52:25 +08:00
var nativeConnect = Connection.prototype.connect;
2011-02-23 13:52:25 +08:00
Connection.prototype.connect = function(cb) {
2011-02-23 13:52:25 +08:00
var self = this;
this.connectionParameters.getLibpqConnectionString(function(err, conString) {
if(err) {
return cb ? cb(err) : self.emit('error', err);
}
if(cb) {
var errCallback;
var connectCallback = function() {
//remove single-fire connection error callback
self.removeListener('error', errCallback);
cb(null);
};
errCallback = function(err) {
//remove singel-fire connection success callback
self.removeListener('connect', connectCallback);
cb(err);
};
self.once('connect', connectCallback);
self.once('error', errCallback);
}
nativeConnect.call(self, conString);
});
};
Connection.prototype._copy = function (text, stream) {
var q = new NativeQuery(text, function (error) {
if (error) {
q.stream.error(error);
} else {
q.stream.close();
}
2013-01-24 07:52:55 +08:00
});
q.stream = stream;
this._queryQueue.push(q);
this._pulseQueryQueue();
return q.stream;
};
Connection.prototype.copyFrom = function (text) {
return this._copy(text, new CopyFromStream());
};
Connection.prototype.copyTo = function (text) {
return this._copy(text, new CopyToStream());
};
Connection.prototype.sendCopyFromChunk = function (chunk) {
2013-01-24 07:52:55 +08:00
this._sendCopyFromChunk(chunk);
};
Connection.prototype.endCopyFrom = function (msg) {
this._endCopyFrom(msg);
};
Connection.prototype.query = function(config, values, callback) {
2013-01-24 07:52:55 +08:00
var query = (config instanceof NativeQuery) ? config :
new NativeQuery(config, values, callback);
this._queryQueue.push(query);
2011-02-23 13:52:25 +08:00
this._pulseQueryQueue();
return query;
};
2011-02-23 13:52:25 +08:00
var nativeCancel = Connection.prototype.cancel;
Connection.prototype.cancel = function(client, query) {
if (client._activeQuery == query) {
this.connect(nativeCancel.bind(client));
} else if (client._queryQueue.indexOf(query) != -1) {
client._queryQueue.splice(client._queryQueue.indexOf(query), 1);
}
};
Connection.prototype._pulseQueryQueue = function(initialConnection) {
2011-02-23 13:52:25 +08:00
if(!this._connected) {
return;
}
if(this._activeQuery) {
return;
}
var query = this._queryQueue.shift();
if(!query) {
if(!initialConnection) {
//TODO remove all the pause-drain stuff for v1.0
if(this._drainPaused) {
this._drainPaused++;
} else {
this.emit('drain');
}
}
2011-02-23 13:52:25 +08:00
return;
}
this._activeQuery = query;
2011-03-06 02:01:57 +08:00
if(query.name) {
if(this._namedQueries[query.name]) {
this._sendQueryPrepared(query.name, query.values||[]);
} else {
this._namedQuery = true;
this._namedQueries[query.name] = true;
this._sendPrepare(query.name, query.text, (query.values||[]).length);
}
} else if(query.values) {
2011-02-25 11:33:54 +08:00
//call native function
this._sendQueryWithParams(query.text, query.values);
2011-02-25 11:33:54 +08:00
} else {
//call native function
this._sendQuery(query.text);
}
};
2011-02-23 13:52:25 +08:00
//TODO remove all the pause-drain stuff for v1.0
Connection.prototype.pauseDrain = function() {
this._drainPaused = 1;
};
//TODO remove all the pause-drain stuff for v1.0
Connection.prototype.resumeDrain = function() {
if(this._drainPaused > 1) {
this.emit('drain');
}
this._drainPaused = 0;
};
Connection.prototype.sendCopyFail = function(msg) {
this.endCopyFrom(msg);
};
2011-08-30 12:06:07 +08:00
var clientBuilder = function(config) {
2011-03-04 02:46:24 +08:00
config = config || {};
2011-02-23 13:52:25 +08:00
var connection = new Connection();
connection._queryQueue = [];
connection._namedQueries = {};
2011-02-23 13:52:25 +08:00
connection._activeQuery = null;
connection.connectionParameters = new ConnectionParameters(config);
2011-08-29 15:35:08 +08:00
//attach properties to normalize interface with pure js client
connection.user = connection.connectionParameters.user;
connection.password = connection.connectionParameters.password;
connection.database = connection.connectionParameters.database;
connection.host = connection.connectionParameters.host;
connection.port = connection.connectionParameters.port;
2011-02-23 13:52:25 +08:00
connection.on('connect', function() {
connection._connected = true;
connection._pulseQueryQueue(true);
2011-02-23 13:52:25 +08:00
});
2011-02-24 12:41:54 +08:00
//proxy some events to active query
connection.on('_row', function(row) {
2011-03-01 13:09:09 +08:00
connection._activeQuery.handleRow(row);
});
connection.on('_cmdStatus', function(status) {
//set this here so we can pass it to the query
//when the query completes
connection._lastMeta = status;
});
//TODO: emit more native error properties (make it match js error)
connection.on('_error', function(err) {
//create Error object from object literal
var error = new Error(err.message || "Unknown native driver error");
for(var key in err) {
error[key] = err[key];
}
2011-03-07 11:32:58 +08:00
//give up on trying to wait for named query prepare
this._namedQuery = false;
if(connection._activeQuery) {
connection._activeQuery.handleError(error);
} else {
connection.emit('error', error);
}
});
connection.on('_readyForQuery', function() {
var q = this._activeQuery;
2011-03-07 11:32:58 +08:00
//a named query finished being prepared
if(this._namedQuery) {
this._namedQuery = false;
this._sendQueryPrepared(q.name, q.values||[]);
2011-03-07 11:32:58 +08:00
} else {
connection._activeQuery.handleReadyForQuery(connection._lastMeta);
2011-03-07 11:32:58 +08:00
connection._activeQuery = null;
connection._pulseQueryQueue();
}
2011-02-24 09:40:52 +08:00
});
connection.on('copyInResponse', function () {
2012-10-08 01:12:30 +08:00
//connection is ready to accept chunks
2013-01-24 07:52:55 +08:00
//start to send data from stream
connection._activeQuery.streamData(connection);
});
connection.on('copyOutResponse', function(msg) {
2013-01-24 07:52:55 +08:00
if (connection._activeQuery.stream === undefined) {
connection._activeQuery._canceledDueToError = new Error('No destination stream defined');
(new clientBuilder({port: connection.port, host: connection.host})).cancel(connection, connection._activeQuery);
2013-01-24 07:52:55 +08:00
}
});
connection.on('copyData', function (chunk) {
2012-10-08 01:12:30 +08:00
//recieve chunk from connection
//move it to stream
connection._activeQuery.handleCopyFromChunk(chunk);
});
2011-02-23 13:52:25 +08:00
return connection;
2011-02-24 09:40:52 +08:00
};
// expose a Query constructor
clientBuilder.Query = NativeQuery;
2011-08-30 12:06:07 +08:00
module.exports = clientBuilder;