Skip to content

Commit f8bbc7e

Browse files
committed
Upgrade from abstract-leveldown to abstract-level
1 parent 5663c67 commit f8bbc7e

File tree

9 files changed

+552
-841
lines changed

9 files changed

+552
-841
lines changed

.github/workflows/sauce.yml

+11-11
Original file line numberDiff line numberDiff line change
@@ -18,14 +18,14 @@ jobs:
1818
SAUCE_CONNECT_DOWNLOAD_ON_INSTALL: true
1919
- name: Add host
2020
run: echo "127.0.0.1 airtap.local" | sudo tee -a /etc/hosts
21-
- name: Test
22-
run: npm run test-browsers
23-
env:
24-
SAUCE_USERNAME: ${{ secrets.SAUCE_USERNAME }}
25-
SAUCE_ACCESS_KEY: ${{ secrets.SAUCE_ACCESS_KEY }}
26-
- name: Coverage
27-
run: npm run coverage
28-
- name: Codecov
29-
uses: codecov/codecov-action@v2
30-
with:
31-
file: coverage/lcov.info
21+
# - name: Test
22+
# run: npm run test-browsers
23+
# env:
24+
# SAUCE_USERNAME: ${{ secrets.SAUCE_USERNAME }}
25+
# SAUCE_ACCESS_KEY: ${{ secrets.SAUCE_ACCESS_KEY }}
26+
# - name: Coverage
27+
# run: npm run coverage
28+
# - name: Codecov
29+
# uses: codecov/codecov-action@v2
30+
# with:
31+
# file: coverage/lcov.info

LICENSE

+1-1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,6 @@
11
The MIT License (MIT)
22

3-
Copyright © 2013-present Rod Vagg and the contributors to memdown.
3+
Copyright © 2013 Rod Vagg and the contributors to memory-level.
44

55
Permission is hereby granted, free of charge, to any person obtaining a copy
66
of this software and associated documentation files (the "Software"), to deal

README.md

+37-73
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
1-
# memdown
1+
# memory-level
22

3-
> In-memory [`abstract-leveldown`] store for Node.js and browsers.
3+
**In-memory [`abstract-level`][abstract-level] database for Node.js and browsers, backed by a [fully persistent red-black tree](https://www.npmjs.com/package/functional-red-black-tree).** The successor to [`memdown`](https://github.com/Level/memdown) and [`level-mem`](https://github.com/Level/mem).
4+
5+
> :pushpin: Which module should I use? What is `abstract-level`? Head over to the [FAQ](https://github.com/Level/community#faq).
46
57
[![level badge][level-badge]](https://github.com/Level/awesome)
68
[![npm](https://img.shields.io/npm/v/memdown.svg)](https://www.npmjs.com/package/memdown)
@@ -11,109 +13,79 @@
1113
[![Common Changelog](https://common-changelog.org/badge.svg)](https://common-changelog.org)
1214
[![Donate](https://img.shields.io/badge/donate-orange?logo=open-collective&logoColor=fff)](https://opencollective.com/level)
1315

14-
## Example
16+
## Usage
1517

1618
_If you are upgrading: please see [`UPGRADING.md`](./UPGRADING.md)._
1719

1820
```js
19-
const levelup = require('levelup')
20-
const memdown = require('memdown')
21+
const { MemoryLevel } = require('memory-level')
2122

22-
const db = levelup(memdown())
23+
// Create a database
24+
const db = new MemoryLevel({ valueEncoding: 'json' })
2325

24-
db.put('hey', 'you', (err) => {
25-
if (err) throw err
26+
// Add an entry with key 'a' and value 1
27+
await db.put('a', 1)
2628

27-
db.get('hey', { asBuffer: false }, (err, value) => {
28-
if (err) throw err
29-
console.log(value) // 'you'
30-
})
31-
})
32-
```
29+
// Add multiple entries
30+
await db.batch([{ type: 'put', key: 'b', value: 2 }])
3331

34-
With `async/await`:
32+
// Get value of key 'a': 1
33+
const value = await db.get('a')
3534

36-
```js
37-
await db.put('hey', 'you')
38-
const value = await db.get('hey', { asBuffer: false })
35+
// Iterate entries with keys that are greater than 'a'
36+
for await (const [key, value] of db.iterator({ gt: 'a' })) {
37+
console.log(value) // 2
38+
}
3939
```
4040

41-
Your data is discarded when the process ends or you release a reference to the store. Note as well, though the internals of `memdown` operate synchronously - [`levelup`] does not.
42-
43-
## Browser support
44-
45-
[![Sauce Test Status](https://app.saucelabs.com/browser-matrix/level-ci.svg)](https://app.saucelabs.com/u/level-ci)
46-
47-
## Data types
48-
49-
Keys and values can be strings or Buffers. Any other key type will be irreversibly stringified. The only exceptions are `null` and `undefined`. Keys and values of that type are rejected.
41+
With callbacks:
5042

5143
```js
52-
const db = levelup(memdown())
53-
54-
db.put('example', 123, (err) => {
44+
db.put('example', { hello: 'world' }, (err) => {
5545
if (err) throw err
5646

57-
db.createReadStream({
58-
keyAsBuffer: false,
59-
valueAsBuffer: false
60-
}).on('data', (entry) => {
61-
console.log(typeof entry.key) // 'string'
62-
console.log(typeof entry.value) // 'string'
47+
db.get('example', (err, value) => {
48+
if (err) throw err
49+
console.log(value) // { hello: 'world' }
6350
})
6451
})
6552
```
6653

67-
If you desire non-destructive encoding (e.g. to store and retrieve numbers as-is), wrap `memdown` with [`encoding-down`]. Alternatively install [`level-mem`] which conveniently bundles [`levelup`], `memdown` and [`encoding-down`]. Such an approach is also recommended if you want to achieve universal (isomorphic) behavior. For example, you could have [`leveldown`] in a backend and `memdown` in the frontend.
54+
<!-- ## Browser support
6855
69-
```js
70-
const encode = require('encoding-down')
71-
const db = levelup(encode(memdown(), { valueEncoding: 'json' }))
72-
73-
db.put('example', 123, (err) => {
74-
if (err) throw err
56+
[![Sauce Test Status](https://app.saucelabs.com/browser-matrix/level-ci.svg)](https://app.saucelabs.com/u/level-ci) -->
7557

76-
db.createReadStream({
77-
keyAsBuffer: false,
78-
valueAsBuffer: false
79-
}).on('data', (entry) => {
80-
console.log(typeof entry.key) // 'string'
81-
console.log(typeof entry.value) // 'number'
82-
})
83-
})
84-
```
58+
## API
8559

86-
## Snapshot guarantees
60+
The API of `memory-level` follows that of [`abstract-level`](https://github.com/Level/abstract-level) with a one additional constructor option (see below). The `createIfMissing` and `errorIfExists` options of `abstract-level` are not relevant here. Data is discarded when the last reference to the database is released (i.e. `db = null`). Closing or reopening the database has no effect on the data. Data is _not_ copied: when storing a Buffer value for example, subsequent mutations to that Buffer will affect the stored data too.
8761

88-
A `memdown` store is backed by [a fully persistent data structure](https://www.npmjs.com/package/functional-red-black-tree) and thus has snapshot guarantees. Meaning that reads operate on a snapshot in time, unaffected by simultaneous writes.
62+
### `db = new MemoryLevel([options])`
8963

90-
## Test
64+
Besides `abstract-level` options, the optional `options` object may contain:
9165

92-
In addition to the regular `npm test`, you can test `memdown` in a browser of choice with:
66+
- `storeEncoding` (string): one of `'buffer'`, `'view'`, `'utf8'`. How to store data internally. This affects which data types can be stored non-destructively. The default is `'buffer'` (that means Buffer) which is non-destructive. In browsers it may be preferable to use `'view'` (Uint8Array) to be able to exclude the [`buffer`](https://github.com/feross/buffer) shim. Or if there's no need to store binary data, then `'utf8'` (String). Regardless of the `storeEncoding`, `memory-level` supports input that is of any of the aforementioned types, but internally converts it to one type in order to provide a consistent sort order.
9367

94-
```
95-
npm run test-browser-local
96-
```
68+
## Install
9769

98-
To check code coverage:
70+
With [npm](https://npmjs.org) do:
9971

10072
```
101-
npm run coverage
73+
npm install memory-level
10274
```
10375

10476
## Contributing
10577

106-
[`Level/memdown`](https://github.com/Level/memdown) is an **OPEN Open Source Project**. This means that:
78+
[`Level/memory-level`](https://github.com/Level/memory-level) is an **OPEN Open Source Project**. This means that:
10779

10880
> Individuals making significant and valuable contributions are given commit-access to the project to contribute as they see fit. This project is more like an open wiki than a standard guarded open source project.
10981
11082
See the [Contribution Guide](https://github.com/Level/community/blob/master/CONTRIBUTING.md) for more details.
11183

112-
## Big Thanks
84+
<!-- ## Big Thanks
11385
11486
Cross-browser Testing Platform and Open Source ♥ Provided by [Sauce Labs](https://saucelabs.com).
11587
116-
[![Sauce Labs logo](./sauce-labs.svg)](https://saucelabs.com)
88+
[![Sauce Labs logo](./sauce-labs.svg)](https://saucelabs.com) -->
11789

11890
## Donate
11991

@@ -123,14 +95,6 @@ Support us with a monthly donation on [Open Collective](https://opencollective.c
12395

12496
[MIT](LICENSE)
12597

126-
[`abstract-leveldown`]: https://github.com/Level/abstract-leveldown
127-
128-
[`levelup`]: https://github.com/Level/levelup
129-
130-
[`encoding-down`]: https://github.com/Level/encoding-down
131-
132-
[`leveldown`]: https://github.com/Level/leveldown
133-
134-
[`level-mem`]: https://github.com/Level/mem
98+
[abstract-level]: https://github.com/Level/abstract-level
13599

136100
[level-badge]: https://leveljs.org/img/badge.svg

UPGRADING.md

+63-71
Original file line numberDiff line numberDiff line change
@@ -1,111 +1,103 @@
11
# Upgrade Guide
22

3-
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].
3+
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-
## 6.0.0
5+
## 1.0.0
66

7-
Legacy range options have been removed ([Level/community#86](https://github.com/Level/community/issues/86)). If you previously did:
7+
**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.**
88

9-
```js
10-
db.iterator({ start: 'a', end: 'z' })
11-
```
12-
13-
An error would now be thrown and you must instead do:
14-
15-
```js
16-
db.iterator({ gte: 'a', lte: 'z' })
17-
```
18-
19-
This release also drops support of legacy runtime environments ([Level/community#98](https://github.com/Level/community/issues/98)):
20-
21-
- Node.js 6 and 8
22-
- Internet Explorer 11
23-
- Safari 9-11
24-
- Stock Android browser (AOSP).
25-
26-
Lastly, and less likely to be a breaking change, the [`immediate`](https://github.com/calvinmetcalf/immediate) browser shim for `process.nextTick()` has been replaced with the smaller [`queue-microtask`](https://github.com/feross/queue-microtask).
27-
28-
## 5.0.0
29-
30-
Support of keys & values other than strings and Buffers has been dropped. Internally `memdown` now stores keys & values as Buffers which solves a number of compatibility issues ([#186](https://github.com/Level/memdown/issues/186)). If you pass in a key or value that isn't a string or Buffer, it will be irreversibly stringified.
31-
32-
## 4.0.0
33-
34-
This is an upgrade to `abstract-leveldown@6` which solves long-standing issues around serialization and type support.
35-
36-
### Range options are now serialized
37-
38-
Previously, range options like `lt` were passed through as-is by `abstract-leveldown`, unlike keys. This makes no difference for `memdown` as it does not serialize anything.
39-
40-
### The rules for range options have been relaxed
41-
42-
Because `null`, `undefined`, zero-length strings and zero-length buffers are significant types in encodings like `bytewise` and `charwise`, they became valid as range options in `abstract-leveldown`. This means `db.iterator({ gt: undefined })` is not the same as `db.iterator({})`.
9+
We've put together several upgrade guides for different modules. See the [FAQ](https://github.com/Level/community#faq) to find the best upgrade guide for you. This upgrade guide describes how to replace `memdown` or `level-mem` with `memory-level`. If you are using any of the following, please also read the upgrade guide of [`abstract-level@1`](https://github.com/Level/abstract-level/blob/main/UPGRADING.md#100) which goes into more detail about these:
4310

44-
For `memdown`, when used by itself, the behavior of `null`, `undefined`, zero-length strings and zero-length buffers is undefined.
11+
- Specific error messages (replaced with error codes)
12+
- The callback argument of the constructor (gone)
13+
- The `'binary'` encoding (renamed to `'buffer'`, with `'binary'` as an alias)
14+
- The `db.iterator().end()` method (renamed to `close()`, with `end()` as an alias)
15+
- Zero-length keys and range options (now valid)
16+
- The `'ascii'`, `'ucs2'`, `'utf16le'` and `'id'` encodings (gone)
17+
- The undocumented `encoding` alias for the `valueEncoding` option (gone)
18+
- The `db.supports.bufferKeys` property.
4519

46-
### Nullish values are rejected
20+
Support of Node.js 10 has been dropped.
4721

48-
In addition to rejecting `null` and `undefined` as _keys_, `abstract-leveldown` now also rejects these types as _values_, due to preexisting significance in streams and iterators.
22+
### Upgrade from `level-mem` to `memory-level`
4923

50-
### Zero-length array keys are rejected
24+
Using `new` is now required. If you previously did:
5125

52-
Though this was already the case, `abstract-leveldown` has replaced the behavior with an explicit `Array.isArray()` check and a new error message.
53-
54-
### Browser support
55-
56-
IE10 has been dropped.
26+
```js
27+
const mem = require('level-mem')
28+
const db1 = mem()
29+
const db2 = mem({ valueEncoding: 'json' })
30+
```
5731

58-
## 3.0.0
32+
You must now do:
5933

60-
Dropped support for node 4. No other breaking changes.
34+
```js
35+
const { MemoryLevel } = require('memory-level')
36+
const db1 = new MemoryLevel()
37+
const db2 = new MemoryLevel({ valueEncoding: 'json' })
38+
```
6139

62-
## 2.0.0
40+
Node.js readable streams must now be created with a new standalone module called [`level-read-stream`](https://github.com/Level/read-stream), rather than database methods like `db.createReadStream()`. Please see its [upgrade guide](https://github.com/Level/read-stream/blob/main/UPGRADING.md#100) for details.
6341

64-
This release drops Node.js 0.12, brings `memdown` up to par with latest [`levelup`][levelup] (v2) and [`abstract-leveldown`][abstract-leveldown] (v4), simplifies serialization and removes global state.
42+
### Upgrade from `memdown` to `memory-level`
6543

66-
### Targets latest [`levelup`][levelup]
44+
_This section is only relevant if you're using `memdown` directly, rather than as a transitive dependency of `level-mem`._
6745

68-
Usage has changed to:
46+
The `asBuffer`, `valueAsBuffer` and `keyAsBuffer` options have been replaced with encoding options. The default encoding is `'utf8'` which means operations return strings rather than Buffers by default. If you previously did:
6947

7048
```js
71-
const levelup = require('levelup')
7249
const memdown = require('memdown')
50+
const db = memdown()
7351

74-
const db = levelup(memdown())
52+
db.get('example', { asBuffer: false }, callback)
53+
db.get('example', callback)
7554
```
7655

77-
From the old:
56+
You must now do:
7857

7958
```js
80-
const db = levelup('mydb', { db: memdown })
81-
```
59+
const { MemoryLevel } = require('memory-level')
60+
const db = new MemoryLevel()
8261

83-
### No stringification of keys and values
62+
db.get('example', callback)
63+
db.get('example', { valueEncoding: 'buffer' }, callback)
64+
```
8465

85-
This means that in addition to Buffers, you can store any JS type without the need for [`encoding-down`][encoding-down]. This release also makes behavior consistent in Node.js and browsers. Please refer to the [README](./README.md) for a detailed explanation.
66+
Or using promises (new):
8667

87-
### No global state or `location` argument
68+
```js
69+
const str = await db.get('example')
70+
const buf = await db.get('example', { valueEncoding: 'buffer' })
71+
```
8872

89-
If you previously did this to make a global store:
73+
Or using Uint8Array (new):
9074

9175
```js
92-
const db = levelup('mydb', { db: memdown })
76+
const arr = await db.get('example', { valueEncoding: 'view' })
9377
```
9478

95-
You must now attach the store to a global yourself (if you desire global state):
79+
If you were wrapping `memdown` with `levelup`, `encoding-down` and / or `subleveldown`, remove those modules. If you previously did:
9680

9781
```js
98-
const db = window.mydb = levelup(memdown())
99-
```
82+
const memdown = require('memdown')
83+
const levelup = require('levelup')
84+
const enc = require('encoding-down')
85+
const subleveldown = require('subleveldown')
10086

101-
### No `null` batch operations
87+
const db = levelup(enc(memdown()))
88+
const sublevel = subleveldown(db, 'foo')
89+
```
10290

103-
Instead of skipping `null` operations, `db.batch([null])` will throw an error courtesy of [`abstract-leveldown`][abstract-leveldown].
91+
You must now do:
10492

105-
[changelog]: CHANGELOG.md
93+
```js
94+
const { MemoryLevel } = require('memory-level')
95+
const db = new MemoryLevel()
96+
const sublevel = db.sublevel('foo')
97+
```
10698

107-
[abstract-leveldown]: https://github.com/Level/abstract-leveldown
99+
Lastly, private properties like `_store` (unlikely used externally) are no longer accessible.
108100

109-
[levelup]: https://github.com/Level/levelup
101+
---
110102

111-
[encoding-down]: https://github.com/Level/encoding-down
103+
_For earlier releases, before `memory-level` was forked from `memdown`, please see [the upgrade guide of `memdown`](https://github.com/Level/memdown/blob/master/UPGRADING.md)._

0 commit comments

Comments
 (0)