Skip to content
This repository was archived by the owner on Sep 28, 2021. It is now read-only.

Commit 68f1204

Browse files
Alan Shawvasco-santos
Alan Shaw
authored andcommitted
refactor: use new IPFS async/await APIs (#30)
BREAKING CHANGE: Switch to using async/await and async iterators.
1 parent 8828822 commit 68f1204

9 files changed

+324
-388
lines changed

.travis.yml

+1
Original file line numberDiff line numberDiff line change
@@ -8,6 +8,7 @@ stages:
88

99
node_js:
1010
- '10'
11+
- '12'
1112

1213
os:
1314
- linux

package.json

+10-7
Original file line numberDiff line numberDiff line change
@@ -36,19 +36,22 @@
3636
"debug": "^4.1.1",
3737
"file-type": "^8.0.0",
3838
"filesize": "^3.6.1",
39-
"get-stream": "^3.0.0",
40-
"ipfs-unixfs": "~0.1.16",
39+
"it-buffer": "^0.1.1",
40+
"it-concat": "^1.0.0",
41+
"it-reader": "^2.1.0",
42+
"it-to-stream": "^0.1.1",
4143
"mime-types": "^2.1.21",
4244
"multihashes": "~0.4.14",
43-
"p-try-each": "^1.0.1",
44-
"stream-to-blob": "^2.0.0"
45+
"p-try-each": "^1.0.1"
4546
},
4647
"devDependencies": {
47-
"aegir": "^18.0.3",
48+
"aegir": "^20.5.0",
4849
"chai": "^4.2.0",
4950
"dirty-chai": "^2.0.1",
50-
"ipfs": "0.36.0",
51-
"ipfsd-ctl": "~0.42.2"
51+
"get-stream": "^3.0.0",
52+
"ipfs": "github:ipfs/js-ipfs#refactor/async-await-roundup",
53+
"ipfsd-ctl": "^1.0.2",
54+
"it-all": "^1.0.1"
5255
},
5356
"contributors": [
5457
"Alex Potsides <alex@achingbrain.net>",

src/dir-view/index.js

+1-2
Original file line numberDiff line numberDiff line change
@@ -17,7 +17,7 @@ function getParentHref (path) {
1717
function buildFilesList (path, links) {
1818
const rows = links.map((link) => {
1919
let row = [
20-
`<div class="ipfs-icon ipfs-_blank">&nbsp;</div>`,
20+
'<div class="ipfs-icon ipfs-_blank">&nbsp;</div>',
2121
`<a href="${path}${path.endsWith('/') ? '' : '/'}${link.Name}">${link.Name}</a>`,
2222
filesize(link.Tsize)
2323
]
@@ -78,5 +78,4 @@ function render (path, links) {
7878
`
7979
}
8080

81-
exports = module.exports
8281
exports.render = render

src/index.js

+52-78
Original file line numberDiff line numberDiff line change
@@ -1,115 +1,89 @@
1-
/* global Response */
1+
/* global Response, Blob */
22

33
'use strict'
44

5-
const stream = require('stream')
6-
const toBlob = require('stream-to-blob')
7-
8-
const debug = require('debug')
9-
const log = debug('ipfs:http:response')
5+
const toStream = require('it-to-stream')
6+
const concat = require('it-concat')
7+
const toBuffer = require('it-buffer')
8+
const log = require('debug')('ipfs:http:response')
109

1110
const resolver = require('./resolver')
1211
const pathUtils = require('./utils/path')
1312
const detectContentType = require('./utils/content-type')
1413

1514
// TODO: pass path and add Etag and X-Ipfs-Path + tests
16-
const header = (status = 200, statusText = 'OK', headers = {}) => ({
15+
const getHeader = (status = 200, statusText = 'OK', headers = {}) => ({
1716
status,
1817
statusText,
1918
headers
2019
})
2120

22-
const response = async (ipfsNode, ipfsPath) => {
23-
// handle hash resolve error (simple hash, test for directory now)
24-
const handleResolveError = async (node, path, error) => {
25-
if (error) {
26-
const errorString = error.toString()
27-
28-
if (errorString.includes('dag node is a directory')) {
29-
try {
30-
const content = await resolver.directory(node, path, error.cid)
31-
// dir render
32-
if (typeof content === 'string') {
33-
return new Response(content, header(200, 'OK', { 'Content-Type': 'text/html' }))
34-
}
35-
36-
// redirect to dir entry point (index)
37-
return Response.redirect(pathUtils.joinURLParts(path, content[0].Name))
38-
} catch (error) {
39-
log(error)
40-
return new Response(errorString, header(500, error.toString()))
41-
}
42-
}
43-
44-
if (errorString.startsWith('Error: no link named')) {
45-
return new Response(errorString, header(404, errorString))
46-
}
21+
// handle hash resolve error (simple hash, test for directory now)
22+
const handleResolveError = async (node, path, error) => {
23+
const errorString = error.toString()
4724

48-
if (errorString.startsWith('Error: multihash length inconsistent') || errorString.startsWith('Error: Non-base58 character')) {
49-
return new Response(errorString, header(400, errorString))
25+
if (errorString.includes('dag node is a directory')) {
26+
try {
27+
const content = await resolver.directory(node, path, error.cid)
28+
// dir render
29+
if (typeof content === 'string') {
30+
return new Response(content, getHeader(200, 'OK', { 'Content-Type': 'text/html' }))
5031
}
5132

33+
// redirect to dir entry point (index)
34+
return Response.redirect(pathUtils.joinURLParts(path, content[0].Name))
35+
} catch (error) {
5236
log(error)
53-
return new Response(errorString, header(500, errorString))
37+
return new Response(errorString, getHeader(500, error.toString()))
5438
}
5539
}
5640

41+
if (errorString.startsWith('Error: no link named')) {
42+
return new Response(errorString, getHeader(404, errorString))
43+
}
44+
45+
if (errorString.startsWith('Error: multihash length inconsistent') || errorString.startsWith('Error: Non-base58 character')) {
46+
return new Response(errorString, getHeader(400, errorString))
47+
}
48+
49+
return new Response(errorString, getHeader(500, errorString))
50+
}
51+
52+
const getResponse = async (ipfsNode, ipfsPath) => {
5753
// remove trailing slash for files if needed
5854
if (ipfsPath.endsWith('/')) {
5955
return Response.redirect(pathUtils.removeTrailingSlash(ipfsPath))
6056
}
6157

6258
try {
6359
const resolvedData = await resolver.cid(ipfsNode, ipfsPath)
60+
const { source, contentType } = await detectContentType(ipfsPath, ipfsNode.cat(resolvedData.cid))
6461

65-
const readableStream = ipfsNode.catReadableStream(resolvedData.cid)
66-
const responseStream = new stream.PassThrough({ highWaterMark: 1 })
67-
readableStream.pipe(responseStream)
68-
69-
return new Promise((resolve, reject) => {
70-
readableStream.once('error', (error) => {
71-
if (error) {
72-
log(error)
73-
return resolve(new Response(error.toString(), header(500, 'Error fetching the file')))
74-
}
75-
})
76-
77-
// return only after first chunk being checked
78-
let contentTypeDetected = false
79-
readableStream.on('data', async (chunk) => {
80-
// check mime on first chunk
81-
if (contentTypeDetected) {
82-
return
83-
}
84-
85-
contentTypeDetected = true
86-
// return Response with mime type
87-
const contentType = detectContentType(ipfsPath, chunk)
88-
89-
if (typeof Blob === 'undefined') {
90-
return contentType
91-
? resolve(new Response(responseStream, header(200, 'OK', { 'Content-Type': contentType })))
92-
: resolve(new Response(responseStream, header()))
93-
}
94-
95-
try {
96-
const blob = await toBlob(responseStream)
97-
98-
return contentType
99-
? resolve(new Response(blob, header(200, 'OK', { 'Content-Type': contentType })))
100-
: resolve(new Response(blob, header()))
101-
} catch (err) {
102-
return resolve(new Response(err.toString(), header(500, 'Error fetching the file')))
103-
}
104-
})
105-
})
62+
if (typeof Blob === 'undefined') {
63+
const responseStream = toStream.readable(toBuffer(source))
64+
65+
return contentType
66+
? new Response(responseStream, getHeader(200, 'OK', { 'Content-Type': contentType }))
67+
: new Response(responseStream, getHeader())
68+
}
69+
70+
try {
71+
const data = await concat(source)
72+
const blob = new Blob([data.slice()])
73+
74+
return contentType
75+
? new Response(blob, getHeader(200, 'OK', { 'Content-Type': contentType }))
76+
: new Response(blob, getHeader())
77+
} catch (err) {
78+
return new Response(err.toString(), getHeader(500, 'Error fetching the file'))
79+
}
10680
} catch (error) {
10781
log(error)
10882
return handleResolveError(ipfsNode, ipfsPath, error)
10983
}
11084
}
11185

11286
module.exports = {
113-
getResponse: response,
114-
resolver: resolver
87+
getResponse,
88+
resolver
11589
}

src/resolver.js

+6-9
Original file line numberDiff line numberDiff line change
@@ -2,7 +2,6 @@
22

33
const pTryEach = require('p-try-each')
44
const mh = require('multihashes')
5-
const CID = require('cids')
65
const debug = require('debug')
76
const log = debug('jsipfs:http:response:resolver')
87
log.error = debug('jsipfs:http:response:resolver:error')
@@ -21,7 +20,7 @@ const findIndexFile = (ipfs, path) => {
2120

2221
return {
2322
name: file,
24-
cid: new CID(stats.hash)
23+
cid: stats.cid
2524
}
2625
}
2726
}))
@@ -48,18 +47,16 @@ const directory = async (ipfs, path, cid) => {
4847
const cid = async (ipfs, path) => {
4948
const stats = await ipfs.files.stat(path)
5049

51-
const cid = new CID(stats.hash)
52-
5350
if (stats.type.includes('directory')) {
5451
const err = new Error('This dag node is a directory')
55-
err.cid = cid
52+
err.cid = stats.cid
5653
err.fileName = stats.name
5754
err.dagDirType = stats.type
5855

5956
throw err
6057
}
6158

62-
return { cid }
59+
return { cid: stats.cid }
6360
}
6461

6562
const multihash = async (ipfs, path) => {
@@ -71,7 +68,7 @@ const multihash = async (ipfs, path) => {
7168
}
7269

7370
module.exports = {
74-
directory: directory,
75-
cid: cid,
76-
multihash: multihash
71+
directory,
72+
cid,
73+
multihash
7774
}

src/utils/content-type.js

+23-3
Original file line numberDiff line numberDiff line change
@@ -2,20 +2,40 @@
22

33
const fileType = require('file-type')
44
const mime = require('mime-types')
5+
const Reader = require('it-reader')
56

6-
const detectContentType = (path, chunk) => {
7+
const detectContentType = async (path, source) => {
78
let fileSignature
89

910
// try to guess the filetype based on the first bytes
1011
// note that `file-type` doesn't support svgs, therefore we assume it's a svg if path looks like it
1112
if (!path.endsWith('.svg')) {
12-
fileSignature = fileType(chunk)
13+
try {
14+
const reader = Reader(source)
15+
const { value, done } = await reader.next(fileType.minimumBytes)
16+
17+
if (done) return { source: reader }
18+
19+
fileSignature = fileType(value.slice())
20+
21+
source = (async function * () { // eslint-disable-line require-await
22+
yield value
23+
yield * reader
24+
})()
25+
} catch (err) {
26+
if (err.code !== 'ERR_UNDER_READ') throw err
27+
28+
// not enough bytes for sniffing, just yield the data
29+
source = (async function * () { // eslint-disable-line require-await
30+
yield err.buffer // these are the bytes that were read (if any)
31+
})()
32+
}
1333
}
1434

1535
// if we were unable to, fallback to the `path` which might contain the extension
1636
const mimeType = mime.lookup(fileSignature ? fileSignature.ext : path)
1737

18-
return mime.contentType(mimeType)
38+
return { source, contentType: mime.contentType(mimeType) }
1939
}
2040

2141
module.exports = detectContentType

src/utils/path.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -43,7 +43,7 @@ function removeSlashFromBothEnds (url) {
4343

4444
function joinURLParts (...urls) {
4545
urls = urls.filter((url) => url.length > 0)
46-
urls = [ '' ].concat(urls.map((url) => removeSlashFromBothEnds(url)))
46+
urls = [''].concat(urls.map((url) => removeSlashFromBothEnds(url)))
4747

4848
return urls.join('/')
4949
}

0 commit comments

Comments
 (0)