-
-
Notifications
You must be signed in to change notification settings - Fork 1.3k
/
Copy pathindex.ts
124 lines (102 loc) · 2.72 KB
/
index.ts
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
import { Submittable, Connection, QueryResult } from 'pg'
const Result = require('pg/lib/result.js')
const EventEmitter = require('events').EventEmitter
let nextUniqueID = 1 // concept borrowed from org.postgresql.core.v3.QueryExecutorImpl
interface BatchQueryConfig {
name?: string
text: string
values?: string[][]
}
class BatchQuery implements Submittable {
name: string | null
text: string
values: string[][]
connection: Connection | null
_portal: string | null
_result: typeof Result | null
_results: typeof Result[]
callback: Function | null
_canceledDueToError: Boolean
public constructor(batchQuery: BatchQueryConfig) {
const { name, values, text } = batchQuery
this.name = name
this.values = values
this.text = text
this.connection = null
this._portal = null
this._result = new Result()
this._results = []
this.callback = null
this._canceledDueToError = false
for (const row of values) {
if (!Array.isArray(values)) {
throw new Error('Batch commands require each set of values to be an array. e.g. values: any[][]')
}
}
}
public submit(connection: Connection): void {
this.connection = connection
// creates a named prepared statement
this.connection.parse(
{
text: this.text,
name: this.name,
types: []
},
true
)
this.values.map(val => {
this._portal = 'C_' + nextUniqueID++
this.connection.bind({
statement: this.name,
values: val,
portal: this._portal
}, true)
// maybe we could avoid this for non-select queries
this.connection.describe({
type: 'P',
name: this._portal,
}, true)
this.connection.execute({portal: this._portal}, true)
})
this.connection.sync()
}
execute(): Promise<QueryResult[]> {
return new Promise((resolve, reject) => {
this.callback = (err, rows) => (err ? reject(err) : resolve(rows))
})
}
handleError(err, connection) {
console.log(err)
this.connection.sync()
}
handleReadyForQuery(con) {
if (this.callback) {
try {
this.callback(null, this._results)
}
catch(err) {
process.nextTick(() => {
throw err
})
}
}
}
handleRowDescription(msg) {
this._result.addFields(msg.fields)
}
handleDataRow(msg) {
const row = this._result.parseRow(msg.fields)
this._result.addRow(row)
}
handleCommandComplete(msg) {
this._result.addCommandComplete(msg)
this._results.push(this._result)
this._result = new Result()
this.connection.close({ type: 'P', name: this._portal }, true)
}
handleEmptyQuery() {
this.connection.sync()
}
}
export = BatchQuery