2010-11-03 13:27:11 +08:00
|
|
|
var EventEmitter = require('events').EventEmitter;
|
2011-10-11 08:40:52 +08:00
|
|
|
var util = require('util');
|
|
|
|
|
2012-07-06 11:08:18 +08:00
|
|
|
var Result = require(__dirname + '/result');
|
|
|
|
var utils = require(__dirname + '/utils');
|
2010-11-03 13:27:11 +08:00
|
|
|
|
2012-08-10 07:31:32 +08:00
|
|
|
var Query = function(config, values, callback) {
|
|
|
|
// use of "new" optional
|
2013-03-07 00:26:40 +08:00
|
|
|
if(!(this instanceof Query)) { return new Query(config, values, callback); }
|
2013-01-24 08:12:12 +08:00
|
|
|
|
2012-12-11 14:30:16 +08:00
|
|
|
config = utils.normalizeQueryConfig(config, values, callback);
|
2013-01-24 08:12:12 +08:00
|
|
|
|
2010-11-03 13:27:11 +08:00
|
|
|
this.text = config.text;
|
|
|
|
this.values = config.values;
|
|
|
|
this.rows = config.rows;
|
|
|
|
this.types = config.types;
|
|
|
|
this.name = config.name;
|
2011-02-14 23:42:04 +08:00
|
|
|
this.binary = config.binary;
|
2013-01-16 19:04:28 +08:00
|
|
|
this.stream = config.stream;
|
2011-07-13 12:08:16 +08:00
|
|
|
//use unique portal name each time
|
2013-01-21 21:35:52 +08:00
|
|
|
this.portal = config.portal || "";
|
2010-11-15 14:10:21 +08:00
|
|
|
this.callback = config.callback;
|
2014-04-12 13:29:20 +08:00
|
|
|
if(process.domain && config.callback) {
|
|
|
|
this.callback = process.domain.bind(config.callback);
|
|
|
|
}
|
2011-02-05 08:51:34 +08:00
|
|
|
this._fieldNames = [];
|
|
|
|
this._fieldConverters = [];
|
2013-07-09 06:45:06 +08:00
|
|
|
this._result = new Result(config.rowMode);
|
2011-02-05 09:30:30 +08:00
|
|
|
this.isPreparedStatement = false;
|
2013-01-18 06:24:08 +08:00
|
|
|
this._canceledDueToError = false;
|
2010-11-03 13:27:11 +08:00
|
|
|
EventEmitter.call(this);
|
|
|
|
};
|
|
|
|
|
2011-10-11 08:40:52 +08:00
|
|
|
util.inherits(Query, EventEmitter);
|
2010-11-03 13:27:11 +08:00
|
|
|
|
2013-03-06 22:48:52 +08:00
|
|
|
Query.prototype.requiresPreparation = function() {
|
2012-12-11 12:44:58 +08:00
|
|
|
//named queries must always be prepared
|
2013-01-21 21:35:52 +08:00
|
|
|
if(this.name) { return true; }
|
2012-12-11 12:44:58 +08:00
|
|
|
//always prepare if there are max number of rows expected per
|
2013-01-24 08:12:12 +08:00
|
|
|
//portal execution
|
2013-01-21 21:35:52 +08:00
|
|
|
if(this.rows) { return true; }
|
2012-12-11 12:44:58 +08:00
|
|
|
//don't prepare empty text queries
|
2013-01-21 21:35:52 +08:00
|
|
|
if(!this.text) { return false; }
|
2012-12-11 12:44:58 +08:00
|
|
|
//binary should be prepared to specify results should be in binary
|
|
|
|
//unless there are no parameters
|
2013-01-21 21:35:52 +08:00
|
|
|
if(this.binary && !this.values) { return false; }
|
2012-12-11 12:44:58 +08:00
|
|
|
//prepare if there are values
|
|
|
|
return (this.values || 0).length > 0;
|
2010-11-03 13:27:11 +08:00
|
|
|
};
|
|
|
|
|
|
|
|
|
2011-02-05 08:51:34 +08:00
|
|
|
//associates row metadata from the supplied
|
|
|
|
//message with this query object
|
|
|
|
//metadata used when parsing row results
|
2013-03-06 22:48:52 +08:00
|
|
|
Query.prototype.handleRowDescription = function(msg) {
|
2013-07-08 22:30:10 +08:00
|
|
|
this._result.addFields(msg.fields);
|
2011-02-05 08:51:34 +08:00
|
|
|
};
|
2011-01-24 09:01:28 +08:00
|
|
|
|
2013-03-06 22:48:52 +08:00
|
|
|
Query.prototype.handleDataRow = function(msg) {
|
2013-07-08 22:30:10 +08:00
|
|
|
var row = this._result.parseRow(msg.fields);
|
|
|
|
this.emit('row', row, this._result);
|
2010-11-15 06:50:38 +08:00
|
|
|
|
2011-02-05 08:51:34 +08:00
|
|
|
//if there is a callback collect rows
|
2013-07-08 22:30:10 +08:00
|
|
|
if(this.callback) {
|
|
|
|
this._result.addRow(row);
|
2011-02-05 08:51:34 +08:00
|
|
|
}
|
|
|
|
};
|
2010-12-03 07:47:54 +08:00
|
|
|
|
2013-10-21 22:20:21 +08:00
|
|
|
Query.prototype.handleCommandComplete = function(msg, con) {
|
2014-10-20 10:53:08 +08:00
|
|
|
if(this.name) {
|
|
|
|
con.parsedStatements[this.name] = true;
|
|
|
|
}
|
2011-02-05 08:51:34 +08:00
|
|
|
this._result.addCommandComplete(msg);
|
2013-10-21 22:20:21 +08:00
|
|
|
//need to sync after each command complete of a prepared statement
|
|
|
|
if(this.isPreparedStatement) {
|
2013-10-21 22:39:49 +08:00
|
|
|
con.sync();
|
2013-10-21 22:20:21 +08:00
|
|
|
}
|
2011-02-05 08:51:34 +08:00
|
|
|
};
|
2010-11-15 06:50:38 +08:00
|
|
|
|
2013-03-06 22:48:52 +08:00
|
|
|
Query.prototype.handleReadyForQuery = function() {
|
2013-03-07 00:26:40 +08:00
|
|
|
if(this._canceledDueToError) {
|
2013-01-18 20:04:21 +08:00
|
|
|
return this.handleError(this._canceledDueToError);
|
|
|
|
}
|
2011-02-05 09:15:57 +08:00
|
|
|
if(this.callback) {
|
|
|
|
this.callback(null, this._result);
|
|
|
|
}
|
|
|
|
this.emit('end', this._result);
|
|
|
|
};
|
2011-01-24 09:01:28 +08:00
|
|
|
|
2013-10-21 22:20:21 +08:00
|
|
|
Query.prototype.handleError = function(err, connection) {
|
|
|
|
//need to sync after error during a prepared statement
|
|
|
|
if(this.isPreparedStatement) {
|
|
|
|
connection.sync();
|
|
|
|
}
|
2013-03-07 00:26:40 +08:00
|
|
|
if(this._canceledDueToError) {
|
2013-01-18 06:24:08 +08:00
|
|
|
err = this._canceledDueToError;
|
|
|
|
this._canceledDueToError = false;
|
|
|
|
}
|
2011-02-05 09:15:57 +08:00
|
|
|
//if callback supplied do not emit error event as uncaught error
|
|
|
|
//events will bubble up to node process
|
|
|
|
if(this.callback) {
|
2014-04-07 00:53:47 +08:00
|
|
|
return this.callback(err);
|
2011-02-05 09:15:57 +08:00
|
|
|
}
|
2014-04-07 00:53:47 +08:00
|
|
|
this.emit('error', err);
|
2011-02-05 09:15:57 +08:00
|
|
|
};
|
2010-11-15 06:50:38 +08:00
|
|
|
|
2013-03-06 22:48:52 +08:00
|
|
|
Query.prototype.submit = function(connection) {
|
2010-11-03 13:27:11 +08:00
|
|
|
if(this.requiresPreparation()) {
|
|
|
|
this.prepare(connection);
|
|
|
|
} else {
|
|
|
|
connection.query(this.text);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2013-03-06 22:48:52 +08:00
|
|
|
Query.prototype.hasBeenParsed = function(connection) {
|
2010-11-03 13:27:11 +08:00
|
|
|
return this.name && connection.parsedStatements[this.name];
|
|
|
|
};
|
|
|
|
|
2013-10-21 22:20:21 +08:00
|
|
|
Query.prototype.handlePortalSuspended = function(connection) {
|
|
|
|
this._getRows(connection, this.rows);
|
|
|
|
};
|
|
|
|
|
|
|
|
Query.prototype._getRows = function(connection, rows) {
|
2011-02-05 09:30:30 +08:00
|
|
|
connection.execute({
|
2011-07-13 12:08:16 +08:00
|
|
|
portal: this.portalName,
|
2013-10-21 22:20:21 +08:00
|
|
|
rows: rows
|
2011-04-17 00:42:23 +08:00
|
|
|
}, true);
|
2011-02-05 09:30:30 +08:00
|
|
|
connection.flush();
|
|
|
|
};
|
|
|
|
|
2013-03-06 22:48:52 +08:00
|
|
|
Query.prototype.prepare = function(connection) {
|
2010-11-03 13:27:11 +08:00
|
|
|
var self = this;
|
2011-02-05 09:30:30 +08:00
|
|
|
//prepared statements need sync to be called after each command
|
|
|
|
//complete or when an error is encountered
|
|
|
|
this.isPreparedStatement = true;
|
|
|
|
//TODO refactor this poor encapsulation
|
2010-11-03 13:27:11 +08:00
|
|
|
if(!this.hasBeenParsed(connection)) {
|
|
|
|
connection.parse({
|
|
|
|
text: self.text,
|
|
|
|
name: self.name,
|
|
|
|
types: self.types
|
2011-04-17 00:42:23 +08:00
|
|
|
}, true);
|
2010-11-03 13:27:11 +08:00
|
|
|
}
|
|
|
|
|
2011-05-02 13:32:30 +08:00
|
|
|
//TODO is there some better way to prepare values for the database?
|
2010-11-03 13:27:11 +08:00
|
|
|
if(self.values) {
|
2012-07-06 11:08:18 +08:00
|
|
|
for(var i = 0, len = self.values.length; i < len; i++) {
|
|
|
|
self.values[i] = utils.prepareValue(self.values[i]);
|
|
|
|
}
|
2010-11-03 13:27:11 +08:00
|
|
|
}
|
|
|
|
|
|
|
|
//http://developer.postgresql.org/pgdocs/postgres/protocol-flow.html#PROTOCOL-FLOW-EXT-QUERY
|
|
|
|
connection.bind({
|
2011-07-13 12:08:16 +08:00
|
|
|
portal: self.portalName,
|
2010-11-03 13:27:11 +08:00
|
|
|
statement: self.name,
|
2011-02-14 23:42:04 +08:00
|
|
|
values: self.values,
|
|
|
|
binary: self.binary
|
2011-04-17 00:42:23 +08:00
|
|
|
}, true);
|
2010-11-03 13:27:11 +08:00
|
|
|
|
|
|
|
connection.describe({
|
|
|
|
type: 'P',
|
2011-07-13 12:08:16 +08:00
|
|
|
name: self.portalName || ""
|
2011-04-17 00:42:23 +08:00
|
|
|
}, true);
|
2010-11-03 13:27:11 +08:00
|
|
|
|
2013-10-21 22:20:21 +08:00
|
|
|
this._getRows(connection, this.rows);
|
2010-11-03 13:27:11 +08:00
|
|
|
};
|
2013-03-06 22:48:52 +08:00
|
|
|
|
2013-10-22 13:23:43 +08:00
|
|
|
Query.prototype.handleCopyInResponse = function (connection) {
|
2013-03-07 00:26:40 +08:00
|
|
|
if(this.stream) this.stream.startStreamingToConnection(connection);
|
2013-01-24 08:12:12 +08:00
|
|
|
else connection.sendCopyFail('No source stream defined');
|
2012-09-27 18:28:00 +08:00
|
|
|
};
|
2013-03-06 22:48:52 +08:00
|
|
|
|
2013-10-22 13:23:43 +08:00
|
|
|
Query.prototype.handleCopyData = function (msg, connection) {
|
|
|
|
var chunk = msg.chunk;
|
2013-03-07 00:26:40 +08:00
|
|
|
if(this.stream) {
|
2013-01-18 06:24:08 +08:00
|
|
|
this.stream.handleChunk(chunk);
|
2013-01-24 08:12:12 +08:00
|
|
|
}
|
2013-01-18 06:24:08 +08:00
|
|
|
//if there are no stream (for example when copy to query was sent by
|
2013-01-24 08:12:12 +08:00
|
|
|
//query method instead of copyTo) error will be handled
|
|
|
|
//on copyOutResponse event, so silently ignore this error here
|
2013-01-21 21:35:52 +08:00
|
|
|
};
|
2010-11-03 13:27:11 +08:00
|
|
|
module.exports = Query;
|