forked from internetarchive/dweb-mirror
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathHashStore.js
194 lines (184 loc) · 7.43 KB
/
HashStore.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
// This is level 5, there are major differences in level 8
// see https://github.com/Level/level/blob/master/UPGRADING.md
// The changes should be contained here, but might not be trivial and its unclear what if any value gained.
const level = require('level');
const debug = require('debug')('dweb-mirror:HashStore');
const each = require('async/each');
const waterfall = require('async/waterfall');
/**
* A generic Hash Store built on top of level,
* table/key/value triples.
* Note this could probably build on top of Redis or Gun as well -
* Redis might end up too large for memory, and actually want local item store, not global one
* There is one of these at the top directory of each cache directory.
*/
/**
* Common parameters across all methods
* table name of table (by convention, in dweb-mirror we use the mapping as the table name e.g. `sha1.filepath`
* key name of key (by convention its the left side of the table name e.g. the sha1.
* For some functions (put) it may be an object mapping keys to val in which case val is ignored
* val value to store
*/
class HashStore {
/**
* @param config = { dir } Prefix for Directory where data stored (actual directories are DIR followed by TABLE so this should typically end with a punctuation or /}
* @returns {HashStore}
*/
constructor(config) {
this.config = config;
this.tables = {}; // Caches pointers to leveldb objects that manage tables (each being a stored in a directory)
return this;
}
_tablepath(table) { // Return the file system path to where we have, or will create, a table
return `${this.config.dir}${table}`;
}
_db(table, retries = undefined, cb) {
if (typeof retries === 'function') { cb = retries, retries = 10; }
if (!this.tables[table]) {
const tablepath = this._tablepath(table);
level(tablepath, {}, (err, res) => {
if (err) {
debug('Hashstore failed to open db at %s: %s', tablepath, err.message);
if (retries) {
setTimeout(() => this._db(table, --retries, cb), 100);
} else {
cb(err);
}
} else {
this.tables[table] = res;
cb(null, res);
}
});
} else {
cb(null, this.tables[table]); // Note file might not be open yet, if not any put/get/del will be queued by level till its ready
}
}
destroy(table, cb) {
level.destroy(this._tablepath(table), cb);
}
destroyAll(cb) {
each(Object.keys(this.tables), (table, cb2) => this.destroy(table, cb2), cb);
}
put(table, key, val, cb) {
/*
Set a key to a val for a specific table.
val = any valid persistent value acceptable to level (not sure what limits are)
key = any valid key for level (not sure what limits are)
cb(err)
*/
if (cb) { try { f.call(this, cb); } catch (err) { cb(err); } } else { return new Promise((resolve, reject) => { try { f.call(this, (err, res) => { if (err) { reject(err); } else { resolve(res); } }); } catch (err) { reject(err); } }); } // Promisify pattern v2
function f(cb2) {
debug('%s.%o <- %o', table, key, val);
waterfall([
cb3 => this._db(table, cb3),
(db, cb3) => {
if (typeof key === 'object') {
db.batch(Object.keys(key).map(k => ({ type: 'put', key: k, value: key[k] })), cb3);
} else {
db.put(key, val, cb3);
}
}
], (err) => {
if (err) {
debug('put %s %s <- %s failed %s', table, key, val, err.message);
}
cb2(err);
});
}
}
/**
* Retrieve value stored to key, returns `undefined` if not found (not an error)
*/
async get(table, key, cb) {
if (cb) { try { f.call(this, cb); } catch (err) { cb(err); } } else { return new Promise((resolve, reject) => { try { f.call(this, (err, res) => { if (err) { reject(err); } else { resolve(res); } }); } catch (err) { reject(err); } }); } // Promisify pattern v2
function f(cb2) {
// This is similar to level.get except not finding the value is not an error, it returns undefined.
// noinspection JSPotentiallyInvalidUsageOfClassThis
if (!table || !key) {
debug('Error: <hashstore>.get requires table (%o) and key (%o)', table, key);
cb(new Error('<hashstore>.get requires table and key'));
}
return waterfall([
cb3 => this._db(table, cb3),
(db, cb3) => db.get(key, cb3)
], (err, val) => {
if (err && (err.type === 'NotFoundError')) { // Undefined is not an error
debug('get %s %s failed %s', table, key, err.message);
cb2(null, undefined);
} else {
cb2(null, val);
}
});
}
}
/* UNUSED - make async with callback if going to use it
//Delete value stored at the key, future `get` will return undefined.
async del(table, key) {
if (typeof key === "object") { // Delete all keys in object
await this._db(table).batch(Object.keys(key).map(k => {return {type: "del", key: k};}));
} else {
await this._db(table).del(key);
}
}
*/
/* UNUSED - make async with callback if going to use it
Creates a stream, that calls cb(key, value) for each key/value pair.
async map(table, cb, {end=undefined}={}) {
// cb(data) => data.key, data.value
// Returns a stream so can add further .on
// UNTESTED
return this._db(table)
.createReadStream()
.on('data', cb );
}
*/
async keys(table, cb) {
if (cb) { try { f.call(this, cb); } catch (err) { cb(err); } } else { return new Promise((resolve, reject) => { try { f.call(this, (err, res) => { if (err) { reject(err); } else { resolve(res); } }); } catch (err) { reject(err); } }); } // Promisify pattern v2
function f(cb) {
const keys = [];
// noinspection JSPotentiallyInvalidUsageOfClassThis
const db = this._db(table, (err, db) => {
if (err) {
debug('keys %s failed', table);
cb(err);
} else {
db
.createKeyStream()
// Note close comes after End
.on('data', (key) => keys.push(key))
.on('end', () => { debug('%s keys on end = %o', table, keys); cb(null, keys); }) // Gets to end of stream
// .on('close', () => // Gets to end of stream, or closed from outside - not used as get "end" as well
.on('error', (err) => { console.error('Error in stream from', table); cb(err); });
}
});
}
}
// noinspection JSUnusedGlobalSymbols
static async test() {
try {
this.init({ dir: 'testleveldb.' });
await this.put('Testtable', 'testkey', 'testval');
let res = await this.get('Testtable', 'testkey');
console.assert(res === 'testval');
await this.put('Testtable', { A: 'AAA', B: 'BBB' });
res = await this.get('Testtable', 'A');
console.assert(res === 'AAA');
res = await this.get('Testtable', 'B');
console.assert(res === 'BBB');
res = await this.del('Testtable', 'A');
res = await this.get('Testtable', 'A');
console.assert(res === undefined);
res = await this.keys('Testtable');
console.assert(res.length === 2);
// Test using callback
res = await this.keys('Testtable', (err, res) => {
console.assert(res.length === 2);
});
// Now test batches
// Now test map
} catch (err) {
console.log('Error caught in HashStore.test', err);
}
}
}
exports = module.exports = HashStore;