@toa.io/storages.mongodb 1.0.0-alpha.19 → 1.0.0-alpha.190
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/package.json +8 -8
- package/src/client.js +26 -9
- package/src/factory.js +1 -3
- package/src/record.js +6 -19
- package/src/storage.js +162 -73
- package/src/translate.js +6 -6
- package/test/record.test.js +0 -15
- package/src/collection.js +0 -69
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@toa.io/storages.mongodb",
|
|
3
|
-
"version": "1.0.0-alpha.
|
|
3
|
+
"version": "1.0.0-alpha.190",
|
|
4
4
|
"description": "Toa MongoDB Storage Connector",
|
|
5
5
|
"author": "temich <tema.gurtovoy@gmail.com>",
|
|
6
6
|
"homepage": "https://github.com/toa-io/toa#readme",
|
|
@@ -19,13 +19,13 @@
|
|
|
19
19
|
"test": "echo \"Error: run tests from root\" && exit 1"
|
|
20
20
|
},
|
|
21
21
|
"dependencies": {
|
|
22
|
-
"@toa.io/
|
|
23
|
-
"@toa.io/
|
|
24
|
-
"@toa.io/
|
|
25
|
-
"@toa.io/
|
|
26
|
-
"
|
|
27
|
-
"
|
|
22
|
+
"@toa.io/conveyor": "1.0.0-alpha.173",
|
|
23
|
+
"@toa.io/core": "1.0.0-alpha.190",
|
|
24
|
+
"@toa.io/generic": "1.0.0-alpha.173",
|
|
25
|
+
"@toa.io/pointer": "1.0.0-alpha.182",
|
|
26
|
+
"mongodb": "6.7.0",
|
|
27
|
+
"openspan": "1.0.0-alpha.173",
|
|
28
28
|
"saslprep": "1.0.3"
|
|
29
29
|
},
|
|
30
|
-
"gitHead": "
|
|
30
|
+
"gitHead": "494b1c8f05c21f377f1152a9d89df5cbbd1001de"
|
|
31
31
|
}
|
package/src/client.js
CHANGED
|
@@ -6,6 +6,7 @@
|
|
|
6
6
|
* @typedef {import('@toa.io/core').Locator} Locator
|
|
7
7
|
*/
|
|
8
8
|
|
|
9
|
+
const { console } = require('openspan')
|
|
9
10
|
const { Connector } = require('@toa.io/core')
|
|
10
11
|
const { resolve } = require('@toa.io/pointer')
|
|
11
12
|
const { ID } = require('./deployment')
|
|
@@ -17,6 +18,8 @@ const { MongoClient } = require('mongodb')
|
|
|
17
18
|
const INSTANCES = {}
|
|
18
19
|
|
|
19
20
|
class Client extends Connector {
|
|
21
|
+
name
|
|
22
|
+
|
|
20
23
|
/**
|
|
21
24
|
* @public
|
|
22
25
|
* @type {import('mongodb').Collection}
|
|
@@ -48,6 +51,7 @@ class Client extends Connector {
|
|
|
48
51
|
super()
|
|
49
52
|
|
|
50
53
|
this.locator = locator
|
|
54
|
+
this.name = locator.lowercase
|
|
51
55
|
}
|
|
52
56
|
|
|
53
57
|
/**
|
|
@@ -57,17 +61,30 @@ class Client extends Connector {
|
|
|
57
61
|
*/
|
|
58
62
|
async open () {
|
|
59
63
|
const urls = await this.resolveURLs()
|
|
60
|
-
const
|
|
61
|
-
const collection = this.locator.lowercase
|
|
64
|
+
const dbname = this.resolveDB()
|
|
62
65
|
|
|
63
|
-
this.key = getKey(
|
|
66
|
+
this.key = getKey(dbname, urls)
|
|
64
67
|
|
|
65
|
-
|
|
68
|
+
try {
|
|
69
|
+
INSTANCES[this.key] ??= this.createInstance(urls)
|
|
70
|
+
} catch (error) {
|
|
71
|
+
console.error('Failed to connect to MongoDB', { urls, error })
|
|
72
|
+
}
|
|
66
73
|
|
|
67
74
|
this.instance = await INSTANCES[this.key]
|
|
68
75
|
this.instance.count++
|
|
69
76
|
|
|
70
|
-
|
|
77
|
+
const db = this.instance.client.db(dbname)
|
|
78
|
+
|
|
79
|
+
try {
|
|
80
|
+
this.collection = await db.createCollection(this.name)
|
|
81
|
+
} catch (e) {
|
|
82
|
+
if (e.code !== ALREADY_EXISTS) {
|
|
83
|
+
throw e
|
|
84
|
+
}
|
|
85
|
+
|
|
86
|
+
this.collection = db.collection(this.name)
|
|
87
|
+
}
|
|
71
88
|
}
|
|
72
89
|
|
|
73
90
|
/**
|
|
@@ -95,7 +112,7 @@ class Client extends Connector {
|
|
|
95
112
|
const client = new MongoClient(urls.join(','), OPTIONS)
|
|
96
113
|
const hosts = urls.map((str) => new URL(str).host)
|
|
97
114
|
|
|
98
|
-
console.info('Connecting to MongoDB
|
|
115
|
+
console.info('Connecting to MongoDB', { address: hosts.join(', ') })
|
|
99
116
|
|
|
100
117
|
await client.connect()
|
|
101
118
|
|
|
@@ -139,9 +156,9 @@ function getKey (db, urls) {
|
|
|
139
156
|
}
|
|
140
157
|
|
|
141
158
|
const OPTIONS = {
|
|
142
|
-
ignoreUndefined: true
|
|
143
|
-
connectTimeoutMS: 0,
|
|
144
|
-
serverSelectionTimeoutMS: 0
|
|
159
|
+
ignoreUndefined: true
|
|
145
160
|
}
|
|
146
161
|
|
|
162
|
+
const ALREADY_EXISTS = 48
|
|
163
|
+
|
|
147
164
|
exports.Client = Client
|
package/src/factory.js
CHANGED
|
@@ -1,15 +1,13 @@
|
|
|
1
1
|
'use strict'
|
|
2
2
|
|
|
3
3
|
const { Client } = require('./client')
|
|
4
|
-
const { Collection } = require('./collection')
|
|
5
4
|
const { Storage } = require('./storage')
|
|
6
5
|
|
|
7
6
|
class Factory {
|
|
8
7
|
storage (locator, entity) {
|
|
9
8
|
const client = new Client(locator)
|
|
10
|
-
const connection = new Collection(client)
|
|
11
9
|
|
|
12
|
-
return new Storage(
|
|
10
|
+
return new Storage(client, entity)
|
|
13
11
|
}
|
|
14
12
|
}
|
|
15
13
|
|
package/src/record.js
CHANGED
|
@@ -1,29 +1,16 @@
|
|
|
1
1
|
'use strict'
|
|
2
2
|
|
|
3
|
-
|
|
4
|
-
|
|
5
|
-
* @returns {toa.mongodb.Record}
|
|
6
|
-
*/
|
|
7
|
-
const to = (entity) => {
|
|
8
|
-
const {
|
|
9
|
-
id,
|
|
10
|
-
...rest
|
|
11
|
-
} = entity
|
|
3
|
+
function to (entity) {
|
|
4
|
+
const { id, ...rest } = entity
|
|
12
5
|
|
|
13
6
|
return /** @type {toa.mongodb.Record} */ { _id: id, ...rest }
|
|
14
7
|
}
|
|
15
8
|
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
*/
|
|
20
|
-
const from = (record) => {
|
|
21
|
-
if (record === undefined || record === null) return null
|
|
9
|
+
function from (record) {
|
|
10
|
+
if (record === undefined || record === null)
|
|
11
|
+
return null
|
|
22
12
|
|
|
23
|
-
const {
|
|
24
|
-
_id,
|
|
25
|
-
...rest
|
|
26
|
-
} = record
|
|
13
|
+
const { _id, ...rest } = record
|
|
27
14
|
|
|
28
15
|
return { id: _id, ...rest }
|
|
29
16
|
}
|
package/src/storage.js
CHANGED
|
@@ -1,58 +1,85 @@
|
|
|
1
1
|
'use strict'
|
|
2
2
|
|
|
3
|
-
const {
|
|
4
|
-
|
|
5
|
-
exceptions
|
|
6
|
-
} = require('@toa.io/core')
|
|
7
|
-
|
|
3
|
+
const { Connector, exceptions } = require('@toa.io/core')
|
|
4
|
+
const { console } = require('openspan')
|
|
8
5
|
const { translate } = require('./translate')
|
|
9
|
-
const {
|
|
10
|
-
|
|
11
|
-
from
|
|
12
|
-
} = require('./record')
|
|
6
|
+
const { to, from } = require('./record')
|
|
7
|
+
const { ReturnDocument } = require('mongodb')
|
|
13
8
|
|
|
14
9
|
class Storage extends Connector {
|
|
15
|
-
#
|
|
10
|
+
#client
|
|
11
|
+
|
|
12
|
+
/** @type {import('mongodb').Collection} */
|
|
13
|
+
#collection
|
|
16
14
|
#entity
|
|
17
15
|
|
|
18
|
-
constructor (
|
|
16
|
+
constructor (client, entity) {
|
|
19
17
|
super()
|
|
20
18
|
|
|
21
|
-
this.#
|
|
19
|
+
this.#client = client
|
|
22
20
|
this.#entity = entity
|
|
23
21
|
|
|
24
|
-
this.depends(
|
|
22
|
+
this.depends(client)
|
|
23
|
+
}
|
|
24
|
+
|
|
25
|
+
get raw () {
|
|
26
|
+
return this.#collection
|
|
25
27
|
}
|
|
26
28
|
|
|
27
29
|
async open () {
|
|
30
|
+
this.#collection = this.#client.collection
|
|
31
|
+
|
|
28
32
|
await this.index()
|
|
29
33
|
}
|
|
30
34
|
|
|
31
35
|
async get (query) {
|
|
32
|
-
const {
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
} = translate(query)
|
|
36
|
+
const { criteria, options } = translate(query)
|
|
37
|
+
|
|
38
|
+
this.debug('findOne', { criteria, options })
|
|
36
39
|
|
|
37
|
-
const record = await this.#
|
|
40
|
+
const record = await this.#collection.findOne(criteria, options)
|
|
38
41
|
|
|
39
42
|
return from(record)
|
|
40
43
|
}
|
|
41
44
|
|
|
42
45
|
async find (query) {
|
|
43
|
-
const {
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
46
|
+
const { criteria, options, sample } = translate(query)
|
|
47
|
+
|
|
48
|
+
criteria._deleted = null
|
|
49
|
+
|
|
50
|
+
let cursor
|
|
47
51
|
|
|
48
|
-
|
|
52
|
+
if (sample === undefined) {
|
|
53
|
+
this.debug('find', { criteria, options })
|
|
54
|
+
|
|
55
|
+
cursor = this.#collection.find(criteria, options)
|
|
56
|
+
} else {
|
|
57
|
+
const pipeline = toPipeline(criteria, options, sample)
|
|
58
|
+
|
|
59
|
+
this.debug('aggregate', { pipeline })
|
|
60
|
+
|
|
61
|
+
cursor = this.#collection.aggregate(pipeline)
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
const recordset = await cursor.toArray()
|
|
49
65
|
|
|
50
66
|
return recordset.map((item) => from(item))
|
|
51
67
|
}
|
|
52
68
|
|
|
69
|
+
async stream (query = undefined) {
|
|
70
|
+
const { criteria, options } = translate(query)
|
|
71
|
+
|
|
72
|
+
this.debug('find (stream)', { criteria, options })
|
|
73
|
+
|
|
74
|
+
return this.#collection.find(criteria, options).stream({ transform: from })
|
|
75
|
+
}
|
|
76
|
+
|
|
53
77
|
async add (entity) {
|
|
54
78
|
const record = to(entity)
|
|
55
|
-
|
|
79
|
+
|
|
80
|
+
this.debug('insertOne', { record })
|
|
81
|
+
|
|
82
|
+
const result = await this.#collection.insertOne(record)
|
|
56
83
|
|
|
57
84
|
return result.acknowledged
|
|
58
85
|
}
|
|
@@ -62,41 +89,47 @@ class Storage extends Connector {
|
|
|
62
89
|
_id: entity.id,
|
|
63
90
|
_version: entity._version - 1
|
|
64
91
|
}
|
|
65
|
-
|
|
92
|
+
|
|
93
|
+
const record = to(entity)
|
|
94
|
+
|
|
95
|
+
this.debug('findOneAndReplace', { criteria, record })
|
|
96
|
+
|
|
97
|
+
const result = await this.#collection.findOneAndReplace(criteria, record)
|
|
66
98
|
|
|
67
99
|
return result !== null
|
|
68
100
|
}
|
|
69
101
|
|
|
70
|
-
async store (entity) {
|
|
102
|
+
async store (entity, attempt = 0) {
|
|
71
103
|
try {
|
|
72
|
-
if (entity._version === 1)
|
|
104
|
+
if (entity._version === 1)
|
|
73
105
|
return await this.add(entity)
|
|
74
|
-
|
|
106
|
+
else
|
|
75
107
|
return await this.set(entity)
|
|
76
|
-
}
|
|
77
108
|
} catch (error) {
|
|
78
109
|
if (error.code === ERR_DUPLICATE_KEY) {
|
|
79
|
-
|
|
80
110
|
const id = error.keyPattern === undefined
|
|
81
111
|
? error.message.includes(' index: _id_ ') // AWS DocumentDB
|
|
82
112
|
: error.keyPattern._id === 1
|
|
83
113
|
|
|
84
|
-
if (id)
|
|
114
|
+
if (id)
|
|
85
115
|
return false
|
|
86
|
-
|
|
87
|
-
throw new exceptions.DuplicateException()
|
|
88
|
-
|
|
89
|
-
|
|
116
|
+
else
|
|
117
|
+
throw new exceptions.DuplicateException(this.#client.name, entity)
|
|
118
|
+
} else if (error.cause?.code === 'ECONNREFUSED') {
|
|
119
|
+
// This is temporary and should be replaced with a class decorator.
|
|
120
|
+
if (attempt > 10)
|
|
121
|
+
throw error
|
|
122
|
+
|
|
123
|
+
await new Promise((resolve) => setTimeout(resolve, 1000))
|
|
124
|
+
|
|
125
|
+
return this.store(entity)
|
|
126
|
+
} else
|
|
90
127
|
throw error
|
|
91
|
-
}
|
|
92
128
|
}
|
|
93
129
|
}
|
|
94
130
|
|
|
95
131
|
async upsert (query, changeset) {
|
|
96
|
-
const {
|
|
97
|
-
criteria,
|
|
98
|
-
options
|
|
99
|
-
} = translate(query)
|
|
132
|
+
const { criteria, options } = translate(query)
|
|
100
133
|
|
|
101
134
|
if (!('_deleted' in changeset) || changeset._deleted === null) {
|
|
102
135
|
delete criteria._deleted
|
|
@@ -108,20 +141,43 @@ class Storage extends Connector {
|
|
|
108
141
|
$inc: { _version: 1 }
|
|
109
142
|
}
|
|
110
143
|
|
|
111
|
-
options.returnDocument =
|
|
144
|
+
options.returnDocument = ReturnDocument.AFTER
|
|
145
|
+
|
|
146
|
+
this.debug('findOneAndUpdate', { criteria, update, options })
|
|
112
147
|
|
|
113
|
-
const result = await this.#
|
|
148
|
+
const result = await this.#collection.findOneAndUpdate(criteria, update, options)
|
|
114
149
|
|
|
115
150
|
return from(result)
|
|
116
151
|
}
|
|
117
152
|
|
|
153
|
+
async ensure (query, properties, state) {
|
|
154
|
+
let { criteria, options } = translate(query)
|
|
155
|
+
|
|
156
|
+
if (query === undefined)
|
|
157
|
+
criteria = properties
|
|
158
|
+
|
|
159
|
+
const update = { $setOnInsert: to(state) }
|
|
160
|
+
|
|
161
|
+
options.upsert = true
|
|
162
|
+
options.returnDocument = ReturnDocument.AFTER
|
|
163
|
+
|
|
164
|
+
console.debug('Database query', { collection: this.#collection.name, method: 'findOneAndUpdate', criteria, update, options })
|
|
165
|
+
|
|
166
|
+
const result = await this.#collection.findOneAndUpdate(criteria, update, options)
|
|
167
|
+
|
|
168
|
+
if (result._deleted !== undefined && result._deleted !== null)
|
|
169
|
+
return null
|
|
170
|
+
else
|
|
171
|
+
return from(result)
|
|
172
|
+
}
|
|
173
|
+
|
|
118
174
|
async index () {
|
|
119
175
|
const indexes = []
|
|
120
176
|
|
|
121
177
|
if (this.#entity.unique !== undefined) {
|
|
122
178
|
for (const [name, fields] of Object.entries(this.#entity.unique)) {
|
|
123
|
-
const
|
|
124
|
-
const unique = await this.uniqueIndex(name, fields,
|
|
179
|
+
const optional = this.getOptional(fields)
|
|
180
|
+
const unique = await this.uniqueIndex(name, fields, optional)
|
|
125
181
|
|
|
126
182
|
indexes.push(unique)
|
|
127
183
|
}
|
|
@@ -131,23 +187,24 @@ class Storage extends Connector {
|
|
|
131
187
|
for (const [suffix, declaration] of Object.entries(this.#entity.index)) {
|
|
132
188
|
const name = 'index_' + suffix
|
|
133
189
|
const fields = Object.fromEntries(Object.entries(declaration)
|
|
134
|
-
.map(([name, type]) => [name, INDEX_TYPES[type]]))
|
|
190
|
+
.map(([name, type]) => [name, INDEX_TYPES[type] ?? type]))
|
|
135
191
|
|
|
136
|
-
const
|
|
192
|
+
const optional = this.getOptional(Object.keys(fields))
|
|
193
|
+
const options = { name, sparse: optional.length > 0 }
|
|
137
194
|
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
195
|
+
console.info('Creating index', { fields, options })
|
|
196
|
+
|
|
197
|
+
await this.#collection.createIndex(fields, options)
|
|
198
|
+
.catch((e) => console.warn('MongoDB index creation failed', { collection: this.#collection.name, name, fields, error: e }))
|
|
142
199
|
|
|
143
200
|
indexes.push(name)
|
|
144
201
|
}
|
|
145
202
|
}
|
|
146
203
|
|
|
147
|
-
await this.
|
|
204
|
+
await this.removeObsoleteIndexes(indexes)
|
|
148
205
|
}
|
|
149
206
|
|
|
150
|
-
async uniqueIndex (name, properties,
|
|
207
|
+
async uniqueIndex (name, properties, optional) {
|
|
151
208
|
const fields = properties.reduce((acc, property) => {
|
|
152
209
|
acc[property] = 1
|
|
153
210
|
return acc
|
|
@@ -155,48 +212,80 @@ class Storage extends Connector {
|
|
|
155
212
|
|
|
156
213
|
name = 'unique_' + name
|
|
157
214
|
|
|
158
|
-
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
|
|
215
|
+
const options = { name, unique: true }
|
|
216
|
+
|
|
217
|
+
if (optional.length > 0)
|
|
218
|
+
options.partialFilterExpression = Object.fromEntries(optional.map((field) => [field, { $exists: true }]))
|
|
219
|
+
|
|
220
|
+
console.info('Creating unique index', { name, fields, options })
|
|
221
|
+
|
|
222
|
+
await this.#collection.createIndex(fields, options)
|
|
223
|
+
.catch((e) => console.warn('MongoDB unique index creation failed',
|
|
224
|
+
{ collection: this.#collection.name, name, fields, error: e }))
|
|
163
225
|
|
|
164
226
|
return name
|
|
165
227
|
}
|
|
166
228
|
|
|
167
|
-
async
|
|
168
|
-
const current = await this
|
|
229
|
+
async removeObsoleteIndexes (desired) {
|
|
230
|
+
const current = await this.getCurrentIndexes()
|
|
169
231
|
const obsolete = current.filter((name) => !desired.includes(name))
|
|
170
232
|
|
|
171
233
|
if (obsolete.length > 0) {
|
|
172
|
-
console.info(
|
|
234
|
+
console.info('Removing obsolete indexes', { collection: this.#collection.name, indexes: obsolete.join(', ') })
|
|
235
|
+
|
|
236
|
+
await Promise.all(obsolete.map((name) => this.#collection.dropIndex(name)))
|
|
237
|
+
}
|
|
238
|
+
}
|
|
239
|
+
|
|
240
|
+
async getCurrentIndexes () {
|
|
241
|
+
try {
|
|
242
|
+
const array = await this.#collection.listIndexes().toArray()
|
|
173
243
|
|
|
174
|
-
|
|
244
|
+
return array.map(({ name }) => name).filter((name) => name !== '_id_')
|
|
245
|
+
} catch {
|
|
246
|
+
return []
|
|
175
247
|
}
|
|
176
248
|
}
|
|
177
249
|
|
|
178
|
-
|
|
250
|
+
getOptional (fields) {
|
|
179
251
|
const optional = []
|
|
180
252
|
|
|
181
|
-
for (const field of fields) {
|
|
182
|
-
if (!(field in this.#entity.schema.properties))
|
|
253
|
+
for (const field of fields) {
|
|
254
|
+
if (!field.includes('.') && !(field in this.#entity.schema.properties))
|
|
183
255
|
throw new Error(`Index field '${field}' is not defined.`)
|
|
184
|
-
}
|
|
185
256
|
|
|
186
|
-
if (!this.#entity.schema.required?.includes(field))
|
|
257
|
+
if (!this.#entity.schema.required?.includes(field))
|
|
187
258
|
optional.push(field)
|
|
188
|
-
}
|
|
189
259
|
}
|
|
190
260
|
|
|
191
|
-
|
|
192
|
-
|
|
261
|
+
return optional
|
|
262
|
+
}
|
|
193
263
|
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
197
|
-
|
|
264
|
+
debug (method, attributes) {
|
|
265
|
+
console.debug('Database query', {
|
|
266
|
+
collection: this.#collection.name,
|
|
267
|
+
method,
|
|
268
|
+
...attributes
|
|
269
|
+
})
|
|
198
270
|
}
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
function toPipeline (criteria, options, sample) {
|
|
274
|
+
const pipeline = []
|
|
275
|
+
|
|
276
|
+
if (criteria !== undefined)
|
|
277
|
+
pipeline.push({ $match: criteria })
|
|
278
|
+
|
|
279
|
+
if (sample !== undefined)
|
|
280
|
+
pipeline.push({ $sample: { size: sample } })
|
|
281
|
+
|
|
282
|
+
if (options?.sort !== undefined)
|
|
283
|
+
pipeline.push({ $sort: options.sort })
|
|
284
|
+
|
|
285
|
+
if (options?.projection !== undefined)
|
|
286
|
+
pipeline.push({ $project: options.projection })
|
|
199
287
|
|
|
288
|
+
return pipeline
|
|
200
289
|
}
|
|
201
290
|
|
|
202
291
|
const INDEX_TYPES = {
|
package/src/translate.js
CHANGED
|
@@ -9,18 +9,18 @@ const parse = { ...require('./translate/criteria'), ...require('./translate/opti
|
|
|
9
9
|
const translate = (query) => {
|
|
10
10
|
const result = {
|
|
11
11
|
criteria: query?.criteria === undefined ? {} : parse.criteria(query.criteria),
|
|
12
|
-
options: query?.options === undefined ? {} : parse.options(query.options)
|
|
12
|
+
options: query?.options === undefined ? {} : parse.options(query.options),
|
|
13
|
+
sample: query?.options?.sample
|
|
13
14
|
}
|
|
14
15
|
|
|
15
|
-
if (query?.id !== undefined)
|
|
16
|
+
if (query?.id !== undefined)
|
|
16
17
|
result.criteria._id = query.id
|
|
17
|
-
}
|
|
18
18
|
|
|
19
|
-
if (query?.version !== undefined)
|
|
19
|
+
if (query?.version !== undefined)
|
|
20
20
|
result.criteria._version = query.version
|
|
21
|
-
}
|
|
22
21
|
|
|
23
|
-
|
|
22
|
+
if (query?.search !== undefined)
|
|
23
|
+
result.criteria.$text = { $search: query.search }
|
|
24
24
|
|
|
25
25
|
return result
|
|
26
26
|
}
|
package/test/record.test.js
CHANGED
|
@@ -47,19 +47,4 @@ describe('from', () => {
|
|
|
47
47
|
_version: 0
|
|
48
48
|
})
|
|
49
49
|
})
|
|
50
|
-
|
|
51
|
-
it('should not modify argument', () => {
|
|
52
|
-
/** @type {toa.mongodb.Record} */
|
|
53
|
-
const record = {
|
|
54
|
-
_id: '1',
|
|
55
|
-
_version: 0
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
from(record)
|
|
59
|
-
|
|
60
|
-
expect(record).toStrictEqual({
|
|
61
|
-
_id: '1',
|
|
62
|
-
_version: 0
|
|
63
|
-
})
|
|
64
|
-
})
|
|
65
50
|
})
|
package/src/collection.js
DELETED
|
@@ -1,69 +0,0 @@
|
|
|
1
|
-
'use strict'
|
|
2
|
-
|
|
3
|
-
const { Connector } = require('@toa.io/core')
|
|
4
|
-
|
|
5
|
-
class Collection extends Connector {
|
|
6
|
-
#client
|
|
7
|
-
#collection
|
|
8
|
-
|
|
9
|
-
constructor (client) {
|
|
10
|
-
super()
|
|
11
|
-
|
|
12
|
-
this.#client = client
|
|
13
|
-
|
|
14
|
-
this.depends(client)
|
|
15
|
-
}
|
|
16
|
-
|
|
17
|
-
async open () {
|
|
18
|
-
this.#collection = this.#client.collection
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
/** @hot */
|
|
22
|
-
async get (query, options) {
|
|
23
|
-
return /** @type {toa.mongodb.Record} */ this.#collection.findOne(query, options)
|
|
24
|
-
}
|
|
25
|
-
|
|
26
|
-
/** @hot */
|
|
27
|
-
async find (query, options) {
|
|
28
|
-
const cursor = this.#collection.find(query, options)
|
|
29
|
-
|
|
30
|
-
return cursor.toArray()
|
|
31
|
-
}
|
|
32
|
-
|
|
33
|
-
/** @hot */
|
|
34
|
-
async add (record) {
|
|
35
|
-
return await this.#collection.insertOne(record)
|
|
36
|
-
}
|
|
37
|
-
|
|
38
|
-
/** @hot */
|
|
39
|
-
async replace (query, record, options) {
|
|
40
|
-
return await this.#collection.findOneAndReplace(query, record, options)
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
/** @hot */
|
|
44
|
-
async update (query, update, options) {
|
|
45
|
-
return this.#collection.findOneAndUpdate(query, update, options)
|
|
46
|
-
}
|
|
47
|
-
|
|
48
|
-
async index (keys, options) {
|
|
49
|
-
return this.#collection.createIndex(keys, options)
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
async indexes () {
|
|
53
|
-
try {
|
|
54
|
-
const array = await this.#collection.listIndexes().toArray()
|
|
55
|
-
|
|
56
|
-
return array.map(({ name }) => name).filter((name) => name !== '_id_')
|
|
57
|
-
} catch {
|
|
58
|
-
return []
|
|
59
|
-
}
|
|
60
|
-
}
|
|
61
|
-
|
|
62
|
-
async dropIndexes (names) {
|
|
63
|
-
const all = names.map((name) => this.#collection.dropIndex(name))
|
|
64
|
-
|
|
65
|
-
return Promise.all(all)
|
|
66
|
-
}
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
exports.Collection = Collection
|