2016-04-01 22:25:16 +00:00
|
|
|
// Copyright IBM Corp. 2015,2016. All Rights Reserved.
|
|
|
|
// Node module: loopback-datasource-juggler
|
|
|
|
// This file is licensed under the MIT License.
|
|
|
|
// License text available at https://opensource.org/licenses/MIT
|
|
|
|
|
2016-07-22 19:26:07 +00:00
|
|
|
var g = require('strong-globalize')();
|
2015-05-15 17:26:49 +00:00
|
|
|
var debug = require('debug')('loopback:connector:transaction');
|
|
|
|
var uuid = require('node-uuid');
|
|
|
|
var utils = require('./utils');
|
|
|
|
var jutil = require('./jutil');
|
|
|
|
var ObserverMixin = require('./observer');
|
|
|
|
|
2015-05-13 16:33:49 +00:00
|
|
|
var Transaction = require('loopback-connector').Transaction;
|
|
|
|
|
|
|
|
module.exports = TransactionMixin;
|
|
|
|
|
2015-05-18 16:00:49 +00:00
|
|
|
/**
|
|
|
|
* TransactionMixin class. Use to add transaction APIs to a model class.
|
|
|
|
*
|
|
|
|
* @class TransactionMixin
|
|
|
|
*/
|
2015-05-13 16:33:49 +00:00
|
|
|
function TransactionMixin() {
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Begin a new transaction
|
|
|
|
* @param {Object|String} [options] Options can be one of the forms:
|
|
|
|
* - Object: {isolationLevel: '...', timeout: 1000}
|
|
|
|
* - String: isolationLevel
|
|
|
|
*
|
|
|
|
* Valid values of `isolationLevel` are:
|
|
|
|
*
|
|
|
|
* - Transaction.READ_COMMITTED = 'READ COMMITTED'; // default
|
|
|
|
* - Transaction.READ_UNCOMMITTED = 'READ UNCOMMITTED';
|
|
|
|
* - Transaction.SERIALIZABLE = 'SERIALIZABLE';
|
|
|
|
* - Transaction.REPEATABLE_READ = 'REPEATABLE READ';
|
|
|
|
*
|
|
|
|
* @param {Function} cb Callback function. It calls back with (err, transaction).
|
|
|
|
* To pass the transaction context to one of the CRUD methods, use the `options`
|
|
|
|
* argument with `transaction` property, for example,
|
|
|
|
*
|
|
|
|
* ```js
|
|
|
|
*
|
|
|
|
* MyModel.beginTransaction('READ COMMITTED', function(err, tx) {
|
|
|
|
* MyModel.create({x: 1, y: 'a'}, {transaction: tx}, function(err, inst) {
|
|
|
|
* MyModel.find({x: 1}, {transaction: tx}, function(err, results) {
|
|
|
|
* // ...
|
|
|
|
* tx.commit(function(err) {...});
|
|
|
|
* });
|
|
|
|
* });
|
|
|
|
* });
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* The transaction can be committed or rolled back. If timeout happens, the
|
|
|
|
* transaction will be rolled back. Please note a transaction is typically
|
|
|
|
* associated with a pooled connection. Committing or rolling back a transaction
|
|
|
|
* will release the connection back to the pool.
|
|
|
|
*
|
2015-05-15 17:26:49 +00:00
|
|
|
* Once the transaction is committed or rolled back, the connection property
|
|
|
|
* will be set to null to mark the transaction to be inactive. Trying to commit
|
|
|
|
* or rollback an inactive transaction will receive an error from the callback.
|
|
|
|
*
|
|
|
|
* Please also note that the transaction is only honored with the same data
|
|
|
|
* source/connector instance. CRUD methods will not join the current transaction
|
|
|
|
* if its model is not attached the same data source.
|
|
|
|
*
|
2015-05-13 16:33:49 +00:00
|
|
|
*/
|
|
|
|
TransactionMixin.beginTransaction = function(options, cb) {
|
2015-05-15 17:26:49 +00:00
|
|
|
cb = cb || utils.createPromiseCallback();
|
2015-05-13 16:33:49 +00:00
|
|
|
if (Transaction) {
|
|
|
|
var connector = this.getConnector();
|
2015-05-15 17:26:49 +00:00
|
|
|
Transaction.begin(connector, options, function(err, transaction) {
|
|
|
|
if (err) return cb(err);
|
|
|
|
if (transaction) {
|
|
|
|
// Set an informational transaction id
|
|
|
|
transaction.id = uuid.v1();
|
|
|
|
}
|
|
|
|
if (options.timeout) {
|
|
|
|
setTimeout(function() {
|
|
|
|
var context = {
|
|
|
|
transaction: transaction,
|
2016-04-01 11:48:17 +00:00
|
|
|
operation: 'timeout',
|
2015-05-15 17:26:49 +00:00
|
|
|
};
|
|
|
|
transaction.notifyObserversOf('timeout', context, function(err) {
|
|
|
|
if (!err) {
|
|
|
|
transaction.rollback(function() {
|
|
|
|
debug('Transaction %s is rolled back due to timeout',
|
|
|
|
transaction.id);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}, options.timeout);
|
|
|
|
}
|
|
|
|
cb(err, transaction);
|
|
|
|
});
|
2015-05-13 16:33:49 +00:00
|
|
|
} else {
|
|
|
|
process.nextTick(function() {
|
2016-07-22 19:26:07 +00:00
|
|
|
var err = new Error(g.f('{{Transaction}} is not supported'));
|
2015-05-13 16:33:49 +00:00
|
|
|
cb(err);
|
|
|
|
});
|
|
|
|
}
|
2015-05-15 17:26:49 +00:00
|
|
|
return cb.promise;
|
2015-05-13 16:33:49 +00:00
|
|
|
};
|
|
|
|
|
2015-05-15 17:26:49 +00:00
|
|
|
// Promisify the transaction apis
|
|
|
|
if (Transaction) {
|
|
|
|
jutil.mixin(Transaction.prototype, ObserverMixin);
|
|
|
|
/**
|
|
|
|
* Commit a transaction and release it back to the pool
|
|
|
|
* @param {Function} cb Callback function
|
|
|
|
* @returns {Promise|undefined}
|
|
|
|
*/
|
|
|
|
Transaction.prototype.commit = function(cb) {
|
|
|
|
var self = this;
|
|
|
|
cb = cb || utils.createPromiseCallback();
|
|
|
|
// Report an error if the transaction is not active
|
|
|
|
if (!self.connection) {
|
2015-05-18 19:02:07 +00:00
|
|
|
process.nextTick(function() {
|
2016-07-22 19:26:07 +00:00
|
|
|
cb(new Error(g.f('The {{transaction}} is not active: %s', self.id)));
|
2015-05-15 17:26:49 +00:00
|
|
|
});
|
2015-05-18 19:02:07 +00:00
|
|
|
return cb.promise;
|
2015-05-15 17:26:49 +00:00
|
|
|
}
|
|
|
|
var context = {
|
|
|
|
transaction: self,
|
2016-04-01 11:48:17 +00:00
|
|
|
operation: 'commit',
|
2015-05-15 17:26:49 +00:00
|
|
|
};
|
2015-05-20 22:02:44 +00:00
|
|
|
|
|
|
|
function work(done) {
|
|
|
|
self.connector.commit(self.connection, done);
|
|
|
|
}
|
|
|
|
|
|
|
|
self.notifyObserversAround('commit', context, work, function(err) {
|
|
|
|
// Deference the connection to mark the transaction is not active
|
|
|
|
// The connection should have been released back the pool
|
|
|
|
self.connection = null;
|
|
|
|
cb(err);
|
2015-05-15 17:26:49 +00:00
|
|
|
});
|
2015-05-20 22:02:44 +00:00
|
|
|
|
2015-05-15 17:26:49 +00:00
|
|
|
return cb.promise;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Rollback a transaction and release it back to the pool
|
|
|
|
* @param {Function} cb Callback function
|
|
|
|
* @returns {Promise|undefined}
|
|
|
|
*/
|
|
|
|
Transaction.prototype.rollback = function(cb) {
|
|
|
|
var self = this;
|
|
|
|
cb = cb || utils.createPromiseCallback();
|
|
|
|
// Report an error if the transaction is not active
|
|
|
|
if (!self.connection) {
|
2015-05-18 19:02:07 +00:00
|
|
|
process.nextTick(function() {
|
2016-07-22 19:26:07 +00:00
|
|
|
cb(new Error(g.f('The {{transaction}} is not active: %s', self.id)));
|
2015-05-15 17:26:49 +00:00
|
|
|
});
|
2015-05-18 19:02:07 +00:00
|
|
|
return cb.promise;
|
2015-05-15 17:26:49 +00:00
|
|
|
}
|
|
|
|
var context = {
|
|
|
|
transaction: self,
|
2016-04-01 11:48:17 +00:00
|
|
|
operation: 'rollback',
|
2015-05-15 17:26:49 +00:00
|
|
|
};
|
2015-05-20 22:02:44 +00:00
|
|
|
|
|
|
|
function work(done) {
|
|
|
|
self.connector.rollback(self.connection, done);
|
|
|
|
}
|
|
|
|
|
|
|
|
self.notifyObserversAround('rollback', context, work, function(err) {
|
|
|
|
// Deference the connection to mark the transaction is not active
|
|
|
|
// The connection should have been released back the pool
|
|
|
|
self.connection = null;
|
|
|
|
cb(err);
|
2015-05-15 17:26:49 +00:00
|
|
|
});
|
2015-05-20 22:02:44 +00:00
|
|
|
|
2015-05-15 17:26:49 +00:00
|
|
|
return cb.promise;
|
|
|
|
};
|
|
|
|
|
|
|
|
Transaction.prototype.toJSON = function() {
|
|
|
|
return this.id;
|
|
|
|
};
|
|
|
|
|
|
|
|
Transaction.prototype.toString = function() {
|
|
|
|
return this.id;
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
2015-05-18 16:00:49 +00:00
|
|
|
TransactionMixin.Transaction = Transaction;
|