Skip to content

Commit 77dac64

Browse files
committed
1 parent c94f9a4 commit 77dac64

File tree

9 files changed

+73
-100
lines changed

9 files changed

+73
-100
lines changed

.github/workflows/sauce.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@ jobs:
1212
- name: Set up node
1313
uses: actions/setup-node@v3
1414
with:
15-
node-version: 14
15+
node-version: 16
1616
- name: Install
1717
run: npm install
1818
env:

.github/workflows/test.yml

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ jobs:
77
runs-on: ubuntu-latest
88
strategy:
99
matrix:
10-
node: [12, 14, 16]
10+
node: [16, 18, 20]
1111
name: Node ${{ matrix.node }}
1212
steps:
1313
- name: Checkout

README.md

-13
Original file line numberDiff line numberDiff line change
@@ -38,19 +38,6 @@ for await (const [key, value] of db.iterator({ gt: 'a' })) {
3838
}
3939
```
4040

41-
With callbacks:
42-
43-
```js
44-
db.put('example', { hello: 'world' }, (err) => {
45-
if (err) throw err
46-
47-
db.get('example', (err, value) => {
48-
if (err) throw err
49-
console.log(value) // { hello: 'world' }
50-
})
51-
})
52-
```
53-
5441
<!-- ## Browser support
5542
5643
[![Sauce Test Status](https://app.saucelabs.com/browser-matrix/level-ci.svg)](https://app.saucelabs.com/u/level-ci) -->

UPGRADING.md

+4
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,10 @@
22

33
This document describes breaking changes and how to upgrade. For a complete list of changes including minor and patch releases, please refer to the [`CHANGELOG`](CHANGELOG.md).
44

5+
## 2.0.0
6+
7+
This release upgrades to `abstract-level` 2.0.0 which adds [hooks](https://github.com/Level/abstract-level#hooks) and drops callbacks, not-found errors and support of Node.js < 16. Please refer to the [upgrade guide of `abstract-level`](https://github.com/Level/abstract-level/blob/v2.0.0/UPGRADING.md).
8+
59
## 1.0.0
610

711
**Introducing `memory-level`: a fork of [`memdown`](https://github.com/Level/memdown) that removes the need for [`level-mem`](https://github.com/Level/mem), [`levelup`](https://github.com/Level/levelup) and more. It implements the [`abstract-level`](https://github.com/Level/abstract-level) interface instead of [`abstract-leveldown`](https://github.com/Level/abstract-leveldown) and thus has the same API as `level-mem` and `levelup` including encodings, promises and events. In addition, you can now choose to use Uint8Array instead of Buffer. Sublevels are builtin.**

index.d.ts

+1-4
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,7 @@
11
import {
22
AbstractLevel,
33
AbstractDatabaseOptions,
4-
AbstractOpenOptions,
5-
NodeCallback
4+
AbstractOpenOptions
65
} from 'abstract-level'
76

87
/**
@@ -22,8 +21,6 @@ export class MemoryLevel<KDefault = string, VDefault = string>
2221

2322
open (): Promise<void>
2423
open (options: OpenOptions): Promise<void>
25-
open (callback: NodeCallback<void>): void
26-
open (options: OpenOptions, callback: NodeCallback<void>): void
2724
}
2825

2926
/**

index.js

+52-60
Original file line numberDiff line numberDiff line change
@@ -62,19 +62,19 @@ class MemoryIterator extends AbstractIterator {
6262
this[kInit](db[kTree], options)
6363
}
6464

65-
_next (callback) {
66-
if (!this[kIterator].valid) return this.nextTick(callback)
65+
async _next () {
66+
if (!this[kIterator].valid) return undefined
6767

6868
const key = this[kIterator].key
6969
const value = this[kIterator].value
7070

71-
if (!this[kTest](key)) return this.nextTick(callback)
71+
if (!this[kTest](key)) return undefined
7272

7373
this[kIterator][this[kAdvance]]()
74-
this.nextTick(callback, null, key, value)
74+
return [key, value]
7575
}
7676

77-
_nextv (size, options, callback) {
77+
async _nextv (size, options) {
7878
const it = this[kIterator]
7979
const entries = []
8080

@@ -83,10 +83,10 @@ class MemoryIterator extends AbstractIterator {
8383
it[this[kAdvance]]()
8484
}
8585

86-
this.nextTick(callback, null, entries)
86+
return entries
8787
}
8888

89-
_all (options, callback) {
89+
async _all (options) {
9090
const size = this.limit - this.count
9191
const it = this[kIterator]
9292
const entries = []
@@ -96,7 +96,7 @@ class MemoryIterator extends AbstractIterator {
9696
it[this[kAdvance]]()
9797
}
9898

99-
this.nextTick(callback, null, entries)
99+
return entries
100100
}
101101
}
102102

@@ -106,17 +106,17 @@ class MemoryKeyIterator extends AbstractKeyIterator {
106106
this[kInit](db[kTree], options)
107107
}
108108

109-
_next (callback) {
110-
if (!this[kIterator].valid) return this.nextTick(callback)
109+
async _next () {
110+
if (!this[kIterator].valid) return undefined
111111

112112
const key = this[kIterator].key
113-
if (!this[kTest](key)) return this.nextTick(callback)
113+
if (!this[kTest](key)) return undefined
114114

115115
this[kIterator][this[kAdvance]]()
116-
this.nextTick(callback, null, key)
116+
return key
117117
}
118118

119-
_nextv (size, options, callback) {
119+
async _nextv (size, options) {
120120
const it = this[kIterator]
121121
const keys = []
122122

@@ -125,10 +125,10 @@ class MemoryKeyIterator extends AbstractKeyIterator {
125125
it[this[kAdvance]]()
126126
}
127127

128-
this.nextTick(callback, null, keys)
128+
return keys
129129
}
130130

131-
_all (options, callback) {
131+
async _all (options) {
132132
const size = this.limit - this.count
133133
const it = this[kIterator]
134134
const keys = []
@@ -138,7 +138,7 @@ class MemoryKeyIterator extends AbstractKeyIterator {
138138
it[this[kAdvance]]()
139139
}
140140

141-
this.nextTick(callback, null, keys)
141+
return keys
142142
}
143143
}
144144

@@ -148,19 +148,19 @@ class MemoryValueIterator extends AbstractValueIterator {
148148
this[kInit](db[kTree], options)
149149
}
150150

151-
_next (callback) {
152-
if (!this[kIterator].valid) return this.nextTick(callback)
151+
async _next (options) {
152+
if (!this[kIterator].valid) return undefined
153153

154154
const key = this[kIterator].key
155155
const value = this[kIterator].value
156156

157-
if (!this[kTest](key)) return this.nextTick(callback)
157+
if (!this[kTest](key)) return undefined
158158

159159
this[kIterator][this[kAdvance]]()
160-
this.nextTick(callback, null, value)
160+
return value
161161
}
162162

163-
_nextv (size, options, callback) {
163+
async _nextv (size, options) {
164164
const it = this[kIterator]
165165
const values = []
166166

@@ -169,10 +169,10 @@ class MemoryValueIterator extends AbstractValueIterator {
169169
it[this[kAdvance]]()
170170
}
171171

172-
this.nextTick(callback, null, values)
172+
return values
173173
}
174174

175-
_all (options, callback) {
175+
async _all (options) {
176176
const size = this.limit - this.count
177177
const it = this[kIterator]
178178
const values = []
@@ -182,7 +182,7 @@ class MemoryValueIterator extends AbstractValueIterator {
182182
it[this[kAdvance]]()
183183
}
184184

185-
this.nextTick(callback, null, values)
185+
return values
186186
}
187187
}
188188

@@ -270,6 +270,7 @@ class MemoryLevel extends AbstractLevel {
270270
}
271271

272272
// To help migrating from level-mem to abstract-level
273+
// TODO (v2): remove
273274
if (typeof location === 'function' || typeof options === 'function' || typeof _ === 'function') {
274275
throw new ModuleError('The levelup-style callback argument has been removed', {
275276
code: 'LEVEL_LEGACY'
@@ -291,45 +292,40 @@ class MemoryLevel extends AbstractLevel {
291292
permanence: false,
292293
createIfMissing: false,
293294
errorIfExists: false,
294-
encodings: { [storeEncoding]: true }
295+
encodings: { [storeEncoding]: true },
296+
signals: {
297+
// Would have no value here because the operations are synchronous
298+
iterators: false
299+
}
295300
}, forward)
296301

297302
this[kTree] = createRBT(compare)
298303
}
299304

300-
_put (key, value, options, callback) {
305+
async _put (key, value, options) {
301306
const it = this[kTree].find(key)
302307

303308
if (it.valid) {
304309
this[kTree] = it.update(value)
305310
} else {
306311
this[kTree] = this[kTree].insert(key, value)
307312
}
308-
309-
this.nextTick(callback)
310313
}
311314

312-
_get (key, options, callback) {
313-
const value = this[kTree].get(key)
314-
315-
if (typeof value === 'undefined') {
316-
// TODO: use error code (not urgent, abstract-level normalizes this)
317-
return this.nextTick(callback, new Error('NotFound'))
318-
}
319-
320-
this.nextTick(callback, null, value)
315+
async _get (key, options) {
316+
// Is undefined if not found
317+
return this[kTree].get(key)
321318
}
322319

323-
_getMany (keys, options, callback) {
324-
this.nextTick(callback, null, keys.map(key => this[kTree].get(key)))
320+
async _getMany (keys, options) {
321+
return keys.map(key => this[kTree].get(key))
325322
}
326323

327-
_del (key, options, callback) {
324+
async _del (key, options) {
328325
this[kTree] = this[kTree].remove(key)
329-
this.nextTick(callback)
330326
}
331327

332-
_batch (operations, options, callback) {
328+
async _batch (operations, options) {
333329
let tree = this[kTree]
334330

335331
for (const op of operations) {
@@ -344,38 +340,35 @@ class MemoryLevel extends AbstractLevel {
344340
}
345341

346342
this[kTree] = tree
347-
this.nextTick(callback)
348343
}
349344

350-
_clear (options, callback) {
345+
async _clear (options) {
351346
if (options.limit === -1 && !Object.keys(options).some(isRangeOption)) {
352347
// Delete everything by creating a new empty tree.
353348
this[kTree] = createRBT(compare)
354-
return this.nextTick(callback)
349+
return
355350
}
356351

357352
const iterator = this._keys({ ...options })
358353
const limit = iterator.limit
359354

360355
let count = 0
361356

362-
const loop = () => {
357+
while (true) {
363358
// TODO: add option to control "batch size"
364359
for (let i = 0; i < 500; i++) {
365-
if (++count > limit) return callback()
366-
if (!iterator[kIterator].valid) return callback()
367-
if (!iterator[kTest](iterator[kIterator].key)) return callback()
360+
if (++count > limit) return
361+
if (!iterator[kIterator].valid) return
362+
if (!iterator[kTest](iterator[kIterator].key)) return
368363

369364
// Must also include changes made in parallel to clear()
370365
this[kTree] = this[kTree].remove(iterator[kIterator].key)
371366
iterator[kIterator][iterator[kAdvance]]()
372367
}
373368

374369
// Some time to breathe
375-
this.nextTick(loop)
370+
await breathe()
376371
}
377-
378-
this.nextTick(loop)
379372
}
380373

381374
_iterator (options) {
@@ -393,18 +386,17 @@ class MemoryLevel extends AbstractLevel {
393386

394387
exports.MemoryLevel = MemoryLevel
395388

396-
// Use setImmediate() in Node.js to allow IO in between our callbacks
389+
let breathe
390+
391+
// Use setImmediate() in Node.js to allow IO in between work
397392
if (typeof process !== 'undefined' && !process.browser && typeof global !== 'undefined' && typeof global.setImmediate === 'function') {
398393
const setImmediate = global.setImmediate
399394

400-
// Automatically applies to iterators, sublevels and chained batches as well
401-
MemoryLevel.prototype.nextTick = function (fn, ...args) {
402-
if (args.length === 0) {
403-
setImmediate(fn)
404-
} else {
405-
setImmediate(() => fn(...args))
406-
}
395+
breathe = function () {
396+
return new Promise(setImmediate)
407397
}
398+
} else {
399+
breathe = async function () {}
408400
}
409401

410402
function isRangeOption (k) {

package.json

+7-6
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,8 @@
77
"main": "index.js",
88
"types": "./index.d.ts",
99
"scripts": {
10-
"test": "standard && hallmark && (nyc -s node test.js | faucet) && nyc report",
10+
"test": "standard && hallmark && (nyc -s node test.js | tap-arc) && nyc report",
11+
"test-pessimistic": "node test.js | tap-arc -pv",
1112
"test-browsers": "airtap --coverage --verbose test.js",
1213
"test-browsers-local": "airtap --coverage -p local test.js",
1314
"coverage": "nyc report -r lcovonly"
@@ -19,22 +20,22 @@
1920
"CHANGELOG.md"
2021
],
2122
"dependencies": {
22-
"abstract-level": "^1.0.0",
23+
"abstract-level": "^2.0.1",
2324
"functional-red-black-tree": "^1.0.1",
2425
"module-error": "^1.0.1"
2526
},
2627
"devDependencies": {
27-
"@voxpelli/tsconfig": "^4.0.0",
28+
"@voxpelli/tsconfig": "^15.0.0",
2829
"airtap": "^4.0.3",
2930
"airtap-playwright": "^1.0.1",
3031
"airtap-sauce": "^1.1.0",
3132
"buffer": "^6.0.3",
32-
"faucet": "^0.0.3",
3333
"hallmark": "^4.0.0",
3434
"nyc": "^15.1.0",
3535
"standard": "^17.0.0",
36+
"tap-arc": "^0.3.5",
3637
"tape": "^5.0.1",
37-
"typescript": "^4.5.5"
38+
"typescript": "^5.6.3"
3839
},
3940
"repository": {
4041
"type": "git",
@@ -49,6 +50,6 @@
4950
"memory"
5051
],
5152
"engines": {
52-
"node": ">=12"
53+
"node": ">=16"
5354
}
5455
}

0 commit comments

Comments
 (0)