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

Commit d9d0c08

Browse files
committed
feat: initial implementation
1 parent f236f1b commit d9d0c08

20 files changed

+27221
-1
lines changed

.gitignore

+64
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,64 @@
1+
dist/
2+
3+
# Logs
4+
logs
5+
*.log
6+
npm-debug.log*
7+
yarn-debug.log*
8+
yarn-error.log*
9+
10+
# Runtime data
11+
pids
12+
*.pid
13+
*.seed
14+
*.pid.lock
15+
16+
# Directory for instrumented libs generated by jscoverage/JSCover
17+
lib-cov
18+
19+
# Coverage directory used by tools like istanbul
20+
coverage
21+
22+
# nyc test coverage
23+
.nyc_output
24+
25+
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
26+
.grunt
27+
28+
# Bower dependency directory (https://bower.io/)
29+
bower_components
30+
31+
# node-waf configuration
32+
.lock-wscript
33+
34+
# Compiled binary addons (http://nodejs.org/api/addons.html)
35+
build/Release
36+
37+
# Dependency directories
38+
node_modules/
39+
jspm_packages/
40+
41+
# Typescript v1 declaration files
42+
typings/
43+
44+
# Optional npm cache directory
45+
.npm
46+
47+
# Optional eslint cache
48+
.eslintcache
49+
50+
# Optional REPL history
51+
.node_repl_history
52+
53+
# Output of 'npm pack'
54+
*.tgz
55+
56+
# Yarn Integrity file
57+
.yarn-integrity
58+
59+
# dotenv environment variables file
60+
.env
61+
62+
# while testing npm5
63+
package-lock.json
64+
yarn.lock

README.md

+22-1
Original file line numberDiff line numberDiff line change
@@ -1 +1,22 @@
1-
# ipfs-http-response
1+
# js-ipfs-http-response
2+
3+
> Creates an HTTP response from an IPFS Hash
4+
5+
### Installation
6+
7+
> TODO
8+
9+
## Usage
10+
11+
This project consists on creating a HTTP response from an IPFS Hash. This response can be a file, a directory list view or the entry point of a web page.
12+
13+
```js
14+
const ipfsHttpResponse = require('ipfs-http-response')
15+
16+
ipfsHttpResponse(ipfsNode, ipfsPath)
17+
.then((response) => {
18+
...
19+
})
20+
```
21+
22+
![ipfs-http-response usage](docs/ipfs-http-response.png "ipfs-http-response usage")

ci/Jenkinsfile

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
// Warning: This file is automatically synced from https://github.com/ipfs/ci-sync so if you want to change it, please change it there and ask someone to sync all repositories.
2+
javascript()

docs/ipfs-http-response.png

17.6 KB
Loading

package.json

+50
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,50 @@
1+
{
2+
"name": "js-ipfs-http-response",
3+
"version": "0.1.0",
4+
"description": "Creates an HTTP response from an IPFS Hash",
5+
"main": "src/index.js",
6+
"scripts": {
7+
"lint": "aegir lint",
8+
"release": "aegir release",
9+
"build": "aegir build",
10+
"test": "aegir test -t node"
11+
},
12+
"pre-push": [
13+
"lint",
14+
"test"
15+
],
16+
"repository": {
17+
"type": "git",
18+
"url": "git+https://github.com/ipfs/js-ipfs-http-response.git"
19+
},
20+
"keywords": [
21+
"ipfs",
22+
"http",
23+
"response"
24+
],
25+
"author": "Vasco Santos <vasco.santos@moxy.studio>",
26+
"license": "MIT",
27+
"bugs": {
28+
"url": "https://github.com/ipfs/js-ipfs-http-response/issues"
29+
},
30+
"homepage": "https://github.com/ipfs/js-ipfs-http-response#readme",
31+
"dependencies": {
32+
"async": "^2.6.0",
33+
"cids": "^0.5.3",
34+
"debug": "^3.1.0",
35+
"file-type": "^8.0.0",
36+
"filesize": "^3.6.1",
37+
"ipfs-unixfs": "^0.1.14",
38+
"mime-types": "^2.1.18",
39+
"multihashes": "^0.4.13",
40+
"promisify-es6": "^1.0.3",
41+
"readable-stream-node-to-web": "^1.0.1"
42+
},
43+
"devDependencies": {
44+
"aegir": "^13.1.0",
45+
"chai": "^4.1.2",
46+
"dirty-chai": "^2.0.1",
47+
"ipfs": "^0.28.2",
48+
"ipfsd-ctl": "^0.36.0"
49+
}
50+
}

src/dir-view/index.js

+86
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,86 @@
1+
'use strict'
2+
3+
const filesize = require('filesize')
4+
5+
const mainStyle = require('./style')
6+
const pathUtil = require('../utils/path')
7+
8+
function getParentDirectoryURL (originalParts) {
9+
const parts = originalParts.slice()
10+
11+
if (parts.length > 1) {
12+
parts.pop()
13+
}
14+
15+
return [ '', 'ipfs' ].concat(parts).join('/')
16+
}
17+
18+
function buildFilesList (path, links) {
19+
const rows = links.map((link) => {
20+
let row = [
21+
`<div class="ipfs-icon ipfs-_blank">&nbsp;</div>`,
22+
`<a href="${pathUtil.joinURLParts(path, link.name)}">${link.name}</a>`,
23+
filesize(link.size)
24+
]
25+
26+
row = row.map((cell) => `<td>${cell}</td>`).join('')
27+
28+
return `<tr>${row}</tr>`
29+
})
30+
31+
return rows.join('')
32+
}
33+
34+
function buildTable (path, links) {
35+
const parts = pathUtil.splitPath(path)
36+
const parentDirectoryURL = getParentDirectoryURL(parts)
37+
38+
return `
39+
<table class="table table-striped">
40+
<tbody>
41+
<tr>
42+
<td class="narrow">
43+
<div class="ipfs-icon ipfs-_blank">&nbsp;</div>
44+
</td>
45+
<td class="padding">
46+
<a href="${parentDirectoryURL}">..</a>
47+
</td>
48+
<td></td>
49+
</tr>
50+
${buildFilesList(path, links)}
51+
</tbody>
52+
</table>
53+
`
54+
}
55+
56+
function render (path, links) {
57+
return `
58+
<!DOCTYPE html>
59+
<html>
60+
<head>
61+
<meta charset="utf-8">
62+
<title>${path}</title>
63+
<style>${mainStyle}</style>
64+
</head>
65+
<body>
66+
<div id="header" class="row">
67+
<div class="col-xs-2">
68+
<div id="logo" class="ipfs-logo"></div>
69+
</div>
70+
</div>
71+
<br>
72+
<div class="col-xs-12">
73+
<div class="panel panel-default">
74+
<div class="panel-heading">
75+
<strong>Index of ${path}</strong>
76+
</div>
77+
${buildTable(path, links)}
78+
</div>
79+
</div>
80+
</body>
81+
</html>
82+
`
83+
}
84+
85+
exports = module.exports
86+
exports.render = render

src/dir-view/style.js

+16
Some generated files are not rendered by default. Learn more about customizing how changed files appear on GitHub.

src/index.js

+102
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,102 @@
1+
/* global Response */
2+
3+
'use strict'
4+
5+
const fileType = require('file-type')
6+
const mimeTypes = require('mime-types')
7+
const stream = require('stream')
8+
const nodeToWebStream = require('readable-stream-node-to-web')
9+
10+
const resolver = require('./resolver')
11+
const pathUtils = require('./utils/path')
12+
13+
const header = (status = 200, statusText = 'OK', headers = {}) => ({
14+
status,
15+
statusText,
16+
headers
17+
})
18+
19+
module.exports = (ipfsNode, ipfsPath) => {
20+
// handle hash resolve error (simple hash, test for directory now)
21+
const handleResolveError = (node, path, error) => {
22+
if (error) {
23+
const errorString = error.toString()
24+
25+
return new Promise((resolve, reject) => {
26+
// switch case with true feels so wrong.
27+
switch (true) {
28+
case (errorString === 'Error: This dag node is a directory'):
29+
resolver.directory(node, path, error.fileName)
30+
.then((content) => {
31+
// dir render
32+
if (typeof content === 'string') {
33+
resolve(new Response(content, header(200, 'OK', { 'Content-Type': 'text/html' })))
34+
}
35+
36+
// redirect to dir entry point (index)
37+
resolve(Response.redirect(pathUtils.joinURLParts(path, content[0].name)))
38+
})
39+
.catch((error) => {
40+
resolve(new Response(errorString, header(500, error.toString())))
41+
})
42+
break
43+
case errorString.startsWith('Error: no link named'):
44+
resolve(new Response(errorString, header(404, errorString)))
45+
break
46+
case errorString.startsWith('Error: multihash length inconsistent'):
47+
case errorString.startsWith('Error: Non-base58 character'):
48+
resolve(new Response(errorString, header(400, errorString)))
49+
break
50+
default:
51+
resolve(new Response(errorString, header(500, errorString)))
52+
}
53+
})
54+
}
55+
}
56+
57+
return new Promise((resolve, reject) => {
58+
// remove trailing slash for files if needed
59+
if (ipfsPath.endsWith('/')) {
60+
resolve(Response.redirect(pathUtils.removeTrailingSlash(ipfsPath)))
61+
}
62+
63+
resolver.multihash(ipfsNode, ipfsPath)
64+
.then((resolvedData) => {
65+
const readableStream = ipfsNode.files.catReadableStream(resolvedData.multihash)
66+
const responseStream = new stream.PassThrough({ highWaterMark: 1 })
67+
readableStream.pipe(responseStream)
68+
69+
readableStream.once('error', (error) => {
70+
if (error) {
71+
resolve(new Response(error.toString(), header(500, 'Service Worker Error')))
72+
}
73+
})
74+
75+
// return only after first chunk being checked
76+
let filetypeChecked = false
77+
readableStream.on('data', (chunk) => {
78+
// check mime on first chunk
79+
if (filetypeChecked) {
80+
return
81+
}
82+
filetypeChecked = true
83+
// return Response with mime type
84+
const fileSignature = fileType(chunk)
85+
const mimeType = mimeTypes.lookup(fileSignature ? fileSignature.ext : null)
86+
87+
if (mimeType) {
88+
resolve(
89+
new Response(typeof ReadableStream === 'function' ? nodeToWebStream(responseStream) : responseStream,
90+
header(200, 'OK', { 'Content-Type': mimeTypes.contentType(mimeType) }))
91+
)
92+
} else {
93+
resolve(new Response(typeof ReadableStream === 'function' ? nodeToWebStream(responseStream) : responseStream,
94+
header()))
95+
}
96+
})
97+
})
98+
.catch((error) => {
99+
resolve(handleResolveError(ipfsNode, ipfsPath, error))
100+
})
101+
})
102+
}

0 commit comments

Comments
 (0)