knex/lib/dialects/oracle/index.js

196 lines
5.4 KiB
JavaScript
Raw Normal View History

2016-03-02 17:07:05 +01:00
// Oracle Client
// -------
const { map, flatten, values } = require('lodash');
const { promisify } = require('util');
const inherits = require('inherits');
const Client = require('../../client');
const { bufferToString } = require('../../query/string');
const Formatter = require('./formatter');
const Transaction = require('./transaction');
const QueryCompiler = require('./query/compiler');
const SchemaCompiler = require('./schema/compiler');
const ColumnBuilder = require('./schema/columnbuilder');
const ColumnCompiler = require('./schema/columncompiler');
const TableCompiler = require('./schema/tablecompiler');
const { ReturningHelper, isConnectionError } = require('./utils');
2016-03-02 17:07:05 +01:00
// Always initialize with the "QueryBuilder" and "QueryCompiler"
// objects, which extend the base 'lib/query/builder' and
// 'lib/query/compiler', respectively.
function Client_Oracle(config) {
Client.call(this, config);
2016-03-02 17:07:05 +01:00
}
inherits(Client_Oracle, Client);
2016-03-02 17:07:05 +01:00
Object.assign(Client_Oracle.prototype, {
2016-03-02 17:07:05 +01:00
dialect: 'oracle',
driverName: 'oracle',
_driver() {
return require('oracle');
2016-03-02 17:07:05 +01:00
},
2016-09-12 18:45:35 -04:00
transaction() {
return new Transaction(this, ...arguments);
2016-09-12 18:45:35 -04:00
},
2016-03-02 17:07:05 +01:00
2016-09-12 18:26:49 -04:00
formatter() {
return new Formatter(this, ...arguments);
2016-09-12 18:26:49 -04:00
},
2016-03-02 17:07:05 +01:00
2016-09-13 08:15:58 -04:00
queryCompiler() {
return new QueryCompiler(this, ...arguments);
2016-09-13 08:15:58 -04:00
},
2016-03-02 17:07:05 +01:00
2016-09-13 08:15:58 -04:00
schemaCompiler() {
return new SchemaCompiler(this, ...arguments);
2016-09-13 08:15:58 -04:00
},
2016-03-02 17:07:05 +01:00
2016-09-13 08:15:58 -04:00
columnBuilder() {
return new ColumnBuilder(this, ...arguments);
2016-09-13 08:15:58 -04:00
},
2016-03-02 17:07:05 +01:00
2016-09-13 08:15:58 -04:00
columnCompiler() {
return new ColumnCompiler(this, ...arguments);
2016-09-13 08:15:58 -04:00
},
2016-03-02 17:07:05 +01:00
2016-09-13 08:15:58 -04:00
tableCompiler() {
return new TableCompiler(this, ...arguments);
2016-09-13 08:15:58 -04:00
},
2016-03-02 17:07:05 +01:00
prepBindings(bindings) {
2016-03-02 17:07:05 +01:00
return map(bindings, (value) => {
// returning helper uses always ROWID as string
if (value instanceof ReturningHelper && this.driver) {
return new this.driver.OutParam(this.driver.OCCISTRING);
} else if (typeof value === 'boolean') {
return value ? 1 : 0;
} else if (Buffer.isBuffer(value)) {
return bufferToString(value);
2016-03-02 17:07:05 +01:00
}
return value;
});
2016-03-02 17:07:05 +01:00
},
// Get a raw connection, called by the `pool` whenever a new
// connection needs to be added to the pool.
acquireRawConnection() {
return new Promise((resolver, rejecter) => {
2019-03-05 18:46:58 +01:00
this.driver.connect(this.connectionSettings, (err, connection) => {
if (err) return rejecter(err);
connection.executeAsync = promisify(connection.execute);
2019-03-05 18:46:58 +01:00
if (this.connectionSettings.prefetchRowCount) {
connection.setPrefetchRowCount(
this.connectionSettings.prefetchRowCount
);
2016-09-13 18:12:23 -04:00
}
2019-03-05 18:46:58 +01:00
resolver(connection);
});
});
2016-03-02 17:07:05 +01:00
},
// Used to explicitly close a connection, called internally by the pool
// when a connection times out or the pool is shutdown.
async destroyRawConnection(connection) {
const close = promisify((cb) => connection.close(cb));
return close();
2016-03-02 17:07:05 +01:00
},
// Return the database for the Oracle client.
database() {
return this.connectionSettings.database;
2016-03-02 17:07:05 +01:00
},
// Position the bindings for the query.
positionBindings(sql) {
let questionCount = 0;
2016-03-02 17:07:05 +01:00
return sql.replace(/\?/g, function() {
questionCount += 1;
return `:${questionCount}`;
});
2016-03-02 17:07:05 +01:00
},
_stream(connection, obj, stream, options) {
return new Promise(function(resolver, rejecter) {
2016-09-13 18:12:23 -04:00
stream.on('error', (err) => {
if (isConnectionError(err)) {
connection.__knex__disposed = err;
2016-09-13 18:12:23 -04:00
}
rejecter(err);
});
2016-03-02 17:07:05 +01:00
stream.on('end', resolver);
const queryStream = connection.queryStream(
obj.sql,
obj.bindings,
options
);
queryStream.pipe(stream);
queryStream.on('error', function(error) {
rejecter(error);
stream.emit('error', error);
});
});
2016-03-02 17:07:05 +01:00
},
// Runs the query on the specified connection, providing the bindings
// and any other necessary prep work.
_query(connection, obj) {
2016-03-02 17:07:05 +01:00
if (!obj.sql) throw new Error('The query is empty');
return connection
.executeAsync(obj.sql, obj.bindings)
.then(function(response) {
if (!obj.returning) return response;
const rowIds = obj.outParams.map(
(v, i) => response[`returnParam${i ? i : ''}`]
);
return connection.executeAsync(obj.returningSql, rowIds);
})
.then(function(response) {
obj.response = response;
obj.rowsAffected = response.updateCount;
return obj;
})
.catch((err) => {
if (isConnectionError(err)) {
connection.__knex__disposed = err;
}
throw err;
});
2016-03-02 17:07:05 +01:00
},
// Process the response as returned from the query.
processResponse(obj, runner) {
let { response } = obj;
const { method } = obj;
2016-03-02 17:07:05 +01:00
if (obj.output) return obj.output.call(runner, response);
switch (method) {
case 'select':
case 'pluck':
case 'first':
if (obj.method === 'pluck') response = map(response, obj.pluck);
return obj.method === 'first' ? response[0] : response;
case 'insert':
case 'del':
case 'update':
case 'counter':
if (obj.returning) {
if (obj.returning.length > 1 || obj.returning[0] === '*') {
return response;
}
// return an array with values if only one returning value was specified
return flatten(map(response, values));
}
return obj.rowsAffected;
2016-03-02 17:07:05 +01:00
default:
return response;
}
},
});
2016-09-13 18:12:23 -04:00
module.exports = Client_Oracle;