-
Notifications
You must be signed in to change notification settings - Fork 27
/
Copy pathindex.js
195 lines (173 loc) · 5.69 KB
/
index.js
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
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
'use strict'
const wrappers = require('pouchdb-wrappers')
// determine if a document key is an internal field like _rev
function isntInternalKey (key) {
return key[0] !== '_'
}
// determine if a document should be transformed
function isUntransformable (doc) {
// do not transform local documents
if (typeof doc._id === 'string' && (/^_local/).test(doc._id)) {
return true
}
// do not transform document tombstones
if (doc._deleted) {
return Object.keys(doc).filter(isntInternalKey).length === 0
}
return false
}
module.exports = {
transform,
// api.filter provided for backwards compat with the old "filter-pouch"
filter: transform
}
function transform (config) {
const db = this
// create incoming handler, which transforms documents before write
const incoming = function (doc) {
if (!isUntransformable(doc) && config.incoming) {
return config.incoming(doc)
}
return doc
}
// create outgoing handler, which transforms documents after read
const outgoing = function (doc) {
if (!isUntransformable(doc) && config.outgoing) {
return config.outgoing(doc)
}
return doc
}
const handlers = {
async get (orig, ...args) {
const response = await orig(...args)
if (Array.isArray(response)) {
// open_revs style, it's a list of docs
await Promise.all(response.map(async (row) => {
if (row.ok) {
row.ok = await outgoing(row.ok)
}
}))
return response
} else {
// response is just one doc
return outgoing(response)
}
},
async bulkDocs (orig, docs, ...args) {
if (docs.docs) {
// docs can be an object and not just a list
docs.docs = await Promise.all(docs.docs.map(incoming))
} else {
// docs is just a list
docs = await Promise.all(docs.map(incoming))
}
return orig(docs, ...args)
},
async allDocs (orig, ...args) {
const response = await orig(...args)
await Promise.all(response.rows.map(async (row) => {
// run docs through outgoing handler if include_docs was true
if (row.doc) {
row.doc = await outgoing(row.doc)
}
}))
return response
},
async bulkGet (orig, ...args) {
const mapDoc = async (doc) => {
// only run the outgoing handler if the doc exists ("ok")
// istanbul ignore else
if (doc.ok) {
return { ok: await outgoing(doc.ok) }
} else {
return doc
}
}
const mapResult = async (result) => {
const { id, docs } = result
// istanbul ignore else
if (id && docs && Array.isArray(docs)) {
// only modify docs if everything looks ok
return { id, docs: await Promise.all(docs.map(mapDoc)) }
} else {
// result wasn't ok so we return it unmodified
return result
}
}
let { results, ...res } = await orig(...args)
results = await Promise.all(results.map(mapResult))
return { results, ...res }
},
changes (orig, ...args) {
async function modifyChange (change) {
// transform a change only if it includes a doc
if (change.doc) {
change.doc = await outgoing(change.doc)
return change
}
return change
}
async function modifyChanges (res) {
// transform the response only if it contains results
if (res.results) {
res.results = await Promise.all(res.results.map(modifyChange))
return res
}
return res
}
const changes = orig(...args)
const { on: origOn, then: origThen } = changes
return Object.assign(changes, {
// wrap all listeners, but specifically those for 'change' and 'complete'
on (event, listener) {
const origListener = listener
if (event === 'change') {
listener = async (change) => {
origListener(await modifyChange(change))
}
} else if (event === 'complete') {
// the 'complete' event returns all relevant changes,
// so we submit them all for transformation
listener = async (res) => {
origListener(await modifyChanges(res))
}
}
return origOn.call(changes, event, listener)
},
// `.changes` can be awaited. it then returns all relevant changes
// which we pass to our handler for possible transformation
then (resolve, reject) {
return origThen.call(changes, modifyChanges).then(resolve, reject)
}
})
}
}
if (db.type() === 'http') {
// when using its http adapter, pouchdb uses the adapter's `._put` method,
// rather than `._bulkDocs`,
// so we have to wrap `.put` in addition to `.bulkDocs`.
handlers.put = async function (orig, doc, ...args) {
doc = await incoming(doc)
return orig(doc, ...args)
}
// when using its http adapter, pouchdb cannot intercept query results with `.get`
// so we must wrap the `.query` method directly to transform query results.
handlers.query = async function (orig, ...args) {
const response = await orig(...args)
await Promise.all(response.rows.map(async (row) => {
// modify result rows if they contain a doc
if (row.doc) {
row.doc = await outgoing(row.doc)
}
// because js passes objects by reference,
// there is no need to return anything after updating the row object.
}))
return response
}
}
wrappers.install(db, handlers)
}
/* istanbul ignore next */
if (typeof window !== 'undefined' && window.PouchDB) {
window.PouchDB.plugin(exports)
}