knex/src/dialects/mssql/transaction.js

93 lines
2.5 KiB
JavaScript
Raw Normal View History

import Promise from 'bluebird';
import Transaction from '../../transaction';
import { isUndefined } from 'lodash'
const debug = require('debug')('knex:tx')
2016-03-02 17:07:05 +01:00
2016-09-12 18:45:35 -04:00
export default class Transaction_MSSQL extends Transaction {
2016-03-02 17:07:05 +01:00
begin(conn) {
2016-03-02 17:07:05 +01:00
debug('%s: begin', this.txid)
return conn.tx_.begin()
.then(this._resolver, this._rejecter)
2016-09-12 18:45:35 -04:00
}
2016-03-02 17:07:05 +01:00
savepoint(conn) {
2016-03-02 17:07:05 +01:00
debug('%s: savepoint at', this.txid)
return Promise.resolve()
.then(() => this.query(conn, `SAVE TRANSACTION ${this.txid}`))
2016-09-12 18:45:35 -04:00
}
2016-03-02 17:07:05 +01:00
commit(conn, value) {
2016-03-02 17:07:05 +01:00
this._completed = true
debug('%s: commit', this.txid)
return conn.tx_.commit()
.then(() => this._resolver(value), this._rejecter)
2016-09-12 18:45:35 -04:00
}
2016-03-02 17:07:05 +01:00
release(conn, value) {
2016-03-02 17:07:05 +01:00
return this._resolver(value)
2016-09-12 18:45:35 -04:00
}
2016-03-02 17:07:05 +01:00
rollback(conn, error) {
2016-03-02 17:07:05 +01:00
this._completed = true
debug('%s: rolling back', this.txid)
return conn.tx_.rollback()
.then(
() => {
let err = error;
if(isUndefined(error)) {
err = new Error(`Transaction rejected with non-error: ${error}`)
}
this._rejecter(err)
},
err => {
if (error) err.originalError = error;
return this._rejecter(err);
}
)
2016-09-12 18:45:35 -04:00
}
2016-03-02 17:07:05 +01:00
rollbackTo(conn, error) {
2016-03-02 17:07:05 +01:00
debug('%s: rolling backTo', this.txid)
return Promise.resolve()
.then(() => this.query(conn, `ROLLBACK TRANSACTION ${this.txid}`, 2, error))
2016-03-02 17:07:05 +01:00
.then(() => this._rejecter(error))
2016-09-12 18:45:35 -04:00
}
2016-03-02 17:07:05 +01:00
// Acquire a connection and create a disposer - either using the one passed
// via config or getting one off the client. The disposer will be called once
// the original promise is marked completed.
acquireConnection(config) {
const t = this
const configConnection = config && config.connection
return Promise.try(() => {
return (t.outerTx ? t.outerTx.conn : null) ||
configConnection ||
2016-10-09 14:00:55 -04:00
t.client.acquireConnection();
2016-03-02 17:07:05 +01:00
}).tap(function(conn) {
conn.__knexTxId = t.txid;
2016-03-02 17:07:05 +01:00
if (!t.outerTx) {
t.conn = conn
conn.tx_ = conn.transaction()
}
}).disposer(function(conn) {
if (t.outerTx) return;
if (conn.tx_) {
if (!t._completed) {
debug('%s: unreleased transaction', t.txid)
conn.tx_.rollback();
}
conn.tx_ = null;
}
t.conn = null
if (!configConnection) {
debug('%s: releasing connection', t.txid)
t.client.releaseConnection(conn)
} else {
debug('%s: not releasing external connection', t.txid)
}
})
}
2016-09-12 18:45:35 -04:00
}