-
Notifications
You must be signed in to change notification settings - Fork 0
Expand file tree
/
Copy pathindex.js
More file actions
126 lines (115 loc) · 3.09 KB
/
index.js
File metadata and controls
126 lines (115 loc) · 3.09 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
const { map } = require('lodash');
const Knex = require('./lib/index');
const Runner = require('./lib/runner');
const CLIENTS = {
mariadb: 'mysql2',
mssql: 'mssql',
mysql: 'mysql2',
postgres: 'postgres',
sqlite: 'sqlite3'
};
module.exports = function withSequelize(sequelize, config = {}) {
const dialect = sequelize.getDialect();
const Dialect = require(`./lib/dialects/${CLIENTS[dialect]}/index.js`);
function getType(method) {
switch (method) {
case 'select':
case 'pluck':
case 'first':
return sequelize.QueryTypes.SELECT;
case 'insert':
return sequelize.QueryTypes.INSERT;
case 'del':
return sequelize.QueryTypes.BULKDELETE;
case 'update':
return sequelize.QueryTypes.BULKUPDATE;
default:
return sequelize.QueryTypes.RAW;
}
}
class SequelizeRunner extends Runner {
run() {
const sql = this.builder.toSQL();
if (Array.isArray(sql)) {
return this.queryArray(sql);
}
return this.query(sql);
}
async query(obj) {
const options = {
replacements: obj.bindings,
...obj.options
};
const { method } = obj;
if (!options.type) {
options.type = getType(method);
}
// run query
const data = await sequelize.query(obj.sql, options);
const isRaw = options.type === sequelize.QueryTypes.RAW;
// format data
let resp = data;
if (isRaw) {
resp = dialect === 'postgres' ? data[1] : data[0];
}
// run output
if (obj.output) {
return obj.output.call(this, resp);
}
// format response
switch (method) {
case 'select':
case 'pluck':
case 'first': {
const rows = isRaw ? data[0] : data;
if (method === 'pluck') {
return map(rows, obj.pluck);
}
return method === 'first' ? rows[0] : rows;
}
default:
return resp;
}
}
}
async function transaction(container) {
const tx = await sequelize.transaction();
const instance = Knex({ client: Client });
instance.context.queryBuilder = function() {
return this.client.queryBuilder().options({ transaction: tx });
};
instance.commit = () => tx.commit();
instance.rollback = () => tx.rollback();
const result = container(instance);
if (result && typeof result.then === 'function') {
try {
const resp = await result;
await tx.commit();
return resp;
} catch (err) {
await tx.rollback();
throw err;
}
}
return instance;
};
class Client extends Dialect {
constructor(cfg) {
cfg.useNullAsDefault = config.useNullAsDefault;
super(cfg);
this.connectionSettings = config.connection || {};
}
transaction(container) {
return transaction(container);
}
runner(builder) {
return new SequelizeRunner(this, builder);
}
database() {
return sequelize.getDatabaseName();
}
}
const knex = Knex({ client: Client });
return knex;
};
module.exports.Knex = Knex;