Skip to content

Commit 32e8fca

Browse files
authored
Merge pull request #89 from ipfs/awesome-ipld
Awesome IPLD Endeavour
2 parents 6e39eae + f7e4047 commit 32e8fca

File tree

4 files changed

+159
-150
lines changed

4 files changed

+159
-150
lines changed

package.json

+2-2
Original file line numberDiff line numberDiff line change
@@ -44,7 +44,7 @@
4444
"dependencies": {
4545
"babel-runtime": "^6.11.6",
4646
"base32.js": "^0.1.0",
47-
"ipfs-block": "^0.3.0",
47+
"ipfs-block": "^0.4.0",
4848
"lock": "^0.1.3",
4949
"multihashes": "^0.2.2",
5050
"pull-defer": "^0.2.2",
@@ -67,4 +67,4 @@
6767
"nginnever <ginneversource@gmail.com>",
6868
"npmcdn-to-unpkg-bot <npmcdn-to-unpkg-bot@users.noreply.github.com>"
6969
]
70-
}
70+
}

src/stores/blockstore.js

+113-73
Original file line numberDiff line numberDiff line change
@@ -1,23 +1,23 @@
11
'use strict'
22

33
const Block = require('ipfs-block')
4-
const pull = require('pull-stream')
54
const Lock = require('lock')
65
const base32 = require('base32.js')
76
const path = require('path')
8-
const write = require('pull-write')
97
const parallel = require('run-parallel')
10-
const defer = require('pull-defer/source')
8+
const pull = require('pull-stream')
9+
const pullWrite = require('pull-write')
10+
const pullDefer = require('pull-defer/source')
1111

1212
const PREFIX_LENGTH = 5
13+
const EXTENSION = 'data'
1314

1415
exports = module.exports
1516

16-
function multihashToPath (multihash, extension) {
17-
extension = extension || 'data'
17+
function multihashToPath (multihash) {
1818
const encoder = new base32.Encoder()
1919
const hash = encoder.write(multihash).finalize()
20-
const filename = `${hash}.${extension}`
20+
const filename = `${hash}.${EXTENSION}`
2121
const folder = filename.slice(0, PREFIX_LENGTH)
2222

2323
return path.join(folder, filename)
@@ -27,82 +27,103 @@ exports.setUp = (basePath, BlobStore, locks) => {
2727
const store = new BlobStore(basePath + '/blocks')
2828
const lock = new Lock()
2929

30-
function writeBlock (block, cb) {
31-
if (!block || !block.data) {
32-
return cb(new Error('Invalid block'))
30+
// blockBlob is an object with:
31+
// { data: <>, key: <> }
32+
function writeBlock (blockBlob, callback) {
33+
if (!blockBlob || !blockBlob.data) {
34+
return callback(new Error('Invalid block'))
3335
}
3436

35-
const key = multihashToPath(block.key, block.extension)
36-
37-
lock(key, (release) => pull(
38-
pull.values([block.data]),
39-
store.write(key, release((err) => {
40-
if (err) {
41-
return cb(err)
42-
}
43-
cb(null, {key})
44-
}))
45-
))
37+
const key = multihashToPath(blockBlob.key)
38+
39+
lock(key, (release) => {
40+
pull(
41+
pull.values([
42+
blockBlob.data
43+
]),
44+
store.write(key, release(released))
45+
)
46+
})
47+
48+
// called once the lock is released
49+
function released (err) {
50+
if (err) {
51+
return callback(err)
52+
}
53+
callback(null, { key: key })
54+
}
4655
}
4756

4857
return {
49-
getStream (key, extension) {
58+
// returns a pull-stream of one block being read
59+
getStream (key) {
5060
if (!key) {
5161
return pull.error(new Error('Invalid key'))
5262
}
5363

54-
const p = multihashToPath(key, extension)
55-
const deferred = defer()
64+
const blockPath = multihashToPath(key)
65+
const deferred = pullDefer()
5666

57-
lock(p, (release) => {
58-
const ext = extension === 'data' ? 'protobuf' : extension
67+
lock(blockPath, (release) => {
5968
pull(
60-
store.read(p),
61-
pull.collect(release((err, data) => {
62-
if (err) {
63-
return deferred.abort(err)
64-
}
65-
66-
deferred.resolve(pull.values([
67-
new Block(Buffer.concat(data), ext)
68-
]))
69-
}))
69+
store.read(blockPath),
70+
pull.collect(release(released))
7071
)
7172
})
7273

74+
function released (err, data) {
75+
if (err) {
76+
return deferred.abort(err)
77+
}
78+
79+
deferred.resolve(
80+
pull.values([
81+
new Block(Buffer.concat(data))
82+
])
83+
)
84+
}
85+
7386
return deferred
7487
},
7588

89+
/*
90+
* putStream - write multiple blocks
91+
*
92+
* returns a pull-stream that expects blockBlobs
93+
*
94+
* NOTE: blockBlob is a { data: <>, key: <> } and not a
95+
* ipfs-block instance. This is because Block instances support
96+
* several types of hashing and it is up to the BlockService
97+
* to understand the right one to use (given the CID)
98+
*/
99+
// TODO
100+
// consider using a more explicit name, this can cause some confusion
101+
// since the natural association is
102+
// getStream - createReadStream - read one
103+
// putStream - createWriteStream - write one
104+
// where in fact it is:
105+
// getStream - createReadStream - read one (the same)
106+
// putStream - createFilesWriteStream = write several
107+
//
76108
putStream () {
77109
let ended = false
78110
let written = []
79111
let push = null
80112

81-
const sink = write((blocks, cb) => {
82-
parallel(blocks.map((block) => (cb) => {
83-
writeBlock(block, (err, meta) => {
84-
if (err) {
85-
return cb(err)
86-
}
87-
88-
if (push) {
89-
const read = push
90-
push = null
91-
read(null, meta)
92-
return cb()
93-
}
94-
95-
written.push(meta)
96-
cb()
97-
})
98-
}), cb)
113+
const sink = pullWrite((blockBlobs, cb) => {
114+
const tasks = writeTasks(blockBlobs)
115+
parallel(tasks, cb)
99116
}, null, 100, (err) => {
100117
ended = err || true
101-
if (push) push(ended)
118+
if (push) {
119+
push(ended)
120+
}
102121
})
103122

104123
const source = (end, cb) => {
105-
if (end) ended = end
124+
if (end) {
125+
ended = end
126+
}
106127
if (ended) {
107128
return cb(ended)
108129
}
@@ -114,35 +135,54 @@ exports.setUp = (basePath, BlobStore, locks) => {
114135
push = cb
115136
}
116137

117-
return {source, sink}
118-
},
138+
/*
139+
* Creates individual tasks to write each block blob that can be
140+
* exectured in parallel
141+
*/
142+
function writeTasks (blockBlobs) {
143+
return blockBlobs.map((blockBlob) => {
144+
return (cb) => {
145+
writeBlock(blockBlob, (err, meta) => {
146+
if (err) {
147+
return cb(err)
148+
}
149+
150+
if (push) {
151+
const read = push
152+
push = null
153+
read(null, meta)
154+
return cb()
155+
}
156+
157+
written.push(meta)
158+
cb()
159+
})
160+
}
161+
})
162+
}
119163

120-
has (key, extension, cb) {
121-
if (typeof extension === 'function') {
122-
cb = extension
123-
extension = undefined
164+
return {
165+
source: source,
166+
sink: sink
124167
}
168+
},
125169

170+
has (key, callback) {
126171
if (!key) {
127-
return cb(new Error('Invalid key'))
172+
return callback(new Error('Invalid key'))
128173
}
129174

130-
const p = multihashToPath(key, extension)
131-
store.exists(p, cb)
175+
const blockPath = multihashToPath(key)
176+
store.exists(blockPath, callback)
132177
},
133178

134-
delete (key, extension, cb) {
135-
if (typeof extension === 'function') {
136-
cb = extension
137-
extension = undefined
138-
}
139-
179+
delete (key, callback) {
140180
if (!key) {
141-
return cb(new Error('Invalid key'))
181+
return callback(new Error('Invalid key'))
142182
}
143183

144-
const p = multihashToPath(key, extension)
145-
store.remove(p, cb)
184+
const blockPath = multihashToPath(key)
185+
store.remove(blockPath, callback)
146186
}
147187
}
148188
}

src/stores/locks.js

+22-12
Original file line numberDiff line numberDiff line change
@@ -13,13 +13,17 @@ exports.setUp = (basePath, BlobStore) => {
1313
lock (callback) {
1414
function createLock () {
1515
pull(
16-
pull.values([new Buffer('LOCK')]),
16+
pull.values([
17+
new Buffer('LOCK')
18+
]),
1719
store.write(lockFile, callback)
1820
)
1921
}
2022

2123
function doesExist (err, exists) {
22-
if (err) return callback(err)
24+
if (err) {
25+
return callback(err)
26+
}
2327

2428
if (exists) {
2529
// default 100ms
@@ -37,16 +41,22 @@ exports.setUp = (basePath, BlobStore) => {
3741

3842
unlock (callback) {
3943
series([
40-
(cb) => store.remove(lockFile, cb),
41-
(cb) => store.exists(lockFile, (err, exists) => {
42-
if (err) return cb(err)
43-
44-
if (exists) {
45-
return cb(new Error('failed to remove lock'))
46-
}
47-
48-
cb()
49-
})
44+
(cb) => {
45+
store.remove(lockFile, cb)
46+
},
47+
(cb) => {
48+
store.exists(lockFile, (err, exists) => {
49+
if (err) {
50+
return cb(err)
51+
}
52+
53+
if (exists) {
54+
return cb(new Error('failed to remove lock'))
55+
}
56+
57+
cb()
58+
})
59+
}
5060
], callback)
5161
}
5262
}

0 commit comments

Comments
 (0)