From cc41624ac05f4d15397f5fde2b0515c75bd2b858 Mon Sep 17 00:00:00 2001 From: Harsh Vakharia Date: Tue, 29 Aug 2017 12:53:15 +0530 Subject: [PATCH 01/14] feat: add gateway route to daemon --- package.json | 4 +- src/http-api/gateway/resolver.js | 86 ++++++++++++++++++++++++++ src/http-api/gateway/utils/html.js | 83 +++++++++++++++++++++++++ src/http-api/gateway/utils/path.js | 35 +++++++++++ src/http-api/gateway/utils/style.js | 16 +++++ src/http-api/resources/files.js | 93 +++++++++++++++++++++++++++++ src/http-api/routes/files.js | 12 ++++ 7 files changed, 328 insertions(+), 1 deletion(-) create mode 100644 src/http-api/gateway/resolver.js create mode 100644 src/http-api/gateway/utils/html.js create mode 100644 src/http-api/gateway/utils/path.js create mode 100644 src/http-api/gateway/utils/style.js diff --git a/package.json b/package.json index be5c0cfef3..f714d4f6ce 100644 --- a/package.json +++ b/package.json @@ -126,12 +126,14 @@ "lodash.get": "^4.4.2", "lodash.sortby": "^4.7.0", "lodash.values": "^4.3.0", + "mime-types": "^2.1.13", "mafmt": "^2.1.8", "mkdirp": "^0.5.1", "multiaddr": "^2.3.0", "multihashes": "~0.4.5", "once": "^1.4.0", "path-exists": "^3.0.0", + "promised-for": "^1.0.0", "peer-book": "^0.5.0", "peer-id": "^0.9.0", "peer-info": "^0.10.0", @@ -204,4 +206,4 @@ "tcme ", "ᴠɪᴄᴛᴏʀ ʙᴊᴇʟᴋʜᴏʟᴍ " ] -} \ No newline at end of file +} diff --git a/src/http-api/gateway/resolver.js b/src/http-api/gateway/resolver.js new file mode 100644 index 0000000000..9c5ab7e208 --- /dev/null +++ b/src/http-api/gateway/resolver.js @@ -0,0 +1,86 @@ +'use strict' + +const mh = require('multihashes') +const pf = require('promised-for') + +const html = require('./utils/html') +const PathUtil = require('./utils/path') + +const INDEX_HTML_FILES = [ 'index.html', 'index.htm', 'index.shtml' ] + +const resolveDirectory = (ipfs, path, multihash) => { + return ipfs + .object + .get(multihash, { enc: 'base58' }) + .then((DAGNode) => { + const links = DAGNode.links + const indexFiles = links.filter((link) => INDEX_HTML_FILES.indexOf(link.name) !== -1) + + // found index file in links + if (indexFiles.length > 0) { + return indexFiles + } + + return html.build(path, links) + }) +} + +const resolveMultihash = (ipfs, path) => { + const parts = PathUtil.splitPath(path) + const partsLength = parts.length + + return pf( + { + multihash: parts[0], + index: 0 + }, + (i) => i.index < partsLength, + (i) => { + const currentIndex = i.index + const currentMultihash = i.multihash + + // throws error when invalid multihash is passed + mh.validate(mh.fromB58String(currentMultihash)) + + return ipfs + .object + .get(currentMultihash, { enc: 'base58' }) + .then((DAGNode) => { + if (currentIndex === partsLength - 1) { + // leaf node + return { + multihash: currentMultihash, + index: currentIndex + 1 + } + } else { + // find multihash of requested named-file + // in current DAGNode's links + let multihashOfNextFile + const nextFileName = parts[currentIndex + 1] + const links = DAGNode.links + + for (let link of links) { + if (link.name === nextFileName) { + // found multihash of requested named-file + multihashOfNextFile = mh.toB58String(link.multihash) + break + } + } + + if (!multihashOfNextFile) { + throw new Error(`no link named "${nextFileName}" under ${currentMultihash}`) + } + + return { + multihash: multihashOfNextFile, + index: currentIndex + 1 + } + } + }) + }) +} + +module.exports = { + resolveDirectory, + resolveMultihash +} diff --git a/src/http-api/gateway/utils/html.js b/src/http-api/gateway/utils/html.js new file mode 100644 index 0000000000..dd5abe6b4f --- /dev/null +++ b/src/http-api/gateway/utils/html.js @@ -0,0 +1,83 @@ +'use strict' + +const filesize = require('filesize') + +const HTML_PAGE_STYLE = require('./style') +const PathUtil = require('./path') + +const getParentDirectoryURL = (originalParts) => { + const parts = originalParts.splice() + + if (parts.length > 1) { + parts.pop() + } + + return [ '', 'ipfs' ].concat(parts).join('/') +} + +const buildFilesList = (path, links) => { + const rows = links.map((link) => { + let row = [ + `
 
`, + `${link.name}`, + filesize(link.size) + ] + + row = row.map((cell) => `${cell}`).join('') + + return `${row}` + }) + + return rows.join('') +} + +const buildTable = (path, links) => { + const parts = PathUtil.splitPath(path) + let parentDirectoryURL = getParentDirectoryURL(parts) + + return ` + + + + + + + + ${buildFilesList(path, links)} + +
+
 
+
+ .. +
+ ` +} + +module.exports.build = (path, links) => { + return ` + + + + + ${path} + + + + +
+
+
+
+ Index of ${path} +
+ ${buildTable(path, links)} +
+
+ + + ` +} diff --git a/src/http-api/gateway/utils/path.js b/src/http-api/gateway/utils/path.js new file mode 100644 index 0000000000..a46ea8a96d --- /dev/null +++ b/src/http-api/gateway/utils/path.js @@ -0,0 +1,35 @@ +'use strict' + +const splitPath = (path) => { + if (path[path.length - 1] === '/') path = path.substring(0, path.length - 1) + return path.substring(6).split('/') +} + +const removeLeadingSlash = (url) => { + if (url[0] === '/') url = url.substring(1) + return url +} + +const removeTrailingSlash = (url) => { + if (url.endsWith('/')) url = url.substring(0, url.length - 1) + return url +} + +const removeSlashFromBothEnds = (url) => { + url = removeLeadingSlash(url) + url = removeTrailingSlash(url) + return url +} + +const joinURLParts = (...urls) => { + urls = urls.filter((url) => url.length > 0) + urls = [ '' ].concat(urls.map((url) => removeSlashFromBothEnds(url))) + + return urls.join('/') +} + +module.exports = { + splitPath, + removeTrailingSlash, + joinURLParts +} diff --git a/src/http-api/gateway/utils/style.js b/src/http-api/gateway/utils/style.js new file mode 100644 index 0000000000..9e1548741a --- /dev/null +++ b/src/http-api/gateway/utils/style.js @@ -0,0 +1,16 @@ +'use strict' + +module.exports = `html{font-family:sans-serif;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%}body{margin:0}a{background-color:transparent}a:active,a:hover{outline:0}strong{font-weight:700}button::-moz-focus-inner,input::-moz-focus-inner{padding:0;border:0}table{border-spacing:0;border-collapse:collapse}td{padding:0} @media print{*,:after,:before{color:#000!important;text-shadow:none!important;background:0 0!important;-webkit-box-shadow:none!important;box-shadow:none!important}a,a:visited{text-decoration:underline}a[href]:after{content:" (" attr(href) ")"}tr{page-break-inside:avoid}.table{border-collapse:collapse!important}.table td{background-color:#fff!important}}@font-face{font-family:'Glyphicons Halflings';src:url(../fonts/glyphicons-halflings-regular.eot);src:url(../fonts/glyphicons-halflings-regular.eot?#iefix) format('embedded-opentype'),url(../fonts/glyphicons-halflings-regular.woff2) format('woff2'),url(../fonts/glyphicons-halflings-regular.woff) format('woff'),url(../fonts/glyphicons-halflings-regular.ttf) format('truetype'),url(../fonts/glyphicons-halflings-regular.svg#glyphicons_halflingsregular) format('svg')}*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}:after,:before{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{font-size:10px;-webkit-tap-highlight-color:rgba(0,0,0,0)}body{font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.42857143;color:#333;background-color:#fff}a{color:#337ab7;text-decoration:none}a:focus,a:hover{color:#23527c;text-decoration:underline}a:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.row{margin-right:-15px;margin-left:-15px}.col-xs-12,.col-xs-2{position:relative;min-height:1px;padding-right:15px;padding-left:15px}.col-xs-12,.col-xs-2{float:left}.col-xs-12{width:100%}.col-xs-2{width:16.66666667%}table{background-color:transparent}.table{width:100%;max-width:100%;margin-bottom:20px}.table>tbody>tr>td{padding:8px;line-height:1.42857143;vertical-align:top;border-top:1px solid #ddd}.table-striped>tbody>tr:nth-of-type(odd){background-color:#f9f9f9}.form-control::-moz-placeholder{color:#999;opacity:1}.form-control:-ms-input-placeholder{color:#999}.panel{margin-bottom:20px;background-color:#fff;border:1px solid transparent;border-radius:4px;-webkit-box-shadow:0 1px 1px rgba(0,0,0,.05);box-shadow:0 1px 1px rgba(0,0,0,.05)}.panel-heading{padding:10px 15px;border-bottom:1px solid transparent;border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table{margin-bottom:0}.panel>.table:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child td:first-child{border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child td:last-child{border-bottom-right-radius:3px}.panel>.table>tbody:first-child>tr:first-child td{border-top:0}.panel-default{border-color:#ddd}.panel-default>.panel-heading{color:#333;background-color:#f5f5f5;border-color:#ddd}.row:after,.row:before{display:table;content:" "}.row:after{clear:both}@-ms-viewport{width:device-width}.ipfs-_blank{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-_page{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-aac{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ai{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-aiff{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-avi{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-bmp{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-c{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-cpp{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-css{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dat{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dmg{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-doc{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dotx{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dwg{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dxf{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-eps{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-exe{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-flv{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-gif{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-h{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-hpp{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-html{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ics{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-iso{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-java{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-jpg{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-js{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-key{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-less{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-logo{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-mid{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-mp3{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-mp4{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-mpg{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-odf{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ods{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-odt{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-otp{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ots{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ott{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-pdf{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-php{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-png{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ppt{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-psd{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-py{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-qt{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-rar{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-rb{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-rtf{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-sass{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-scss{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-sql{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-tga{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-tgz{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-tiff{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-txt{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-wav{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-xls{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-xlsx{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-xml{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-yml{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-zip{background-image:url('');background-repeat:no-repeat;background-size:contain} +.narrow {width: 0px;} +.padding { margin: 100px;} +#header { + background: #000; +} +#logo { + height: 25px; + margin: 10px; +} +.ipfs-icon { + width:16px; +} +` diff --git a/src/http-api/resources/files.js b/src/http-api/resources/files.js index 68ecc797bd..6644cad315 100644 --- a/src/http-api/resources/files.js +++ b/src/http-api/resources/files.js @@ -11,6 +11,10 @@ const toPull = require('stream-to-pull-stream') const pushable = require('pull-pushable') const EOL = require('os').EOL const toStream = require('pull-stream-to-stream') +const mime = require('mime-types') + +const GatewayResolver = require('../gateway/resolver') +const PathUtils = require('../gateway/utils/path') exports = module.exports @@ -213,3 +217,92 @@ exports.add = { ) } } + +exports.gateway = { + checkHash: (request, reply) => { + if (!request.params.hash) { + return reply('Path Resolve error: path must contain at least one component').code(400).takeover() + } + + return reply({ + ref: `/ipfs/${request.params.hash}` + }) + }, + handler: (request, reply) => { + const ref = request.pre.args.ref + const ipfs = request.server.app.ipfs + + return GatewayResolver + .resolveMultihash(ipfs, ref) + .then((data) => { + ipfs + .files + .cat(data.multihash) + .then((stream) => { + if (ref.endsWith('/')) { + // remove trailing slash for files + return reply + .redirect(PathUtils.removeTrailingSlash(ref)) + .permanent(true) + } else { + const mimeType = mime.lookup(ref) + + if (!stream._read) { + stream._read = () => {} + stream._readableState = {} + } + + if (mimeType) { + return reply(stream) + .header('Content-Type', mime.contentType(mimeType)) + .header('X-Stream-Output', '1') + } else { + return reply(stream) + .header('X-Stream-Output', '1') + } + } + }) + .catch((err) => { + if (err.toString() === 'Error: This dag node is a directory') { + return GatewayResolver + .resolveDirectory(ipfs, ref, data.multihash) + .then((data) => { + if (typeof data === 'string') { + // no index file found + if (!ref.endsWith('/')) { + // for a directory, if URL doesn't end with a / + // append / and redirect permanent to that URL + return reply.redirect(`${ref}/`).permanent(true) + } else { + // send directory listing + return reply(data) + } + } else { + // found index file + // redirect to URL/ + return reply.redirect(PathUtils.joinURLParts(ref, data[0].name)) + } + }).catch((err) => { + log.error(err) + return reply(err.toString()).code(500) + }) + } else { + log.error(err) + return reply(err.toString()).code(500) + } + }) + }).catch((err) => { + const errorToString = err.toString() + + if (errorToString.startsWith('Error: no link named')) { + return reply(errorToString).code(404) + } else if (errorToString.startsWith('Error: multihash length inconsistent') || + errorToString.startsWith('Error: Non-base58 character')) { + return reply(errorToString).code(400) + } else { + log.error(err) + return reply(errorToString).code(500) + } + }) + } +} diff --git a/src/http-api/routes/files.js b/src/http-api/routes/files.js index da57b3f2f1..6328ef2faa 100644 --- a/src/http-api/routes/files.js +++ b/src/http-api/routes/files.js @@ -4,6 +4,7 @@ const resources = require('./../resources') module.exports = (server) => { const api = server.select('API') + const gateway = server.select('Gateway') api.route({ // TODO fix method @@ -41,4 +42,15 @@ module.exports = (server) => { handler: resources.files.add.handler } }) + + gateway.route({ + method: '*', + path: '/ipfs/{hash*}', + config: { + pre: [ + { method: resources.files.gateway.checkHash, assign: 'args' } + ], + handler: resources.files.gateway.handler + } + }) } From c0303fa5205eb67d2a33f749ac9b061a5561c9fb Mon Sep 17 00:00:00 2001 From: Yahya Date: Tue, 29 Aug 2017 12:03:05 +0200 Subject: [PATCH 02/14] feat: adding Gateway endeavour - improve codebase --- gulpfile.js | 2 +- package.json | 4 + src/cli/commands/daemon.js | 2 +- src/http-api/gateway/resolver.js | 86 ----- src/http-api/resources/files.js | 308 ---------------- .../api}/resources/bitswap.js | 0 src/{http-api => http/api}/resources/block.js | 0 .../api}/resources/bootstrap.js | 0 .../api}/resources/config.js | 0 src/http/api/resources/files.js | 345 ++++++++++++++++++ src/{http-api => http/api}/resources/id.js | 0 src/{http-api => http/api}/resources/index.js | 0 .../api}/resources/object.js | 0 .../api}/resources/pubsub.js | 0 src/{http-api => http/api}/resources/repo.js | 0 src/{http-api => http/api}/resources/swarm.js | 0 .../api}/resources/version.js | 0 src/{http-api => http/api}/routes/bitswap.js | 0 src/{http-api => http/api}/routes/block.js | 0 .../api}/routes/bootstrap.js | 0 src/{http-api => http/api}/routes/config.js | 0 src/{http-api => http/api}/routes/files.js | 12 - src/{http-api => http/api}/routes/id.js | 0 src/{http-api => http/api}/routes/index.js | 0 src/{http-api => http/api}/routes/object.js | 0 src/{http-api => http/api}/routes/pubsub.js | 0 src/{http-api => http/api}/routes/repo.js | 0 src/{http-api => http/api}/routes/swarm.js | 0 src/{http-api => http/api}/routes/version.js | 0 src/{http-api => http}/error-handler.js | 0 src/http/gateway/resolver.js | 114 ++++++ src/http/gateway/resources/gateway.js | 148 ++++++++ src/http/gateway/resources/index.js | 3 + src/http/gateway/routes/gateway.js | 18 + src/http/gateway/routes/index.js | 5 + src/{http-api => http}/gateway/utils/html.js | 0 src/{http-api => http}/gateway/utils/path.js | 0 src/{http-api => http}/gateway/utils/style.js | 0 src/{http-api => http}/index.js | 4 +- test/cli/pubsub.js | 2 +- test/gateway/index.js | 55 +++ test/gateway/spec/gateway.js | 50 +++ test/http-api/index.js | 2 +- test/interop/daemons/js.js | 2 +- test/node.js | 10 + test/utils/ipfs-factory-daemon/index.js | 2 +- 46 files changed, 761 insertions(+), 413 deletions(-) delete mode 100644 src/http-api/gateway/resolver.js delete mode 100644 src/http-api/resources/files.js rename src/{http-api => http/api}/resources/bitswap.js (100%) rename src/{http-api => http/api}/resources/block.js (100%) rename src/{http-api => http/api}/resources/bootstrap.js (100%) rename src/{http-api => http/api}/resources/config.js (100%) create mode 100644 src/http/api/resources/files.js rename src/{http-api => http/api}/resources/id.js (100%) rename src/{http-api => http/api}/resources/index.js (100%) rename src/{http-api => http/api}/resources/object.js (100%) rename src/{http-api => http/api}/resources/pubsub.js (100%) rename src/{http-api => http/api}/resources/repo.js (100%) rename src/{http-api => http/api}/resources/swarm.js (100%) rename src/{http-api => http/api}/resources/version.js (100%) rename src/{http-api => http/api}/routes/bitswap.js (100%) rename src/{http-api => http/api}/routes/block.js (100%) rename src/{http-api => http/api}/routes/bootstrap.js (100%) rename src/{http-api => http/api}/routes/config.js (100%) rename src/{http-api => http/api}/routes/files.js (75%) rename src/{http-api => http/api}/routes/id.js (100%) rename src/{http-api => http/api}/routes/index.js (100%) rename src/{http-api => http/api}/routes/object.js (100%) rename src/{http-api => http/api}/routes/pubsub.js (100%) rename src/{http-api => http/api}/routes/repo.js (100%) rename src/{http-api => http/api}/routes/swarm.js (100%) rename src/{http-api => http/api}/routes/version.js (100%) rename src/{http-api => http}/error-handler.js (100%) create mode 100644 src/http/gateway/resolver.js create mode 100644 src/http/gateway/resources/gateway.js create mode 100644 src/http/gateway/resources/index.js create mode 100644 src/http/gateway/routes/gateway.js create mode 100644 src/http/gateway/routes/index.js rename src/{http-api => http}/gateway/utils/html.js (100%) rename src/{http-api => http}/gateway/utils/path.js (100%) rename src/{http-api => http}/gateway/utils/style.js (100%) rename src/{http-api => http}/index.js (97%) create mode 100644 test/gateway/index.js create mode 100644 test/gateway/spec/gateway.js diff --git a/gulpfile.js b/gulpfile.js index 2be23a3e2b..26304b51a1 100644 --- a/gulpfile.js +++ b/gulpfile.js @@ -4,7 +4,7 @@ const gulp = require('gulp') const parallel = require('async/parallel') const series = require('async/series') const createTempRepo = require('./test/utils/create-repo-nodejs.js') -const HTTPAPI = require('./src/http-api') +const HTTPAPI = require('./src/http') const leftPad = require('left-pad') let nodes = [] diff --git a/package.json b/package.json index f714d4f6ce..34004a2e80 100644 --- a/package.json +++ b/package.json @@ -27,6 +27,7 @@ "test:unit:node": "gulp test:node", "test:unit:node:core": "TEST=core npm run test:unit:node", "test:unit:node:http": "TEST=http npm run test:unit:node", + "test:unit:node:gateway": "TEST=gateway npm run test:unit:node", "test:unit:node:cli": "TEST=cli npm run test:unit:node", "test:unit:browser": "gulp test:browser", "test:interop": "npm run test:interop:node", @@ -94,6 +95,8 @@ "boom": "^5.2.0", "cids": "^0.5.1", "debug": "^3.0.0", + "file-type": "^6.1.0", + "filesize": "^3.5.10", "fsm-event": "^2.1.0", "glob": "^7.1.2", "hapi": "^16.5.2", @@ -128,6 +131,7 @@ "lodash.values": "^4.3.0", "mime-types": "^2.1.13", "mafmt": "^2.1.8", + "mime-types": "^2.1.16", "mkdirp": "^0.5.1", "multiaddr": "^2.3.0", "multihashes": "~0.4.5", diff --git a/src/cli/commands/daemon.js b/src/cli/commands/daemon.js index 0414ca9654..4529e2af54 100644 --- a/src/cli/commands/daemon.js +++ b/src/cli/commands/daemon.js @@ -1,6 +1,6 @@ 'use strict' -const HttpAPI = require('../../http-api') +const HttpAPI = require('../../http') const utils = require('../utils') const print = utils.print diff --git a/src/http-api/gateway/resolver.js b/src/http-api/gateway/resolver.js deleted file mode 100644 index 9c5ab7e208..0000000000 --- a/src/http-api/gateway/resolver.js +++ /dev/null @@ -1,86 +0,0 @@ -'use strict' - -const mh = require('multihashes') -const pf = require('promised-for') - -const html = require('./utils/html') -const PathUtil = require('./utils/path') - -const INDEX_HTML_FILES = [ 'index.html', 'index.htm', 'index.shtml' ] - -const resolveDirectory = (ipfs, path, multihash) => { - return ipfs - .object - .get(multihash, { enc: 'base58' }) - .then((DAGNode) => { - const links = DAGNode.links - const indexFiles = links.filter((link) => INDEX_HTML_FILES.indexOf(link.name) !== -1) - - // found index file in links - if (indexFiles.length > 0) { - return indexFiles - } - - return html.build(path, links) - }) -} - -const resolveMultihash = (ipfs, path) => { - const parts = PathUtil.splitPath(path) - const partsLength = parts.length - - return pf( - { - multihash: parts[0], - index: 0 - }, - (i) => i.index < partsLength, - (i) => { - const currentIndex = i.index - const currentMultihash = i.multihash - - // throws error when invalid multihash is passed - mh.validate(mh.fromB58String(currentMultihash)) - - return ipfs - .object - .get(currentMultihash, { enc: 'base58' }) - .then((DAGNode) => { - if (currentIndex === partsLength - 1) { - // leaf node - return { - multihash: currentMultihash, - index: currentIndex + 1 - } - } else { - // find multihash of requested named-file - // in current DAGNode's links - let multihashOfNextFile - const nextFileName = parts[currentIndex + 1] - const links = DAGNode.links - - for (let link of links) { - if (link.name === nextFileName) { - // found multihash of requested named-file - multihashOfNextFile = mh.toB58String(link.multihash) - break - } - } - - if (!multihashOfNextFile) { - throw new Error(`no link named "${nextFileName}" under ${currentMultihash}`) - } - - return { - multihash: multihashOfNextFile, - index: currentIndex + 1 - } - } - }) - }) -} - -module.exports = { - resolveDirectory, - resolveMultihash -} diff --git a/src/http-api/resources/files.js b/src/http-api/resources/files.js deleted file mode 100644 index 6644cad315..0000000000 --- a/src/http-api/resources/files.js +++ /dev/null @@ -1,308 +0,0 @@ -'use strict' - -const mh = require('multihashes') -const multipart = require('ipfs-multipart') -const debug = require('debug') -const tar = require('tar-stream') -const log = debug('jsipfs:http-api:files') -log.error = debug('jsipfs:http-api:files:error') -const pull = require('pull-stream') -const toPull = require('stream-to-pull-stream') -const pushable = require('pull-pushable') -const EOL = require('os').EOL -const toStream = require('pull-stream-to-stream') -const mime = require('mime-types') - -const GatewayResolver = require('../gateway/resolver') -const PathUtils = require('../gateway/utils/path') - -exports = module.exports - -// common pre request handler that parses the args and returns `key` which is assigned to `request.pre.args` -exports.parseKey = (request, reply) => { - if (!request.query.arg) { - return reply({ - Message: "Argument 'key' is required", - Code: 0 - }).code(400).takeover() - } - - let key = request.query.arg - if (key.indexOf('/ipfs/') === 0) { - key = key.substring(6) - } - - const slashIndex = key.indexOf('/') - if (slashIndex > 0) { - key = key.substring(0, slashIndex) - } - - try { - mh.fromB58String(key) - } catch (err) { - log.error(err) - return reply({ - Message: 'invalid ipfs ref path', - Code: 0 - }).code(500).takeover() - } - - reply({ - key: request.query.arg - }) -} - -exports.cat = { - // uses common parseKey method that returns a `key` - parseArgs: exports.parseKey, - - // main route handler which is called after the above `parseArgs`, but only if the args were valid - handler: (request, reply) => { - const key = request.pre.args.key - const ipfs = request.server.app.ipfs - - ipfs.files.cat(key, (err, stream) => { - if (err) { - log.error(err) - return reply({ - Message: 'Failed to cat file: ' + err, - Code: 0 - }).code(500) - } - - // hapi is not very clever and throws if no - // - _read method - // - _readableState object - // are there :( - if (!stream._read) { - stream._read = () => {} - stream._readableState = {} - } - return reply(stream).header('X-Stream-Output', '1') - }) - } -} - -exports.get = { - // uses common parseKey method that returns a `key` - parseArgs: exports.parseKey, - - // main route handler which is called after the above `parseArgs`, but only if the args were valid - handler: (request, reply) => { - const key = request.pre.args.key - const ipfs = request.server.app.ipfs - const pack = tar.pack() - - ipfs.files.getPull(key, (err, stream) => { - if (err) { - log.error(err) - - reply({ - Message: 'Failed to get file: ' + err, - Code: 0 - }).code(500) - return - } - - pull( - stream, - pull.asyncMap((file, cb) => { - const header = {name: file.path} - if (!file.content) { - header.type = 'directory' - pack.entry(header) - cb() - } else { - header.size = file.size - const packStream = pack.entry(header, cb) - if (!packStream) { - // this happens if the request is aborted - // we just skip things then - log('other side hung up') - return cb() - } - toStream.source(file.content).pipe(packStream) - } - }), - pull.onEnd((err) => { - if (err) { - log.error(err) - pack.emit('error', err) - pack.destroy() - return - } - - pack.finalize() - }) - ) - - // the reply must read the tar stream, - // to pull values through - reply(pack).header('X-Stream-Output', '1') - }) - } -} - -exports.add = { - handler: (request, reply) => { - if (!request.payload) { - return reply({ - Message: 'Array, Buffer, or String is required.', - code: 0 - }).code(400).takeover() - } - - const ipfs = request.server.app.ipfs - // TODO: make pull-multipart - const parser = multipart.reqParser(request.payload) - let filesParsed = false - - const fileAdder = pushable() - - parser.on('file', (fileName, fileStream) => { - const filePair = { - path: fileName, - content: toPull(fileStream) - } - filesParsed = true - fileAdder.push(filePair) - }) - - parser.on('directory', (directory) => { - fileAdder.push({ - path: directory, - content: '' - }) - }) - - parser.on('end', () => { - if (!filesParsed) { - return reply({ - Message: "File argument 'data' is required.", - code: 0 - }).code(400).takeover() - } - fileAdder.end() - }) - - pull( - fileAdder, - ipfs.files.createAddPullStream(), - pull.map((file) => { - return { - Name: file.path ? file.path : file.hash, - Hash: file.hash - } - }), - pull.map((file) => JSON.stringify(file) + EOL), - pull.collect((err, files) => { - if (err) { - return reply({ - Message: err, - Code: 0 - }).code(500) - } - - if (files.length === 0 && filesParsed) { - return reply({ - Message: 'Failed to add files.', - Code: 0 - }).code(500) - } - - reply(files.join('\n')) - .header('x-chunked-output', '1') - .header('content-type', 'application/json') - }) - ) - } -} - -exports.gateway = { - checkHash: (request, reply) => { - if (!request.params.hash) { - return reply('Path Resolve error: path must contain at least one component').code(400).takeover() - } - - return reply({ - ref: `/ipfs/${request.params.hash}` - }) - }, - handler: (request, reply) => { - const ref = request.pre.args.ref - const ipfs = request.server.app.ipfs - - return GatewayResolver - .resolveMultihash(ipfs, ref) - .then((data) => { - ipfs - .files - .cat(data.multihash) - .then((stream) => { - if (ref.endsWith('/')) { - // remove trailing slash for files - return reply - .redirect(PathUtils.removeTrailingSlash(ref)) - .permanent(true) - } else { - const mimeType = mime.lookup(ref) - - if (!stream._read) { - stream._read = () => {} - stream._readableState = {} - } - - if (mimeType) { - return reply(stream) - .header('Content-Type', mime.contentType(mimeType)) - .header('X-Stream-Output', '1') - } else { - return reply(stream) - .header('X-Stream-Output', '1') - } - } - }) - .catch((err) => { - if (err.toString() === 'Error: This dag node is a directory') { - return GatewayResolver - .resolveDirectory(ipfs, ref, data.multihash) - .then((data) => { - if (typeof data === 'string') { - // no index file found - if (!ref.endsWith('/')) { - // for a directory, if URL doesn't end with a / - // append / and redirect permanent to that URL - return reply.redirect(`${ref}/`).permanent(true) - } else { - // send directory listing - return reply(data) - } - } else { - // found index file - // redirect to URL/ - return reply.redirect(PathUtils.joinURLParts(ref, data[0].name)) - } - }).catch((err) => { - log.error(err) - return reply(err.toString()).code(500) - }) - } else { - log.error(err) - return reply(err.toString()).code(500) - } - }) - }).catch((err) => { - const errorToString = err.toString() - - if (errorToString.startsWith('Error: no link named')) { - return reply(errorToString).code(404) - } else if (errorToString.startsWith('Error: multihash length inconsistent') || - errorToString.startsWith('Error: Non-base58 character')) { - return reply(errorToString).code(400) - } else { - log.error(err) - return reply(errorToString).code(500) - } - }) - } -} diff --git a/src/http-api/resources/bitswap.js b/src/http/api/resources/bitswap.js similarity index 100% rename from src/http-api/resources/bitswap.js rename to src/http/api/resources/bitswap.js diff --git a/src/http-api/resources/block.js b/src/http/api/resources/block.js similarity index 100% rename from src/http-api/resources/block.js rename to src/http/api/resources/block.js diff --git a/src/http-api/resources/bootstrap.js b/src/http/api/resources/bootstrap.js similarity index 100% rename from src/http-api/resources/bootstrap.js rename to src/http/api/resources/bootstrap.js diff --git a/src/http-api/resources/config.js b/src/http/api/resources/config.js similarity index 100% rename from src/http-api/resources/config.js rename to src/http/api/resources/config.js diff --git a/src/http/api/resources/files.js b/src/http/api/resources/files.js new file mode 100644 index 0000000000..1ea7cb0099 --- /dev/null +++ b/src/http/api/resources/files.js @@ -0,0 +1,345 @@ +'use strict' + +const mh = require('multihashes') +const multipart = require('ipfs-multipart') +const debug = require('debug') +const tar = require('tar-stream') +const log = debug('jsipfs:http-api:files') +log.error = debug('jsipfs:http-api:files:error') +const pull = require('pull-stream') +const toPull = require('stream-to-pull-stream') +const pushable = require('pull-pushable') +const EOL = require('os').EOL +const toStream = require('pull-stream-to-stream') +// const fileType = require('file-type') +// const mime = require('mime-types') +// const GatewayResolver = require('../gateway/resolver') +// const PathUtils = require('../gateway/utils/path') +// const Stream = require('stream') + +exports = module.exports + +// common pre request handler that parses the args and returns `key` which is assigned to `request.pre.args` +exports.parseKey = (request, reply) => { + if (!request.query.arg) { + return reply({ + Message: "Argument 'key' is required", + Code: 0 + }).code(400).takeover() + } + + let key = request.query.arg + if (key.indexOf('/ipfs/') === 0) { + key = key.substring(6) + } + + const slashIndex = key.indexOf('/') + if (slashIndex > 0) { + key = key.substring(0, slashIndex) + } + + try { + mh.fromB58String(key) + } catch (err) { + log.error(err) + return reply({ + Message: 'invalid ipfs ref path', + Code: 0 + }).code(500).takeover() + } + + reply({ + key: request.query.arg + }) +} + +exports.cat = { + // uses common parseKey method that returns a `key` + parseArgs: exports.parseKey, + + // main route handler which is called after the above `parseArgs`, but only if the args were valid + handler: (request, reply) => { + const key = request.pre.args.key + const ipfs = request.server.app.ipfs + + ipfs.files.cat(key, (err, stream) => { + if (err) { + log.error(err) + return reply({ + Message: 'Failed to cat file: ' + err, + Code: 0 + }).code(500) + } + + // hapi is not very clever and throws if no + // - _read method + // - _readableState object + // are there :( + if (!stream._read) { + stream._read = () => {} + stream._readableState = {} + } + return reply(stream).header('X-Stream-Output', '1') + }) + } +} + +exports.get = { + // uses common parseKey method that returns a `key` + parseArgs: exports.parseKey, + + // main route handler which is called after the above `parseArgs`, but only if the args were valid + handler: (request, reply) => { + const key = request.pre.args.key + const ipfs = request.server.app.ipfs + const pack = tar.pack() + + ipfs.files.getPull(key, (err, stream) => { + if (err) { + log.error(err) + + reply({ + Message: 'Failed to get file: ' + err, + Code: 0 + }).code(500) + return + } + + pull( + stream, + pull.asyncMap((file, cb) => { + const header = {name: file.path} + if (!file.content) { + header.type = 'directory' + pack.entry(header) + cb() + } else { + header.size = file.size + const packStream = pack.entry(header, cb) + if (!packStream) { + // this happens if the request is aborted + // we just skip things then + log('other side hung up') + return cb() + } + toStream.source(file.content).pipe(packStream) + } + }), + pull.onEnd((err) => { + if (err) { + log.error(err) + pack.emit('error', err) + pack.destroy() + return + } + + pack.finalize() + }) + ) + + // the reply must read the tar stream, + // to pull values through + reply(pack).header('X-Stream-Output', '1') + }) + } +} + +exports.add = { + handler: (request, reply) => { + if (!request.payload) { + return reply({ + Message: 'Array, Buffer, or String is required.', + code: 0 + }).code(400).takeover() + } + + const ipfs = request.server.app.ipfs + // TODO: make pull-multipart + const parser = multipart.reqParser(request.payload) + let filesParsed = false + + const fileAdder = pushable() + + parser.on('file', (fileName, fileStream) => { + const filePair = { + path: fileName, + content: toPull(fileStream) + } + filesParsed = true + fileAdder.push(filePair) + }) + + parser.on('directory', (directory) => { + fileAdder.push({ + path: directory, + content: '' + }) + }) + + parser.on('end', () => { + if (!filesParsed) { + return reply({ + Message: "File argument 'data' is required.", + code: 0 + }).code(400).takeover() + } + fileAdder.end() + }) + + pull( + fileAdder, + ipfs.files.createAddPullStream(), + pull.map((file) => { + return { + Name: file.path ? file.path : file.hash, + Hash: file.hash + } + }), + pull.map((file) => JSON.stringify(file) + EOL), + pull.collect((err, files) => { + if (err) { + return reply({ + Message: err, + Code: 0 + }).code(500) + } + + if (files.length === 0 && filesParsed) { + return reply({ + Message: 'Failed to add files.', + Code: 0 + }).code(500) + } + + reply(files.join('\n')) + .header('x-chunked-output', '1') + .header('content-type', 'application/json') + }) + ) + } +} + +// exports.gateway = { +// checkHash: (request, reply) => { +// if (!request.params.hash) { +// return reply('Path Resolve error: path must contain at least one component').code(400).takeover() +// } +// +// return reply({ +// ref: `/ipfs/${request.params.hash}` +// }) +// }, +// handler: (request, reply) => { +// const ref = request.pre.args.ref +// const ipfs = request.server.app.ipfs +// +// return GatewayResolver +// .resolveMultihash(ipfs, ref) +// .then((data) => { +// ipfs +// .files +// .cat(data.multihash) +// .then((stream) => { +// if (ref.endsWith('/')) { +// // remove trailing slash for files +// return reply +// .redirect(PathUtils.removeTrailingSlash(ref)) +// .permanent(true) +// } else { +// if (!stream._read) { +// stream._read = () => {} +// stream._readableState = {} +// } +// // response.continue() +// let filetypeChecked = false +// let stream2 = new Stream.PassThrough({highWaterMark: 1}) +// let response = reply(stream2).hold() +// +// pull( +// toPull.source(stream), +// pull.drain((chunk) => { +// // Check file type. do this once. +// if (chunk.length > 0 && !filetypeChecked) { +// console.log('got first chunk') +// let fileSignature = fileType(chunk) +// console.log('file type: ', fileSignature) +// +// filetypeChecked = true +// const mimeType = mime.lookup((fileSignature) ? fileSignature.ext : null) +// console.log('ref ', ref) +// console.log('mime-type ', mimeType) +// +// if (mimeType) { +// console.log('writing mimeType') +// +// response +// .header('Content-Type', mime.contentType(mimeType)) +// .header('Access-Control-Allow-Headers', 'X-Stream-Output, X-Chunked-Ouput') +// .header('Access-Control-Allow-Methods', 'GET') +// .header('Access-Control-Allow-Origin', '*') +// .header('Access-Control-Expose-Headers', 'X-Stream-Output, X-Chunked-Ouput') +// .send() +// } else { +// response +// .header('Access-Control-Allow-Headers', 'X-Stream-Output, X-Chunked-Ouput') +// .header('Access-Control-Allow-Methods', 'GET') +// .header('Access-Control-Allow-Origin', '*') +// .header('Access-Control-Expose-Headers', 'X-Stream-Output, X-Chunked-Ouput') +// .send() +// } +// } +// +// stream2.write(chunk) +// }, (err) => { +// if (err) throw err +// console.log('stream ended.') +// stream2.end() +// }) +// ) +// } +// }) +// .catch((err) => { +// if (err) { +// log.error(err) +// return reply(err.toString()).code(500) +// } +// }) +// }).catch((err) => { +// console.log('err: ', err.toString(), ' fileName: ', err.fileName) +// +// const errorToString = err.toString() +// if (errorToString === 'Error: This dag node is a directory') { +// return GatewayResolver +// .resolveDirectory(ipfs, ref, err.fileName) +// .then((data) => { +// if (typeof data === 'string') { +// // no index file found +// if (!ref.endsWith('/')) { +// // for a directory, if URL doesn't end with a / +// // append / and redirect permanent to that URL +// return reply.redirect(`${ref}/`).permanent(true) +// } else { +// // send directory listing +// return reply(data) +// } +// } else { +// // found index file +// // redirect to URL/ +// return reply.redirect(PathUtils.joinURLParts(ref, data[0].name)) +// } +// }).catch((err) => { +// log.error(err) +// return reply(err.toString()).code(500) +// }) +// } else if (errorToString.startsWith('Error: no link named')) { +// return reply(errorToString).code(404) +// } else if (errorToString.startsWith('Error: multihash length inconsistent') || +// errorToString.startsWith('Error: Non-base58 character')) { +// return reply(errorToString).code(400) +// } else { +// log.error(err) +// return reply(errorToString).code(500) +// } +// }) +// } +// } diff --git a/src/http-api/resources/id.js b/src/http/api/resources/id.js similarity index 100% rename from src/http-api/resources/id.js rename to src/http/api/resources/id.js diff --git a/src/http-api/resources/index.js b/src/http/api/resources/index.js similarity index 100% rename from src/http-api/resources/index.js rename to src/http/api/resources/index.js diff --git a/src/http-api/resources/object.js b/src/http/api/resources/object.js similarity index 100% rename from src/http-api/resources/object.js rename to src/http/api/resources/object.js diff --git a/src/http-api/resources/pubsub.js b/src/http/api/resources/pubsub.js similarity index 100% rename from src/http-api/resources/pubsub.js rename to src/http/api/resources/pubsub.js diff --git a/src/http-api/resources/repo.js b/src/http/api/resources/repo.js similarity index 100% rename from src/http-api/resources/repo.js rename to src/http/api/resources/repo.js diff --git a/src/http-api/resources/swarm.js b/src/http/api/resources/swarm.js similarity index 100% rename from src/http-api/resources/swarm.js rename to src/http/api/resources/swarm.js diff --git a/src/http-api/resources/version.js b/src/http/api/resources/version.js similarity index 100% rename from src/http-api/resources/version.js rename to src/http/api/resources/version.js diff --git a/src/http-api/routes/bitswap.js b/src/http/api/routes/bitswap.js similarity index 100% rename from src/http-api/routes/bitswap.js rename to src/http/api/routes/bitswap.js diff --git a/src/http-api/routes/block.js b/src/http/api/routes/block.js similarity index 100% rename from src/http-api/routes/block.js rename to src/http/api/routes/block.js diff --git a/src/http-api/routes/bootstrap.js b/src/http/api/routes/bootstrap.js similarity index 100% rename from src/http-api/routes/bootstrap.js rename to src/http/api/routes/bootstrap.js diff --git a/src/http-api/routes/config.js b/src/http/api/routes/config.js similarity index 100% rename from src/http-api/routes/config.js rename to src/http/api/routes/config.js diff --git a/src/http-api/routes/files.js b/src/http/api/routes/files.js similarity index 75% rename from src/http-api/routes/files.js rename to src/http/api/routes/files.js index 6328ef2faa..da57b3f2f1 100644 --- a/src/http-api/routes/files.js +++ b/src/http/api/routes/files.js @@ -4,7 +4,6 @@ const resources = require('./../resources') module.exports = (server) => { const api = server.select('API') - const gateway = server.select('Gateway') api.route({ // TODO fix method @@ -42,15 +41,4 @@ module.exports = (server) => { handler: resources.files.add.handler } }) - - gateway.route({ - method: '*', - path: '/ipfs/{hash*}', - config: { - pre: [ - { method: resources.files.gateway.checkHash, assign: 'args' } - ], - handler: resources.files.gateway.handler - } - }) } diff --git a/src/http-api/routes/id.js b/src/http/api/routes/id.js similarity index 100% rename from src/http-api/routes/id.js rename to src/http/api/routes/id.js diff --git a/src/http-api/routes/index.js b/src/http/api/routes/index.js similarity index 100% rename from src/http-api/routes/index.js rename to src/http/api/routes/index.js diff --git a/src/http-api/routes/object.js b/src/http/api/routes/object.js similarity index 100% rename from src/http-api/routes/object.js rename to src/http/api/routes/object.js diff --git a/src/http-api/routes/pubsub.js b/src/http/api/routes/pubsub.js similarity index 100% rename from src/http-api/routes/pubsub.js rename to src/http/api/routes/pubsub.js diff --git a/src/http-api/routes/repo.js b/src/http/api/routes/repo.js similarity index 100% rename from src/http-api/routes/repo.js rename to src/http/api/routes/repo.js diff --git a/src/http-api/routes/swarm.js b/src/http/api/routes/swarm.js similarity index 100% rename from src/http-api/routes/swarm.js rename to src/http/api/routes/swarm.js diff --git a/src/http-api/routes/version.js b/src/http/api/routes/version.js similarity index 100% rename from src/http-api/routes/version.js rename to src/http/api/routes/version.js diff --git a/src/http-api/error-handler.js b/src/http/error-handler.js similarity index 100% rename from src/http-api/error-handler.js rename to src/http/error-handler.js diff --git a/src/http/gateway/resolver.js b/src/http/gateway/resolver.js new file mode 100644 index 0000000000..ae89956f06 --- /dev/null +++ b/src/http/gateway/resolver.js @@ -0,0 +1,114 @@ +'use strict' + +const mh = require('multihashes') +const promisify = require('promisify-es6') +const eachOfSeries = require('async/eachOfSeries') +const debug = require('debug') +const log = debug('jsipfs:http-gateway:resolver') +log.error = debug('jsipfs:http-gateway:resolver:error') + +const html = require('./utils/html') +const PathUtil = require('./utils/path') + +const INDEX_HTML_FILES = [ 'index.html', 'index.htm', 'index.shtml' ] + +const resolveDirectory = promisify((ipfs, path, multihash, callback) => { + if (!callback) { + callback = noop + } + + mh.validate(mh.fromB58String(multihash)) + + ipfs + .object + .get(multihash, { enc: 'base58' }) + .then((DAGNode) => { + const links = DAGNode.links + const indexFiles = links.filter((link) => INDEX_HTML_FILES.indexOf(link.name) !== -1) + + // found index file in links + if (indexFiles.length > 0) { + return callback(null, indexFiles) + } + + return callback(null, html.build(path, links)) + }) +}) + +const noop = function () {} + +const resolveMultihash = promisify((ipfs, path, callback) => { + if (!callback) { + callback = noop + } + + const parts = PathUtil.splitPath(path) + const partsLength = parts.length + + let currentMultihash = parts[0] + + eachOfSeries(parts, (multihash, currentIndex, next) => { + // throws error when invalid multihash is passed + mh.validate(mh.fromB58String(currentMultihash)) + log('currentMultihash: ', currentMultihash) + log('currentIndex: ', currentIndex, '/', partsLength) + + ipfs + .object + .get(currentMultihash, { enc: 'base58' }) + .then((DAGNode) => { + // log('DAGNode: ', DAGNode) + if (currentIndex === partsLength - 1) { + // leaf node + log('leaf node: ', currentMultihash) + // log('DAGNode: ', DAGNode.links) + + if (DAGNode.links && + DAGNode.links.length > 0 && + DAGNode.links[0].name.length > 0) { + // this is a directory. + let isDirErr = new Error('This dag node is a directory') + // add currentMultihash as a fileName so it can be used by resolveDirectory + isDirErr.fileName = currentMultihash + return next(isDirErr) + } + + next() + } else { + // find multihash of requested named-file + // in current DAGNode's links + let multihashOfNextFile + const nextFileName = parts[currentIndex + 1] + const links = DAGNode.links + + for (let link of links) { + if (link.name === nextFileName) { + // found multihash of requested named-file + multihashOfNextFile = mh.toB58String(link.multihash) + log('found multihash: ', multihashOfNextFile) + break + } + } + + if (!multihashOfNextFile) { + log.error(`no link named "${nextFileName}" under ${currentMultihash}`) + throw new Error(`no link named "${nextFileName}" under ${currentMultihash}`) + } + + currentMultihash = multihashOfNextFile + next() + } + }) + }, (err) => { + if (err) { + log.error(err) + return callback(err) + } + callback(null, {multihash: currentMultihash}) + }) +}) + +module.exports = { + resolveDirectory, + resolveMultihash +} diff --git a/src/http/gateway/resources/gateway.js b/src/http/gateway/resources/gateway.js new file mode 100644 index 0000000000..a3733b9e94 --- /dev/null +++ b/src/http/gateway/resources/gateway.js @@ -0,0 +1,148 @@ +'use strict' + +// const mh = require('multihashes') +// const multipart = require('ipfs-multipart') +const debug = require('debug') +// const tar = require('tar-stream') +const log = debug('jsipfs:http-gateway') +log.error = debug('jsipfs:http-gateway:error') +const pull = require('pull-stream') +const toPull = require('stream-to-pull-stream') +// const pushable = require('pull-pushable') +// const EOL = require('os').EOL +// const toStream = require('pull-stream-to-stream') +const fileType = require('file-type') +const mime = require('mime-types') +const GatewayResolver = require('../resolver') +const PathUtils = require('../utils/path') +const Stream = require('stream') + +exports = module.exports + +module.exports = { + checkHash: (request, reply) => { + if (!request.params.hash) { + return reply({ + Message: 'Path Resolve error: path must contain at least one component', + Code: 0 + }).code(400).takeover() + } + + return reply({ + ref: `/ipfs/${request.params.hash}` + }) + }, + handler: (request, reply) => { + const ref = request.pre.args.ref + const ipfs = request.server.app.ipfs + + return GatewayResolver + .resolveMultihash(ipfs, ref) + .then((data) => { + ipfs + .files + .cat(data.multihash) + .then((stream) => { + if (ref.endsWith('/')) { + // remove trailing slash for files + return reply + .redirect(PathUtils.removeTrailingSlash(ref)) + .permanent(true) + } else { + if (!stream._read) { + stream._read = () => {} + stream._readableState = {} + } + // response.continue() + let filetypeChecked = false + let stream2 = new Stream.PassThrough({highWaterMark: 1}) + let response = reply(stream2).hold() + + pull( + toPull.source(stream), + pull.drain((chunk) => { + // Check file type. do this once. + if (chunk.length > 0 && !filetypeChecked) { + log('got first chunk') + let fileSignature = fileType(chunk) + log('file type: ', fileSignature) + + filetypeChecked = true + const mimeType = mime.lookup((fileSignature) ? fileSignature.ext : null) + log('ref ', ref) + log('mime-type ', mimeType) + + if (mimeType) { + log('writing mimeType') + + response + .header('Content-Type', mime.contentType(mimeType)) + .header('Access-Control-Allow-Headers', 'X-Stream-Output, X-Chunked-Ouput') + .header('Access-Control-Allow-Methods', 'GET') + .header('Access-Control-Allow-Origin', '*') + .header('Access-Control-Expose-Headers', 'X-Stream-Output, X-Chunked-Ouput') + .send() + } else { + response + .header('Access-Control-Allow-Headers', 'X-Stream-Output, X-Chunked-Ouput') + .header('Access-Control-Allow-Methods', 'GET') + .header('Access-Control-Allow-Origin', '*') + .header('Access-Control-Expose-Headers', 'X-Stream-Output, X-Chunked-Ouput') + .send() + } + } + + stream2.write(chunk) + }, (err) => { + if (err) throw err + log('stream ended.') + stream2.end() + }) + ) + } + }) + .catch((err) => { + if (err) { + log.error(err) + return reply(err.toString()).code(500) + } + }) + }).catch((err) => { + log('err: ', err.toString(), ' fileName: ', err.fileName) + + const errorToString = err.toString() + if (errorToString === 'Error: This dag node is a directory') { + return GatewayResolver + .resolveDirectory(ipfs, ref, err.fileName) + .then((data) => { + if (typeof data === 'string') { + // no index file found + if (!ref.endsWith('/')) { + // for a directory, if URL doesn't end with a / + // append / and redirect permanent to that URL + return reply.redirect(`${ref}/`).permanent(true) + } else { + // send directory listing + return reply(data) + } + } else { + // found index file + // redirect to URL/ + return reply.redirect(PathUtils.joinURLParts(ref, data[0].name)) + } + }).catch((err) => { + log.error(err) + return reply(err.toString()).code(500) + }) + } else if (errorToString.startsWith('Error: no link named')) { + return reply(errorToString).code(404) + } else if (errorToString.startsWith('Error: multihash length inconsistent') || + errorToString.startsWith('Error: Non-base58 character')) { + return reply({Message: errorToString, code: 0}).code(400) + } else { + log.error(err) + return reply({Message: errorToString, code: 0}).code(500) + } + }) + } +} diff --git a/src/http/gateway/resources/index.js b/src/http/gateway/resources/index.js new file mode 100644 index 0000000000..03f9d0901d --- /dev/null +++ b/src/http/gateway/resources/index.js @@ -0,0 +1,3 @@ +'use strict' + +exports.gateway = require('./gateway') diff --git a/src/http/gateway/routes/gateway.js b/src/http/gateway/routes/gateway.js new file mode 100644 index 0000000000..e1c0f3222f --- /dev/null +++ b/src/http/gateway/routes/gateway.js @@ -0,0 +1,18 @@ +'use strict' + +const resources = require('../resources') + +module.exports = (server) => { + const gateway = server.select('Gateway') + + gateway.route({ + method: '*', + path: '/ipfs/{hash*}', + config: { + pre: [ + { method: resources.gateway.checkHash, assign: 'args' } + ], + handler: resources.gateway.handler + } + }) +} diff --git a/src/http/gateway/routes/index.js b/src/http/gateway/routes/index.js new file mode 100644 index 0000000000..0e0656c258 --- /dev/null +++ b/src/http/gateway/routes/index.js @@ -0,0 +1,5 @@ +'use strict' + +module.exports = (server) => { + require('./gateway')(server) +} diff --git a/src/http-api/gateway/utils/html.js b/src/http/gateway/utils/html.js similarity index 100% rename from src/http-api/gateway/utils/html.js rename to src/http/gateway/utils/html.js diff --git a/src/http-api/gateway/utils/path.js b/src/http/gateway/utils/path.js similarity index 100% rename from src/http-api/gateway/utils/path.js rename to src/http/gateway/utils/path.js diff --git a/src/http-api/gateway/utils/style.js b/src/http/gateway/utils/style.js similarity index 100% rename from src/http-api/gateway/utils/style.js rename to src/http/gateway/utils/style.js diff --git a/src/http-api/index.js b/src/http/index.js similarity index 97% rename from src/http-api/index.js rename to src/http/index.js index a0d849e9d2..8ad07ef558 100644 --- a/src/http-api/index.js +++ b/src/http/index.js @@ -106,7 +106,9 @@ function HttpApi (repo, config, cliArgs) { errorHandler(this, this.server) // load routes - require('./routes')(this.server) + require('./api/routes')(this.server) + // load gateway routes + require('./gateway/routes')(this.server) // Set default headers setHeader(this.server, diff --git a/test/cli/pubsub.js b/test/cli/pubsub.js index 951c6a85ba..703b83a5b0 100644 --- a/test/cli/pubsub.js +++ b/test/cli/pubsub.js @@ -8,7 +8,7 @@ const expect = chai.expect chai.use(dirtyChai) const delay = require('delay') const waterfall = require('async/waterfall') -const HttpAPI = require('../../src/http-api') +const HttpAPI = require('../../src/http') // TODO needs to use ipfs-factory-daemon const createTempNode = '' const repoPath = require('./index').repoPath diff --git a/test/gateway/index.js b/test/gateway/index.js new file mode 100644 index 0000000000..356dc45a2f --- /dev/null +++ b/test/gateway/index.js @@ -0,0 +1,55 @@ +/* eslint-env mocha */ +'use strict' + +const fs = require('fs') +const chai = require('chai') +const dirtyChai = require('dirty-chai') +const expect = chai.expect +chai.use(dirtyChai) +const API = require('../../src/http') +// const APIctl = require('ipfs-api') +const ncp = require('ncp').ncp +const path = require('path') +const clean = require('../utils/clean') + +describe('HTTP GATEWAY', () => { + const repoExample = path.join(__dirname, '../go-ipfs-repo') + const repoTests = path.join(__dirname, '../repo-tests-run') + + let http = {} + + before((done) => { + http.api = new API(repoTests) + + ncp(repoExample, repoTests, (err) => { + expect(err).to.not.exist() + + http.api.start(false, done) + }) + }) + + after((done) => { + http.api.stop((err) => { + expect(err).to.not.exist() + clean(repoTests) + done() + }) + }) + + describe('## http-gateway spec tests', () => { + fs.readdirSync(path.join(__dirname, '/spec')) + .forEach((file) => require('./spec/' + file)(http)) + }) + + // describe('## interface tests', () => { + // fs.readdirSync(path.join(__dirname, '/interface')) + // .forEach((file) => require('./interface/' + file)) + // }) + // + // describe('## custom ipfs-api tests', () => { + // const ctl = APIctl('/ip4/127.0.0.1/tcp/6001') + // + // fs.readdirSync(path.join(__dirname, '/over-ipfs-api')) + // .forEach((file) => require('./over-ipfs-api/' + file)(ctl)) + // }) +}) diff --git a/test/gateway/spec/gateway.js b/test/gateway/spec/gateway.js new file mode 100644 index 0000000000..822c67a498 --- /dev/null +++ b/test/gateway/spec/gateway.js @@ -0,0 +1,50 @@ +/* eslint-env mocha */ +'use strict' + +const expect = require('chai').expect + +module.exports = (http) => { + describe('/files', () => { + let gateway + + before(() => { + gateway = http.api.server.select('Gateway') + }) + + describe('/ipfs', () => { + it('returns 400 for request without argument', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs' + }, (res) => { + expect(res.statusCode).to.equal(400) + expect(res.result.Message).to.be.a('string') + done() + }) + }) + + it('400 for request with invalid argument', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs/invalid' + }, (res) => { + expect(res.statusCode).to.equal(400) + expect(res.result.Message).to.be.a('string') + done() + }) + }) + + it('valid hash', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs/QmT78zSuBmuS4z925WZfrqQ1qHaJ56DQaTfyMUF7F8ff5o' + }, (res) => { + expect(res.statusCode).to.equal(200) + expect(res.rawPayload).to.deep.equal(new Buffer('hello world' + '\n')) + expect(res.payload).to.equal('hello world' + '\n') + done() + }) + }) + }) + }) +} diff --git a/test/http-api/index.js b/test/http-api/index.js index ad647e92df..16e535322e 100644 --- a/test/http-api/index.js +++ b/test/http-api/index.js @@ -6,7 +6,7 @@ const chai = require('chai') const dirtyChai = require('dirty-chai') const expect = chai.expect chai.use(dirtyChai) -const API = require('../../src/http-api') +const API = require('../../src/http') const APIctl = require('ipfs-api') const ncp = require('ncp').ncp const path = require('path') diff --git a/test/interop/daemons/js.js b/test/interop/daemons/js.js index 66b81975ff..4b4089d494 100644 --- a/test/interop/daemons/js.js +++ b/test/interop/daemons/js.js @@ -6,7 +6,7 @@ const series = require('async/series') const rimraf = require('rimraf') const tmpDir = require('../util').tmpDir -const HttpApi = require('../../../src/http-api') +const HttpApi = require('../../../src/http') function portConfig (port) { port = port + 5 diff --git a/test/node.js b/test/node.js index 22872e6d02..fd97536cf0 100644 --- a/test/node.js +++ b/test/node.js @@ -3,6 +3,7 @@ let testCore = true let testHTTP = true let testCLI = true +let testGatway = true if (process.env.TEST) { switch (process.env.TEST) { @@ -14,6 +15,11 @@ if (process.env.TEST) { testCore = false testCLI = false break + case 'gateway': + testCore = false + testCLI = false + testHTTP = false + break case 'cli': testCore = false testHTTP = false @@ -34,3 +40,7 @@ if (testHTTP) { if (testCLI) { require('./cli') } + +if (testGatway) { + require('./gateway') +} diff --git a/test/utils/ipfs-factory-daemon/index.js b/test/utils/ipfs-factory-daemon/index.js index f9606a7414..37f208cb29 100644 --- a/test/utils/ipfs-factory-daemon/index.js +++ b/test/utils/ipfs-factory-daemon/index.js @@ -3,7 +3,7 @@ const PeerId = require('peer-id') const IPFSAPI = require('ipfs-api') const clean = require('../clean') -const HttpApi = require('../../../src/http-api') +const HttpApi = require('../../../src/http') const series = require('async/series') const eachSeries = require('async/eachSeries') const defaultConfig = require('./default-config.json') From cd0fae9c5b56ceeb19dd179311b37bfbc073302f Mon Sep 17 00:00:00 2001 From: Yahya Date: Fri, 1 Sep 2017 10:27:39 +0200 Subject: [PATCH 03/14] clean up, remove commented out lines. (#971) * clean up, remove commented out lines. * cleaning code, removing commented out blocks. * gateway initial tests. --- src/http/api/resources/files.js | 130 -------------------------------- test/gateway/index.js | 57 +++++++++----- test/gateway/spec/gateway.js | 50 ------------ 3 files changed, 39 insertions(+), 198 deletions(-) delete mode 100644 test/gateway/spec/gateway.js diff --git a/src/http/api/resources/files.js b/src/http/api/resources/files.js index 1ea7cb0099..68ecc797bd 100644 --- a/src/http/api/resources/files.js +++ b/src/http/api/resources/files.js @@ -11,11 +11,6 @@ const toPull = require('stream-to-pull-stream') const pushable = require('pull-pushable') const EOL = require('os').EOL const toStream = require('pull-stream-to-stream') -// const fileType = require('file-type') -// const mime = require('mime-types') -// const GatewayResolver = require('../gateway/resolver') -// const PathUtils = require('../gateway/utils/path') -// const Stream = require('stream') exports = module.exports @@ -218,128 +213,3 @@ exports.add = { ) } } - -// exports.gateway = { -// checkHash: (request, reply) => { -// if (!request.params.hash) { -// return reply('Path Resolve error: path must contain at least one component').code(400).takeover() -// } -// -// return reply({ -// ref: `/ipfs/${request.params.hash}` -// }) -// }, -// handler: (request, reply) => { -// const ref = request.pre.args.ref -// const ipfs = request.server.app.ipfs -// -// return GatewayResolver -// .resolveMultihash(ipfs, ref) -// .then((data) => { -// ipfs -// .files -// .cat(data.multihash) -// .then((stream) => { -// if (ref.endsWith('/')) { -// // remove trailing slash for files -// return reply -// .redirect(PathUtils.removeTrailingSlash(ref)) -// .permanent(true) -// } else { -// if (!stream._read) { -// stream._read = () => {} -// stream._readableState = {} -// } -// // response.continue() -// let filetypeChecked = false -// let stream2 = new Stream.PassThrough({highWaterMark: 1}) -// let response = reply(stream2).hold() -// -// pull( -// toPull.source(stream), -// pull.drain((chunk) => { -// // Check file type. do this once. -// if (chunk.length > 0 && !filetypeChecked) { -// console.log('got first chunk') -// let fileSignature = fileType(chunk) -// console.log('file type: ', fileSignature) -// -// filetypeChecked = true -// const mimeType = mime.lookup((fileSignature) ? fileSignature.ext : null) -// console.log('ref ', ref) -// console.log('mime-type ', mimeType) -// -// if (mimeType) { -// console.log('writing mimeType') -// -// response -// .header('Content-Type', mime.contentType(mimeType)) -// .header('Access-Control-Allow-Headers', 'X-Stream-Output, X-Chunked-Ouput') -// .header('Access-Control-Allow-Methods', 'GET') -// .header('Access-Control-Allow-Origin', '*') -// .header('Access-Control-Expose-Headers', 'X-Stream-Output, X-Chunked-Ouput') -// .send() -// } else { -// response -// .header('Access-Control-Allow-Headers', 'X-Stream-Output, X-Chunked-Ouput') -// .header('Access-Control-Allow-Methods', 'GET') -// .header('Access-Control-Allow-Origin', '*') -// .header('Access-Control-Expose-Headers', 'X-Stream-Output, X-Chunked-Ouput') -// .send() -// } -// } -// -// stream2.write(chunk) -// }, (err) => { -// if (err) throw err -// console.log('stream ended.') -// stream2.end() -// }) -// ) -// } -// }) -// .catch((err) => { -// if (err) { -// log.error(err) -// return reply(err.toString()).code(500) -// } -// }) -// }).catch((err) => { -// console.log('err: ', err.toString(), ' fileName: ', err.fileName) -// -// const errorToString = err.toString() -// if (errorToString === 'Error: This dag node is a directory') { -// return GatewayResolver -// .resolveDirectory(ipfs, ref, err.fileName) -// .then((data) => { -// if (typeof data === 'string') { -// // no index file found -// if (!ref.endsWith('/')) { -// // for a directory, if URL doesn't end with a / -// // append / and redirect permanent to that URL -// return reply.redirect(`${ref}/`).permanent(true) -// } else { -// // send directory listing -// return reply(data) -// } -// } else { -// // found index file -// // redirect to URL/ -// return reply.redirect(PathUtils.joinURLParts(ref, data[0].name)) -// } -// }).catch((err) => { -// log.error(err) -// return reply(err.toString()).code(500) -// }) -// } else if (errorToString.startsWith('Error: no link named')) { -// return reply(errorToString).code(404) -// } else if (errorToString.startsWith('Error: multihash length inconsistent') || -// errorToString.startsWith('Error: Non-base58 character')) { -// return reply(errorToString).code(400) -// } else { -// log.error(err) -// return reply(errorToString).code(500) -// } -// }) -// } -// } diff --git a/test/gateway/index.js b/test/gateway/index.js index 356dc45a2f..afa9cb70d9 100644 --- a/test/gateway/index.js +++ b/test/gateway/index.js @@ -1,13 +1,11 @@ /* eslint-env mocha */ 'use strict' -const fs = require('fs') const chai = require('chai') const dirtyChai = require('dirty-chai') const expect = chai.expect chai.use(dirtyChai) const API = require('../../src/http') -// const APIctl = require('ipfs-api') const ncp = require('ncp').ncp const path = require('path') const clean = require('../utils/clean') @@ -17,6 +15,7 @@ describe('HTTP GATEWAY', () => { const repoTests = path.join(__dirname, '../repo-tests-run') let http = {} + let gateway before((done) => { http.api = new API(repoTests) @@ -24,7 +23,10 @@ describe('HTTP GATEWAY', () => { ncp(repoExample, repoTests, (err) => { expect(err).to.not.exist() - http.api.start(false, done) + http.api.start(false, () => { + gateway = http.api.server.select('Gateway') + done() + }) }) }) @@ -36,20 +38,39 @@ describe('HTTP GATEWAY', () => { }) }) - describe('## http-gateway spec tests', () => { - fs.readdirSync(path.join(__dirname, '/spec')) - .forEach((file) => require('./spec/' + file)(http)) - }) + describe('/ipfs/* route', () => { + it('returns 400 for request without argument', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs' + }, (res) => { + expect(res.statusCode).to.equal(400) + expect(res.result.Message).to.be.a('string') + done() + }) + }) + + it('400 for request with invalid argument', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs/invalid' + }, (res) => { + expect(res.statusCode).to.equal(400) + expect(res.result.Message).to.be.a('string') + done() + }) + }) - // describe('## interface tests', () => { - // fs.readdirSync(path.join(__dirname, '/interface')) - // .forEach((file) => require('./interface/' + file)) - // }) - // - // describe('## custom ipfs-api tests', () => { - // const ctl = APIctl('/ip4/127.0.0.1/tcp/6001') - // - // fs.readdirSync(path.join(__dirname, '/over-ipfs-api')) - // .forEach((file) => require('./over-ipfs-api/' + file)(ctl)) - // }) + it('valid hash', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs/QmT78zSuBmuS4z925WZfrqQ1qHaJ56DQaTfyMUF7F8ff5o' + }, (res) => { + expect(res.statusCode).to.equal(200) + expect(res.rawPayload).to.deep.equal(new Buffer('hello world' + '\n')) + expect(res.payload).to.equal('hello world' + '\n') + done() + }) + }) + }) }) diff --git a/test/gateway/spec/gateway.js b/test/gateway/spec/gateway.js deleted file mode 100644 index 822c67a498..0000000000 --- a/test/gateway/spec/gateway.js +++ /dev/null @@ -1,50 +0,0 @@ -/* eslint-env mocha */ -'use strict' - -const expect = require('chai').expect - -module.exports = (http) => { - describe('/files', () => { - let gateway - - before(() => { - gateway = http.api.server.select('Gateway') - }) - - describe('/ipfs', () => { - it('returns 400 for request without argument', (done) => { - gateway.inject({ - method: 'GET', - url: '/ipfs' - }, (res) => { - expect(res.statusCode).to.equal(400) - expect(res.result.Message).to.be.a('string') - done() - }) - }) - - it('400 for request with invalid argument', (done) => { - gateway.inject({ - method: 'GET', - url: '/ipfs/invalid' - }, (res) => { - expect(res.statusCode).to.equal(400) - expect(res.result.Message).to.be.a('string') - done() - }) - }) - - it('valid hash', (done) => { - gateway.inject({ - method: 'GET', - url: '/ipfs/QmT78zSuBmuS4z925WZfrqQ1qHaJ56DQaTfyMUF7F8ff5o' - }, (res) => { - expect(res.statusCode).to.equal(200) - expect(res.rawPayload).to.deep.equal(new Buffer('hello world' + '\n')) - expect(res.payload).to.equal('hello world' + '\n') - done() - }) - }) - }) - }) -} From ee34c8c2f8b106d841ccf016a8b272db4c0b86d0 Mon Sep 17 00:00:00 2001 From: Yahya Date: Sat, 2 Sep 2017 23:59:24 +0200 Subject: [PATCH 04/14] clean up , working tests on node v8.4.0 License: MIT Signed-off-by: Yahya --- package.json | 1 - src/http/gateway/resources/gateway.js | 8 -------- src/http/gateway/resources/index.js | 4 +++- 3 files changed, 3 insertions(+), 10 deletions(-) diff --git a/package.json b/package.json index 34004a2e80..0eb0591952 100644 --- a/package.json +++ b/package.json @@ -137,7 +137,6 @@ "multihashes": "~0.4.5", "once": "^1.4.0", "path-exists": "^3.0.0", - "promised-for": "^1.0.0", "peer-book": "^0.5.0", "peer-id": "^0.9.0", "peer-info": "^0.10.0", diff --git a/src/http/gateway/resources/gateway.js b/src/http/gateway/resources/gateway.js index a3733b9e94..37ccba22b4 100644 --- a/src/http/gateway/resources/gateway.js +++ b/src/http/gateway/resources/gateway.js @@ -1,24 +1,16 @@ 'use strict' -// const mh = require('multihashes') -// const multipart = require('ipfs-multipart') const debug = require('debug') -// const tar = require('tar-stream') const log = debug('jsipfs:http-gateway') log.error = debug('jsipfs:http-gateway:error') const pull = require('pull-stream') const toPull = require('stream-to-pull-stream') -// const pushable = require('pull-pushable') -// const EOL = require('os').EOL -// const toStream = require('pull-stream-to-stream') const fileType = require('file-type') const mime = require('mime-types') const GatewayResolver = require('../resolver') const PathUtils = require('../utils/path') const Stream = require('stream') -exports = module.exports - module.exports = { checkHash: (request, reply) => { if (!request.params.hash) { diff --git a/src/http/gateway/resources/index.js b/src/http/gateway/resources/index.js index 03f9d0901d..5a464ce193 100644 --- a/src/http/gateway/resources/index.js +++ b/src/http/gateway/resources/index.js @@ -1,3 +1,5 @@ 'use strict' -exports.gateway = require('./gateway') +module.exports = { + gateway: require('./gateway') +} From 6b42bbf7366d0fa8df2bcc6e0b988369399dd8a1 Mon Sep 17 00:00:00 2001 From: Harsh Vakharia Date: Tue, 29 Aug 2017 12:53:15 +0530 Subject: [PATCH 05/14] feat: add HTTP Gateway to the js-ipfs daemon --- gulpfile.js | 2 +- package.json | 6 +- src/cli/commands/daemon.js | 2 +- .../api}/resources/bitswap.js | 0 src/{http-api => http/api}/resources/block.js | 0 .../api}/resources/bootstrap.js | 0 .../api}/resources/config.js | 0 src/{http-api => http/api}/resources/files.js | 0 src/{http-api => http/api}/resources/id.js | 0 src/{http-api => http/api}/resources/index.js | 0 .../api}/resources/object.js | 0 .../api}/resources/pubsub.js | 0 src/{http-api => http/api}/resources/repo.js | 0 src/{http-api => http/api}/resources/swarm.js | 0 .../api}/resources/version.js | 0 src/{http-api => http/api}/routes/bitswap.js | 0 src/{http-api => http/api}/routes/block.js | 0 .../api}/routes/bootstrap.js | 0 src/{http-api => http/api}/routes/config.js | 0 src/{http-api => http/api}/routes/files.js | 0 src/{http-api => http/api}/routes/id.js | 0 src/{http-api => http/api}/routes/index.js | 0 src/{http-api => http/api}/routes/object.js | 0 src/{http-api => http/api}/routes/pubsub.js | 0 src/{http-api => http/api}/routes/repo.js | 0 src/{http-api => http/api}/routes/swarm.js | 0 src/{http-api => http/api}/routes/version.js | 0 src/{http-api => http}/error-handler.js | 0 src/http/gateway/resolver.js | 114 ++++++++++++++ src/http/gateway/resources/gateway.js | 140 ++++++++++++++++++ src/http/gateway/resources/index.js | 5 + src/http/gateway/routes/gateway.js | 18 +++ src/http/gateway/routes/index.js | 5 + src/http/gateway/utils/html.js | 83 +++++++++++ src/http/gateway/utils/path.js | 35 +++++ src/http/gateway/utils/style.js | 16 ++ src/{http-api => http}/index.js | 4 +- test/cli/pubsub.js | 2 +- test/gateway/index.js | 76 ++++++++++ test/http-api/index.js | 2 +- test/interop/daemons/js.js | 2 +- test/node.js | 10 ++ test/utils/ipfs-factory-daemon/index.js | 2 +- 43 files changed, 516 insertions(+), 8 deletions(-) rename src/{http-api => http/api}/resources/bitswap.js (100%) rename src/{http-api => http/api}/resources/block.js (100%) rename src/{http-api => http/api}/resources/bootstrap.js (100%) rename src/{http-api => http/api}/resources/config.js (100%) rename src/{http-api => http/api}/resources/files.js (100%) rename src/{http-api => http/api}/resources/id.js (100%) rename src/{http-api => http/api}/resources/index.js (100%) rename src/{http-api => http/api}/resources/object.js (100%) rename src/{http-api => http/api}/resources/pubsub.js (100%) rename src/{http-api => http/api}/resources/repo.js (100%) rename src/{http-api => http/api}/resources/swarm.js (100%) rename src/{http-api => http/api}/resources/version.js (100%) rename src/{http-api => http/api}/routes/bitswap.js (100%) rename src/{http-api => http/api}/routes/block.js (100%) rename src/{http-api => http/api}/routes/bootstrap.js (100%) rename src/{http-api => http/api}/routes/config.js (100%) rename src/{http-api => http/api}/routes/files.js (100%) rename src/{http-api => http/api}/routes/id.js (100%) rename src/{http-api => http/api}/routes/index.js (100%) rename src/{http-api => http/api}/routes/object.js (100%) rename src/{http-api => http/api}/routes/pubsub.js (100%) rename src/{http-api => http/api}/routes/repo.js (100%) rename src/{http-api => http/api}/routes/swarm.js (100%) rename src/{http-api => http/api}/routes/version.js (100%) rename src/{http-api => http}/error-handler.js (100%) create mode 100644 src/http/gateway/resolver.js create mode 100644 src/http/gateway/resources/gateway.js create mode 100644 src/http/gateway/resources/index.js create mode 100644 src/http/gateway/routes/gateway.js create mode 100644 src/http/gateway/routes/index.js create mode 100644 src/http/gateway/utils/html.js create mode 100644 src/http/gateway/utils/path.js create mode 100644 src/http/gateway/utils/style.js rename src/{http-api => http}/index.js (97%) create mode 100644 test/gateway/index.js diff --git a/gulpfile.js b/gulpfile.js index 2be23a3e2b..26304b51a1 100644 --- a/gulpfile.js +++ b/gulpfile.js @@ -4,7 +4,7 @@ const gulp = require('gulp') const parallel = require('async/parallel') const series = require('async/series') const createTempRepo = require('./test/utils/create-repo-nodejs.js') -const HTTPAPI = require('./src/http-api') +const HTTPAPI = require('./src/http') const leftPad = require('left-pad') let nodes = [] diff --git a/package.json b/package.json index 5e0c8f8669..561a758f26 100644 --- a/package.json +++ b/package.json @@ -27,6 +27,7 @@ "test:unit:node": "gulp test:node", "test:unit:node:core": "TEST=core npm run test:unit:node", "test:unit:node:http": "TEST=http npm run test:unit:node", + "test:unit:node:gateway": "TEST=gateway npm run test:unit:node", "test:unit:node:cli": "TEST=cli npm run test:unit:node", "test:unit:browser": "gulp test:browser", "test:interop": "npm run test:interop:node", @@ -92,8 +93,10 @@ "async": "^2.5.0", "bl": "^1.2.1", "boom": "^5.2.0", - "cids": "~0.5.1", "debug": "^3.0.1", + "cids": "^0.5.1", + "file-type": "^6.1.0", + "filesize": "^3.5.10", "fsm-event": "^2.1.0", "glob": "^7.1.2", "hapi": "^16.5.2", @@ -126,6 +129,7 @@ "lodash.sortby": "^4.7.0", "lodash.values": "^4.3.0", "mafmt": "^3.0.0", + "mime-types": "^2.1.16", "mkdirp": "~0.5.1", "multiaddr": "^3.0.0", "multihashes": "~0.4.9", diff --git a/src/cli/commands/daemon.js b/src/cli/commands/daemon.js index 0414ca9654..4529e2af54 100644 --- a/src/cli/commands/daemon.js +++ b/src/cli/commands/daemon.js @@ -1,6 +1,6 @@ 'use strict' -const HttpAPI = require('../../http-api') +const HttpAPI = require('../../http') const utils = require('../utils') const print = utils.print diff --git a/src/http-api/resources/bitswap.js b/src/http/api/resources/bitswap.js similarity index 100% rename from src/http-api/resources/bitswap.js rename to src/http/api/resources/bitswap.js diff --git a/src/http-api/resources/block.js b/src/http/api/resources/block.js similarity index 100% rename from src/http-api/resources/block.js rename to src/http/api/resources/block.js diff --git a/src/http-api/resources/bootstrap.js b/src/http/api/resources/bootstrap.js similarity index 100% rename from src/http-api/resources/bootstrap.js rename to src/http/api/resources/bootstrap.js diff --git a/src/http-api/resources/config.js b/src/http/api/resources/config.js similarity index 100% rename from src/http-api/resources/config.js rename to src/http/api/resources/config.js diff --git a/src/http-api/resources/files.js b/src/http/api/resources/files.js similarity index 100% rename from src/http-api/resources/files.js rename to src/http/api/resources/files.js diff --git a/src/http-api/resources/id.js b/src/http/api/resources/id.js similarity index 100% rename from src/http-api/resources/id.js rename to src/http/api/resources/id.js diff --git a/src/http-api/resources/index.js b/src/http/api/resources/index.js similarity index 100% rename from src/http-api/resources/index.js rename to src/http/api/resources/index.js diff --git a/src/http-api/resources/object.js b/src/http/api/resources/object.js similarity index 100% rename from src/http-api/resources/object.js rename to src/http/api/resources/object.js diff --git a/src/http-api/resources/pubsub.js b/src/http/api/resources/pubsub.js similarity index 100% rename from src/http-api/resources/pubsub.js rename to src/http/api/resources/pubsub.js diff --git a/src/http-api/resources/repo.js b/src/http/api/resources/repo.js similarity index 100% rename from src/http-api/resources/repo.js rename to src/http/api/resources/repo.js diff --git a/src/http-api/resources/swarm.js b/src/http/api/resources/swarm.js similarity index 100% rename from src/http-api/resources/swarm.js rename to src/http/api/resources/swarm.js diff --git a/src/http-api/resources/version.js b/src/http/api/resources/version.js similarity index 100% rename from src/http-api/resources/version.js rename to src/http/api/resources/version.js diff --git a/src/http-api/routes/bitswap.js b/src/http/api/routes/bitswap.js similarity index 100% rename from src/http-api/routes/bitswap.js rename to src/http/api/routes/bitswap.js diff --git a/src/http-api/routes/block.js b/src/http/api/routes/block.js similarity index 100% rename from src/http-api/routes/block.js rename to src/http/api/routes/block.js diff --git a/src/http-api/routes/bootstrap.js b/src/http/api/routes/bootstrap.js similarity index 100% rename from src/http-api/routes/bootstrap.js rename to src/http/api/routes/bootstrap.js diff --git a/src/http-api/routes/config.js b/src/http/api/routes/config.js similarity index 100% rename from src/http-api/routes/config.js rename to src/http/api/routes/config.js diff --git a/src/http-api/routes/files.js b/src/http/api/routes/files.js similarity index 100% rename from src/http-api/routes/files.js rename to src/http/api/routes/files.js diff --git a/src/http-api/routes/id.js b/src/http/api/routes/id.js similarity index 100% rename from src/http-api/routes/id.js rename to src/http/api/routes/id.js diff --git a/src/http-api/routes/index.js b/src/http/api/routes/index.js similarity index 100% rename from src/http-api/routes/index.js rename to src/http/api/routes/index.js diff --git a/src/http-api/routes/object.js b/src/http/api/routes/object.js similarity index 100% rename from src/http-api/routes/object.js rename to src/http/api/routes/object.js diff --git a/src/http-api/routes/pubsub.js b/src/http/api/routes/pubsub.js similarity index 100% rename from src/http-api/routes/pubsub.js rename to src/http/api/routes/pubsub.js diff --git a/src/http-api/routes/repo.js b/src/http/api/routes/repo.js similarity index 100% rename from src/http-api/routes/repo.js rename to src/http/api/routes/repo.js diff --git a/src/http-api/routes/swarm.js b/src/http/api/routes/swarm.js similarity index 100% rename from src/http-api/routes/swarm.js rename to src/http/api/routes/swarm.js diff --git a/src/http-api/routes/version.js b/src/http/api/routes/version.js similarity index 100% rename from src/http-api/routes/version.js rename to src/http/api/routes/version.js diff --git a/src/http-api/error-handler.js b/src/http/error-handler.js similarity index 100% rename from src/http-api/error-handler.js rename to src/http/error-handler.js diff --git a/src/http/gateway/resolver.js b/src/http/gateway/resolver.js new file mode 100644 index 0000000000..ae89956f06 --- /dev/null +++ b/src/http/gateway/resolver.js @@ -0,0 +1,114 @@ +'use strict' + +const mh = require('multihashes') +const promisify = require('promisify-es6') +const eachOfSeries = require('async/eachOfSeries') +const debug = require('debug') +const log = debug('jsipfs:http-gateway:resolver') +log.error = debug('jsipfs:http-gateway:resolver:error') + +const html = require('./utils/html') +const PathUtil = require('./utils/path') + +const INDEX_HTML_FILES = [ 'index.html', 'index.htm', 'index.shtml' ] + +const resolveDirectory = promisify((ipfs, path, multihash, callback) => { + if (!callback) { + callback = noop + } + + mh.validate(mh.fromB58String(multihash)) + + ipfs + .object + .get(multihash, { enc: 'base58' }) + .then((DAGNode) => { + const links = DAGNode.links + const indexFiles = links.filter((link) => INDEX_HTML_FILES.indexOf(link.name) !== -1) + + // found index file in links + if (indexFiles.length > 0) { + return callback(null, indexFiles) + } + + return callback(null, html.build(path, links)) + }) +}) + +const noop = function () {} + +const resolveMultihash = promisify((ipfs, path, callback) => { + if (!callback) { + callback = noop + } + + const parts = PathUtil.splitPath(path) + const partsLength = parts.length + + let currentMultihash = parts[0] + + eachOfSeries(parts, (multihash, currentIndex, next) => { + // throws error when invalid multihash is passed + mh.validate(mh.fromB58String(currentMultihash)) + log('currentMultihash: ', currentMultihash) + log('currentIndex: ', currentIndex, '/', partsLength) + + ipfs + .object + .get(currentMultihash, { enc: 'base58' }) + .then((DAGNode) => { + // log('DAGNode: ', DAGNode) + if (currentIndex === partsLength - 1) { + // leaf node + log('leaf node: ', currentMultihash) + // log('DAGNode: ', DAGNode.links) + + if (DAGNode.links && + DAGNode.links.length > 0 && + DAGNode.links[0].name.length > 0) { + // this is a directory. + let isDirErr = new Error('This dag node is a directory') + // add currentMultihash as a fileName so it can be used by resolveDirectory + isDirErr.fileName = currentMultihash + return next(isDirErr) + } + + next() + } else { + // find multihash of requested named-file + // in current DAGNode's links + let multihashOfNextFile + const nextFileName = parts[currentIndex + 1] + const links = DAGNode.links + + for (let link of links) { + if (link.name === nextFileName) { + // found multihash of requested named-file + multihashOfNextFile = mh.toB58String(link.multihash) + log('found multihash: ', multihashOfNextFile) + break + } + } + + if (!multihashOfNextFile) { + log.error(`no link named "${nextFileName}" under ${currentMultihash}`) + throw new Error(`no link named "${nextFileName}" under ${currentMultihash}`) + } + + currentMultihash = multihashOfNextFile + next() + } + }) + }, (err) => { + if (err) { + log.error(err) + return callback(err) + } + callback(null, {multihash: currentMultihash}) + }) +}) + +module.exports = { + resolveDirectory, + resolveMultihash +} diff --git a/src/http/gateway/resources/gateway.js b/src/http/gateway/resources/gateway.js new file mode 100644 index 0000000000..37ccba22b4 --- /dev/null +++ b/src/http/gateway/resources/gateway.js @@ -0,0 +1,140 @@ +'use strict' + +const debug = require('debug') +const log = debug('jsipfs:http-gateway') +log.error = debug('jsipfs:http-gateway:error') +const pull = require('pull-stream') +const toPull = require('stream-to-pull-stream') +const fileType = require('file-type') +const mime = require('mime-types') +const GatewayResolver = require('../resolver') +const PathUtils = require('../utils/path') +const Stream = require('stream') + +module.exports = { + checkHash: (request, reply) => { + if (!request.params.hash) { + return reply({ + Message: 'Path Resolve error: path must contain at least one component', + Code: 0 + }).code(400).takeover() + } + + return reply({ + ref: `/ipfs/${request.params.hash}` + }) + }, + handler: (request, reply) => { + const ref = request.pre.args.ref + const ipfs = request.server.app.ipfs + + return GatewayResolver + .resolveMultihash(ipfs, ref) + .then((data) => { + ipfs + .files + .cat(data.multihash) + .then((stream) => { + if (ref.endsWith('/')) { + // remove trailing slash for files + return reply + .redirect(PathUtils.removeTrailingSlash(ref)) + .permanent(true) + } else { + if (!stream._read) { + stream._read = () => {} + stream._readableState = {} + } + // response.continue() + let filetypeChecked = false + let stream2 = new Stream.PassThrough({highWaterMark: 1}) + let response = reply(stream2).hold() + + pull( + toPull.source(stream), + pull.drain((chunk) => { + // Check file type. do this once. + if (chunk.length > 0 && !filetypeChecked) { + log('got first chunk') + let fileSignature = fileType(chunk) + log('file type: ', fileSignature) + + filetypeChecked = true + const mimeType = mime.lookup((fileSignature) ? fileSignature.ext : null) + log('ref ', ref) + log('mime-type ', mimeType) + + if (mimeType) { + log('writing mimeType') + + response + .header('Content-Type', mime.contentType(mimeType)) + .header('Access-Control-Allow-Headers', 'X-Stream-Output, X-Chunked-Ouput') + .header('Access-Control-Allow-Methods', 'GET') + .header('Access-Control-Allow-Origin', '*') + .header('Access-Control-Expose-Headers', 'X-Stream-Output, X-Chunked-Ouput') + .send() + } else { + response + .header('Access-Control-Allow-Headers', 'X-Stream-Output, X-Chunked-Ouput') + .header('Access-Control-Allow-Methods', 'GET') + .header('Access-Control-Allow-Origin', '*') + .header('Access-Control-Expose-Headers', 'X-Stream-Output, X-Chunked-Ouput') + .send() + } + } + + stream2.write(chunk) + }, (err) => { + if (err) throw err + log('stream ended.') + stream2.end() + }) + ) + } + }) + .catch((err) => { + if (err) { + log.error(err) + return reply(err.toString()).code(500) + } + }) + }).catch((err) => { + log('err: ', err.toString(), ' fileName: ', err.fileName) + + const errorToString = err.toString() + if (errorToString === 'Error: This dag node is a directory') { + return GatewayResolver + .resolveDirectory(ipfs, ref, err.fileName) + .then((data) => { + if (typeof data === 'string') { + // no index file found + if (!ref.endsWith('/')) { + // for a directory, if URL doesn't end with a / + // append / and redirect permanent to that URL + return reply.redirect(`${ref}/`).permanent(true) + } else { + // send directory listing + return reply(data) + } + } else { + // found index file + // redirect to URL/ + return reply.redirect(PathUtils.joinURLParts(ref, data[0].name)) + } + }).catch((err) => { + log.error(err) + return reply(err.toString()).code(500) + }) + } else if (errorToString.startsWith('Error: no link named')) { + return reply(errorToString).code(404) + } else if (errorToString.startsWith('Error: multihash length inconsistent') || + errorToString.startsWith('Error: Non-base58 character')) { + return reply({Message: errorToString, code: 0}).code(400) + } else { + log.error(err) + return reply({Message: errorToString, code: 0}).code(500) + } + }) + } +} diff --git a/src/http/gateway/resources/index.js b/src/http/gateway/resources/index.js new file mode 100644 index 0000000000..5a464ce193 --- /dev/null +++ b/src/http/gateway/resources/index.js @@ -0,0 +1,5 @@ +'use strict' + +module.exports = { + gateway: require('./gateway') +} diff --git a/src/http/gateway/routes/gateway.js b/src/http/gateway/routes/gateway.js new file mode 100644 index 0000000000..e1c0f3222f --- /dev/null +++ b/src/http/gateway/routes/gateway.js @@ -0,0 +1,18 @@ +'use strict' + +const resources = require('../resources') + +module.exports = (server) => { + const gateway = server.select('Gateway') + + gateway.route({ + method: '*', + path: '/ipfs/{hash*}', + config: { + pre: [ + { method: resources.gateway.checkHash, assign: 'args' } + ], + handler: resources.gateway.handler + } + }) +} diff --git a/src/http/gateway/routes/index.js b/src/http/gateway/routes/index.js new file mode 100644 index 0000000000..0e0656c258 --- /dev/null +++ b/src/http/gateway/routes/index.js @@ -0,0 +1,5 @@ +'use strict' + +module.exports = (server) => { + require('./gateway')(server) +} diff --git a/src/http/gateway/utils/html.js b/src/http/gateway/utils/html.js new file mode 100644 index 0000000000..dd5abe6b4f --- /dev/null +++ b/src/http/gateway/utils/html.js @@ -0,0 +1,83 @@ +'use strict' + +const filesize = require('filesize') + +const HTML_PAGE_STYLE = require('./style') +const PathUtil = require('./path') + +const getParentDirectoryURL = (originalParts) => { + const parts = originalParts.splice() + + if (parts.length > 1) { + parts.pop() + } + + return [ '', 'ipfs' ].concat(parts).join('/') +} + +const buildFilesList = (path, links) => { + const rows = links.map((link) => { + let row = [ + `
 
`, + `${link.name}`, + filesize(link.size) + ] + + row = row.map((cell) => `${cell}`).join('') + + return `${row}` + }) + + return rows.join('') +} + +const buildTable = (path, links) => { + const parts = PathUtil.splitPath(path) + let parentDirectoryURL = getParentDirectoryURL(parts) + + return ` + + + + + + + + ${buildFilesList(path, links)} + +
+
 
+
+ .. +
+ ` +} + +module.exports.build = (path, links) => { + return ` + + + + + ${path} + + + + +
+
+
+
+ Index of ${path} +
+ ${buildTable(path, links)} +
+
+ + + ` +} diff --git a/src/http/gateway/utils/path.js b/src/http/gateway/utils/path.js new file mode 100644 index 0000000000..a46ea8a96d --- /dev/null +++ b/src/http/gateway/utils/path.js @@ -0,0 +1,35 @@ +'use strict' + +const splitPath = (path) => { + if (path[path.length - 1] === '/') path = path.substring(0, path.length - 1) + return path.substring(6).split('/') +} + +const removeLeadingSlash = (url) => { + if (url[0] === '/') url = url.substring(1) + return url +} + +const removeTrailingSlash = (url) => { + if (url.endsWith('/')) url = url.substring(0, url.length - 1) + return url +} + +const removeSlashFromBothEnds = (url) => { + url = removeLeadingSlash(url) + url = removeTrailingSlash(url) + return url +} + +const joinURLParts = (...urls) => { + urls = urls.filter((url) => url.length > 0) + urls = [ '' ].concat(urls.map((url) => removeSlashFromBothEnds(url))) + + return urls.join('/') +} + +module.exports = { + splitPath, + removeTrailingSlash, + joinURLParts +} diff --git a/src/http/gateway/utils/style.js b/src/http/gateway/utils/style.js new file mode 100644 index 0000000000..9e1548741a --- /dev/null +++ b/src/http/gateway/utils/style.js @@ -0,0 +1,16 @@ +'use strict' + +module.exports = `html{font-family:sans-serif;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%}body{margin:0}a{background-color:transparent}a:active,a:hover{outline:0}strong{font-weight:700}button::-moz-focus-inner,input::-moz-focus-inner{padding:0;border:0}table{border-spacing:0;border-collapse:collapse}td{padding:0} @media print{*,:after,:before{color:#000!important;text-shadow:none!important;background:0 0!important;-webkit-box-shadow:none!important;box-shadow:none!important}a,a:visited{text-decoration:underline}a[href]:after{content:" (" attr(href) ")"}tr{page-break-inside:avoid}.table{border-collapse:collapse!important}.table td{background-color:#fff!important}}@font-face{font-family:'Glyphicons Halflings';src:url(../fonts/glyphicons-halflings-regular.eot);src:url(../fonts/glyphicons-halflings-regular.eot?#iefix) format('embedded-opentype'),url(../fonts/glyphicons-halflings-regular.woff2) format('woff2'),url(../fonts/glyphicons-halflings-regular.woff) format('woff'),url(../fonts/glyphicons-halflings-regular.ttf) format('truetype'),url(../fonts/glyphicons-halflings-regular.svg#glyphicons_halflingsregular) format('svg')}*{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}:after,:before{-webkit-box-sizing:border-box;-moz-box-sizing:border-box;box-sizing:border-box}html{font-size:10px;-webkit-tap-highlight-color:rgba(0,0,0,0)}body{font-family:"Helvetica Neue",Helvetica,Arial,sans-serif;font-size:14px;line-height:1.42857143;color:#333;background-color:#fff}a{color:#337ab7;text-decoration:none}a:focus,a:hover{color:#23527c;text-decoration:underline}a:focus{outline:thin dotted;outline:5px auto -webkit-focus-ring-color;outline-offset:-2px}.row{margin-right:-15px;margin-left:-15px}.col-xs-12,.col-xs-2{position:relative;min-height:1px;padding-right:15px;padding-left:15px}.col-xs-12,.col-xs-2{float:left}.col-xs-12{width:100%}.col-xs-2{width:16.66666667%}table{background-color:transparent}.table{width:100%;max-width:100%;margin-bottom:20px}.table>tbody>tr>td{padding:8px;line-height:1.42857143;vertical-align:top;border-top:1px solid #ddd}.table-striped>tbody>tr:nth-of-type(odd){background-color:#f9f9f9}.form-control::-moz-placeholder{color:#999;opacity:1}.form-control:-ms-input-placeholder{color:#999}.panel{margin-bottom:20px;background-color:#fff;border:1px solid transparent;border-radius:4px;-webkit-box-shadow:0 1px 1px rgba(0,0,0,.05);box-shadow:0 1px 1px rgba(0,0,0,.05)}.panel-heading{padding:10px 15px;border-bottom:1px solid transparent;border-top-left-radius:3px;border-top-right-radius:3px}.panel>.table{margin-bottom:0}.panel>.table:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child{border-bottom-right-radius:3px;border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child td:first-child{border-bottom-left-radius:3px}.panel>.table:last-child>tbody:last-child>tr:last-child td:last-child{border-bottom-right-radius:3px}.panel>.table>tbody:first-child>tr:first-child td{border-top:0}.panel-default{border-color:#ddd}.panel-default>.panel-heading{color:#333;background-color:#f5f5f5;border-color:#ddd}.row:after,.row:before{display:table;content:" "}.row:after{clear:both}@-ms-viewport{width:device-width}.ipfs-_blank{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-_page{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-aac{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ai{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-aiff{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-avi{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-bmp{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-c{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-cpp{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-css{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dat{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dmg{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-doc{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dotx{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dwg{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-dxf{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-eps{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-exe{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-flv{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-gif{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-h{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-hpp{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-html{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ics{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-iso{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-java{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-jpg{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-js{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-key{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-less{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-logo{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-mid{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-mp3{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-mp4{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-mpg{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-odf{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ods{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-odt{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-otp{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ots{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ott{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-pdf{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-php{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-png{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-ppt{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-psd{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-py{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-qt{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-rar{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-rb{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-rtf{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-sass{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-scss{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-sql{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-tga{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-tgz{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-tiff{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-txt{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-wav{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-xls{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-xlsx{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-xml{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-yml{background-image:url('');background-repeat:no-repeat;background-size:contain}.ipfs-zip{background-image:url('');background-repeat:no-repeat;background-size:contain} +.narrow {width: 0px;} +.padding { margin: 100px;} +#header { + background: #000; +} +#logo { + height: 25px; + margin: 10px; +} +.ipfs-icon { + width:16px; +} +` diff --git a/src/http-api/index.js b/src/http/index.js similarity index 97% rename from src/http-api/index.js rename to src/http/index.js index 834ee49844..72de653f0f 100644 --- a/src/http-api/index.js +++ b/src/http/index.js @@ -117,7 +117,9 @@ function HttpApi (repo, config, cliArgs) { errorHandler(this, this.server) // load routes - require('./routes')(this.server) + require('./api/routes')(this.server) + // load gateway routes + require('./gateway/routes')(this.server) // Set default headers setHeader(this.server, diff --git a/test/cli/pubsub.js b/test/cli/pubsub.js index 951c6a85ba..703b83a5b0 100644 --- a/test/cli/pubsub.js +++ b/test/cli/pubsub.js @@ -8,7 +8,7 @@ const expect = chai.expect chai.use(dirtyChai) const delay = require('delay') const waterfall = require('async/waterfall') -const HttpAPI = require('../../src/http-api') +const HttpAPI = require('../../src/http') // TODO needs to use ipfs-factory-daemon const createTempNode = '' const repoPath = require('./index').repoPath diff --git a/test/gateway/index.js b/test/gateway/index.js new file mode 100644 index 0000000000..afa9cb70d9 --- /dev/null +++ b/test/gateway/index.js @@ -0,0 +1,76 @@ +/* eslint-env mocha */ +'use strict' + +const chai = require('chai') +const dirtyChai = require('dirty-chai') +const expect = chai.expect +chai.use(dirtyChai) +const API = require('../../src/http') +const ncp = require('ncp').ncp +const path = require('path') +const clean = require('../utils/clean') + +describe('HTTP GATEWAY', () => { + const repoExample = path.join(__dirname, '../go-ipfs-repo') + const repoTests = path.join(__dirname, '../repo-tests-run') + + let http = {} + let gateway + + before((done) => { + http.api = new API(repoTests) + + ncp(repoExample, repoTests, (err) => { + expect(err).to.not.exist() + + http.api.start(false, () => { + gateway = http.api.server.select('Gateway') + done() + }) + }) + }) + + after((done) => { + http.api.stop((err) => { + expect(err).to.not.exist() + clean(repoTests) + done() + }) + }) + + describe('/ipfs/* route', () => { + it('returns 400 for request without argument', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs' + }, (res) => { + expect(res.statusCode).to.equal(400) + expect(res.result.Message).to.be.a('string') + done() + }) + }) + + it('400 for request with invalid argument', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs/invalid' + }, (res) => { + expect(res.statusCode).to.equal(400) + expect(res.result.Message).to.be.a('string') + done() + }) + }) + + it('valid hash', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs/QmT78zSuBmuS4z925WZfrqQ1qHaJ56DQaTfyMUF7F8ff5o' + }, (res) => { + expect(res.statusCode).to.equal(200) + expect(res.rawPayload).to.deep.equal(new Buffer('hello world' + '\n')) + expect(res.payload).to.equal('hello world' + '\n') + done() + }) + }) + }) +}) diff --git a/test/http-api/index.js b/test/http-api/index.js index ad647e92df..16e535322e 100644 --- a/test/http-api/index.js +++ b/test/http-api/index.js @@ -6,7 +6,7 @@ const chai = require('chai') const dirtyChai = require('dirty-chai') const expect = chai.expect chai.use(dirtyChai) -const API = require('../../src/http-api') +const API = require('../../src/http') const APIctl = require('ipfs-api') const ncp = require('ncp').ncp const path = require('path') diff --git a/test/interop/daemons/js.js b/test/interop/daemons/js.js index 66b81975ff..4b4089d494 100644 --- a/test/interop/daemons/js.js +++ b/test/interop/daemons/js.js @@ -6,7 +6,7 @@ const series = require('async/series') const rimraf = require('rimraf') const tmpDir = require('../util').tmpDir -const HttpApi = require('../../../src/http-api') +const HttpApi = require('../../../src/http') function portConfig (port) { port = port + 5 diff --git a/test/node.js b/test/node.js index 22872e6d02..fd97536cf0 100644 --- a/test/node.js +++ b/test/node.js @@ -3,6 +3,7 @@ let testCore = true let testHTTP = true let testCLI = true +let testGatway = true if (process.env.TEST) { switch (process.env.TEST) { @@ -14,6 +15,11 @@ if (process.env.TEST) { testCore = false testCLI = false break + case 'gateway': + testCore = false + testCLI = false + testHTTP = false + break case 'cli': testCore = false testHTTP = false @@ -34,3 +40,7 @@ if (testHTTP) { if (testCLI) { require('./cli') } + +if (testGatway) { + require('./gateway') +} diff --git a/test/utils/ipfs-factory-daemon/index.js b/test/utils/ipfs-factory-daemon/index.js index 04262fbbec..9c7a61ce73 100644 --- a/test/utils/ipfs-factory-daemon/index.js +++ b/test/utils/ipfs-factory-daemon/index.js @@ -3,7 +3,7 @@ const PeerId = require('peer-id') const IPFSAPI = require('ipfs-api') const clean = require('../clean') -const HttpApi = require('../../../src/http-api') +const HttpApi = require('../../../src/http') const series = require('async/series') const eachSeries = require('async/eachSeries') const defaultConfig = require('./default-config.json') From d05591a486bc356a1bc99be989e918cb8e53aeb0 Mon Sep 17 00:00:00 2001 From: David Dias Date: Sun, 3 Sep 2017 08:45:19 +0100 Subject: [PATCH 06/14] fix: bring back metrics file that got deleted during rebase --- src/http/api/routes/debug.js | 31 +++++++++++++++++++++++++++++++ 1 file changed, 31 insertions(+) create mode 100644 src/http/api/routes/debug.js diff --git a/src/http/api/routes/debug.js b/src/http/api/routes/debug.js new file mode 100644 index 0000000000..d5ae1e434d --- /dev/null +++ b/src/http/api/routes/debug.js @@ -0,0 +1,31 @@ +'use strict' + +const register = require('prom-client').register +const client = require('prom-client') + +// Endpoint for handling debug metrics +module.exports = (server) => { + const api = server.select('API') + // Clear the register to make sure we're not registering multiple ones + register.clear() + const gauge = new client.Gauge({ name: 'number_of_peers', help: 'the_number_of_currently_connected_peers' }) + + api.route({ + method: 'GET', + path: '/debug/metrics/prometheus', + handler: (request, reply) => { + if (!process.env.IPFS_MONITORING) { + return reply('Monitoring is disabled. Enable it by setting environment variable IPFS_MONITORING') + .code(501) // 501 = Not Implemented + } + server.app.ipfs.swarm.peers((err, res) => { + if (err) { + return reply(err).code(500) + } + const count = res.length + gauge.set(count) + reply(register.metrics()).header('Content-Type', register.contentType) + }) + } + }) +} From c31e013b3c9013cebd4cba85f199ffa73f4bac07 Mon Sep 17 00:00:00 2001 From: David Dias Date: Mon, 4 Sep 2017 11:08:11 +0100 Subject: [PATCH 07/14] refactor: gateway resolver --- src/http/gateway/resolver.js | 136 ++++++++++++++++------------------- 1 file changed, 63 insertions(+), 73 deletions(-) diff --git a/src/http/gateway/resolver.js b/src/http/gateway/resolver.js index ae89956f06..bc9ec1d5a0 100644 --- a/src/http/gateway/resolver.js +++ b/src/http/gateway/resolver.js @@ -12,35 +12,30 @@ const PathUtil = require('./utils/path') const INDEX_HTML_FILES = [ 'index.html', 'index.htm', 'index.shtml' ] +function noop () {} + const resolveDirectory = promisify((ipfs, path, multihash, callback) => { - if (!callback) { - callback = noop - } + callback = callback || noop mh.validate(mh.fromB58String(multihash)) - ipfs - .object - .get(multihash, { enc: 'base58' }) - .then((DAGNode) => { - const links = DAGNode.links - const indexFiles = links.filter((link) => INDEX_HTML_FILES.indexOf(link.name) !== -1) + ipfs.object.get(multihash, { enc: 'base58' }, (err, dagNode) => { + if (err) { return callback(err) } - // found index file in links - if (indexFiles.length > 0) { - return callback(null, indexFiles) - } + const links = dagNode.links + const indexFiles = links.filter((link) => INDEX_HTML_FILES.indexOf(link.name) !== -1) - return callback(null, html.build(path, links)) - }) -}) + // found index file in links + if (indexFiles.length > 0) { + return callback(null, indexFiles) + } -const noop = function () {} + return callback(null, html.build(path, links)) + }) +}) const resolveMultihash = promisify((ipfs, path, callback) => { - if (!callback) { - callback = noop - } + callback = callback || noop const parts = PathUtil.splitPath(path) const partsLength = parts.length @@ -53,62 +48,57 @@ const resolveMultihash = promisify((ipfs, path, callback) => { log('currentMultihash: ', currentMultihash) log('currentIndex: ', currentIndex, '/', partsLength) - ipfs - .object - .get(currentMultihash, { enc: 'base58' }) - .then((DAGNode) => { - // log('DAGNode: ', DAGNode) - if (currentIndex === partsLength - 1) { - // leaf node - log('leaf node: ', currentMultihash) - // log('DAGNode: ', DAGNode.links) - - if (DAGNode.links && - DAGNode.links.length > 0 && - DAGNode.links[0].name.length > 0) { - // this is a directory. - let isDirErr = new Error('This dag node is a directory') - // add currentMultihash as a fileName so it can be used by resolveDirectory - isDirErr.fileName = currentMultihash - return next(isDirErr) - } - - next() - } else { - // find multihash of requested named-file - // in current DAGNode's links - let multihashOfNextFile - const nextFileName = parts[currentIndex + 1] - const links = DAGNode.links - - for (let link of links) { - if (link.name === nextFileName) { - // found multihash of requested named-file - multihashOfNextFile = mh.toB58String(link.multihash) - log('found multihash: ', multihashOfNextFile) - break - } - } - - if (!multihashOfNextFile) { - log.error(`no link named "${nextFileName}" under ${currentMultihash}`) - throw new Error(`no link named "${nextFileName}" under ${currentMultihash}`) - } - - currentMultihash = multihashOfNextFile - next() - } - }) + ipfs.object.get(currentMultihash, { enc: 'base58' }, (err, dagNode) => { + if (err) { return next(err) } + + if (currentIndex === partsLength - 1) { + // leaf node + log('leaf node: ', currentMultihash) + + // TODO: Check if it is a directory by using Unixfs Type, right now + // it won't detect empty dirs + if (dagNode.links && + dagNode.links.length > 0 && + dagNode.links[0].name.length > 0) { + // this is a directory. + + let isDirErr = new Error('This dag node is a directory') + // add currentMultihash as a fileName so it can be used by resolveDirectory + isDirErr.fileName = currentMultihash + return next(isDirErr) + } + return next() + } + + // find multihash of requested named-file in current dagNode's links + let multihashOfNextFile + const nextFileName = parts[currentIndex + 1] + const links = dagNode.links + + for (let link of links) { + if (link.name === nextFileName) { + // found multihash of requested named-file + multihashOfNextFile = mh.toB58String(link.multihash) + log('found multihash: ', multihashOfNextFile) + break + } + } + + if (!multihashOfNextFile) { + log.error(`no link named "${nextFileName}" under ${currentMultihash}`) + return next(new Error(`no link named "${nextFileName}" under ${currentMultihash}`)) + } + + currentMultihash = multihashOfNextFile + next() + }) }, (err) => { - if (err) { - log.error(err) - return callback(err) - } - callback(null, {multihash: currentMultihash}) + if (err) { return callback(err) } + callback(null, { multihash: currentMultihash }) }) }) module.exports = { - resolveDirectory, - resolveMultihash + resolveDirectory: resolveDirectory, + resolveMultihash: resolveMultihash } From a624515451a65a5c0cd29768840606475fd92c2f Mon Sep 17 00:00:00 2001 From: Yahya Date: Mon, 4 Sep 2017 14:55:32 +0200 Subject: [PATCH 08/14] fix using js-ipfs-repo in gateway tests. License: MIT Signed-off-by: Yahya --- test/gateway/index.js | 6 +- ...HSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data | 4 + ...DDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data | 23 ++++ ...NLTXZM4PW4VEFWQBJ4ACZQAS37BLGL4HUO5XY.data | Bin 0 -> 309 bytes ...B5FJNHZPTSVA7IB6OHXSQ2XSVEEKMKK6RT75I.data | Bin 0 -> 101 bytes ...7E32LQAL6236OUKZTMHPQSFIXPWXNZHQOV7JQ.data | 55 +++++++++ ...SIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data | 3 + ...YVY4BA22FPHUIODJEXS4LCTQDWA275XAJDAPI.data | Bin 0 -> 309 bytes ...LZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data | 4 + ...774EZOYCYNHPRVFD53ZSAU7237F67XDSQGCYQ.data | Bin 0 -> 10807 bytes ...RTG5JS3JCUJDQM2DRB5RVF33DCUUOFJNGVDUI.data | Bin 0 -> 10849 bytes ...YNRNMP2VDKWBWGAEDDEJDACM3SGG3VIANDDXI.data | Bin 0 -> 10765 bytes ...LJNXLLHZOPGSL2PBC65D4UIVWM6TI5F5TAFNI.data | 24 ++++ ...VXLWEU4FWPVGT24VJT7DUZPTNLF25N25IGGQA.data | 4 + ...WJNESD7XHQSXA5EGJVNTPVHD7444C2KLKXHDI.data | Bin 0 -> 10765 bytes ...AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data | 8 ++ ...2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data | 3 + ...FZU7XAJL2BMIHGVB5ZR2IOKOSTRMLIKPB6K5I.data | Bin 0 -> 402 bytes ...TJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data | 9 ++ ...RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data | 115 ++++++++++++++++++ ...33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data | 3 + ...OUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data | 27 ++++ ...WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data | 3 + ...ML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data | 36 ++++++ ...HTQ7R247ZI7KJWP3QWPQYS43LFULQC5ANLQFI.data | Bin 0 -> 10765 bytes ...GPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data | 4 + ...57JSEZN64SIJ5OIHSGJG4TJSSJLGI3PBJLQVI.data | 0 ...JOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data | 28 +++++ ...OV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data | 3 + ...LP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data | 3 + test/js-ipfs-repo/blocks/SHARDING | 1 + ...D6VA64TGWP67KYA3AIMBUMVWGZ5AQN2L2HSWQ.data | Bin 0 -> 10765 bytes ...ANT6IBNTFN7WR5RPD5F6GN6MBKUUO25DNOTWQ.data | Bin 0 -> 10891 bytes ...NAKDMN4RM26SQETENO5EXBAE5MNXM5DXOBUMI.data | 3 + ...NH7VEGIQJRPL6J7FT2XYVKAXT4MQPXXPUYUNY.data | Bin 0 -> 10765 bytes ...XCX5XMZLMGTYVODWPEDIW7JYEG7YXBIA7IUWY.data | 3 + ...LOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data | 114 +++++++++++++++++ ...2BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data | 2 + ...2FSMG76VV256I4PXBULZZ5ASNLK4FL4EG7XOI.data | Bin 0 -> 10849 bytes ...5KVN6ALXC6QRHK2X4R6EUFRMBB5OSFO2FUYDQ.data | 3 + ...EBXFD5QN6MTI5YBYMCVQJDXPKCOVR6RMLHZFQ.data | Bin 0 -> 10807 bytes test/js-ipfs-repo/blocks/_README | 22 ++++ test/js-ipfs-repo/config | 34 ++++++ test/js-ipfs-repo/datastore/000002.ldb | Bin 0 -> 211 bytes test/js-ipfs-repo/datastore/000005.ldb | Bin 0 -> 1120 bytes test/js-ipfs-repo/datastore/000010.ldb | Bin 0 -> 1143 bytes test/js-ipfs-repo/datastore/CURRENT | 1 + test/js-ipfs-repo/datastore/LOCK | 0 test/js-ipfs-repo/datastore/LOG | 29 +++++ test/js-ipfs-repo/datastore/LOG.old | 10 ++ test/js-ipfs-repo/datastore/MANIFEST-000002 | Bin 0 -> 50 bytes test/js-ipfs-repo/datastore/MANIFEST-000014 | Bin 0 -> 314 bytes test/js-ipfs-repo/version | 1 + 53 files changed, 585 insertions(+), 3 deletions(-) create mode 100644 test/js-ipfs-repo/blocks/2F/CIQEUWUVLBXVFYSYCHHSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data create mode 100644 test/js-ipfs-repo/blocks/5V/CIQFFRR4O52TS2Z7QLDDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data create mode 100644 test/js-ipfs-repo/blocks/5X/CIQJ23BL4UHXA2KTI6NLTXZM4PW4VEFWQBJ4ACZQAS37BLGL4HUO5XY.data create mode 100644 test/js-ipfs-repo/blocks/75/CIQMB7DLJFKD267QJ2B5FJNHZPTSVA7IB6OHXSQ2XSVEEKMKK6RT75I.data create mode 100644 test/js-ipfs-repo/blocks/7J/CIQKKLBWAIBQZOIS5X7E32LQAL6236OUKZTMHPQSFIXPWXNZHQOV7JQ.data create mode 100644 test/js-ipfs-repo/blocks/AE/CIQONICFQZH7QVU6IPSIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data create mode 100644 test/js-ipfs-repo/blocks/AP/CIQHAKDLTL5GMIFGN5YVY4BA22FPHUIODJEXS4LCTQDWA275XAJDAPI.data create mode 100644 test/js-ipfs-repo/blocks/C4/CIQDDZ5EDQK5AP7LRTLZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data create mode 100644 test/js-ipfs-repo/blocks/CY/CIQDMKFEUGKSLXMEXO774EZOYCYNHPRVFD53ZSAU7237F67XDSQGCYQ.data create mode 100644 test/js-ipfs-repo/blocks/DU/CIQLBK52T5EHVHZY5URTG5JS3JCUJDQM2DRB5RVF33DCUUOFJNGVDUI.data create mode 100644 test/js-ipfs-repo/blocks/DX/CIQPDQJBGYDZNMOAGGYNRNMP2VDKWBWGAEDDEJDACM3SGG3VIANDDXI.data create mode 100644 test/js-ipfs-repo/blocks/FN/CIQIXBZMUTXFC5QIGMLJNXLLHZOPGSL2PBC65D4UIVWM6TI5F5TAFNI.data create mode 100644 test/js-ipfs-repo/blocks/GQ/CIQH7OEYWXL34RWYL7VXLWEU4FWPVGT24VJT7DUZPTNLF25N25IGGQA.data create mode 100644 test/js-ipfs-repo/blocks/HD/CIQDDVW2EZIJF4NQH7WJNESD7XHQSXA5EGJVNTPVHD7444C2KLKXHDI.data create mode 100644 test/js-ipfs-repo/blocks/IL/CIQJFGRQHQ45VCQLM7AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data create mode 100644 test/js-ipfs-repo/blocks/IR/CIQERMRAAFXUAUOX3V2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data create mode 100644 test/js-ipfs-repo/blocks/K5/CIQPW4MAGTUNEBGZCEFZU7XAJL2BMIHGVB5ZR2IOKOSTRMLIKPB6K5I.data create mode 100644 test/js-ipfs-repo/blocks/LG/CIQJBQD2O6K4CGJVCCTJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data create mode 100644 test/js-ipfs-repo/blocks/M4/CIQOLBQZSZAODJGGH6RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data create mode 100644 test/js-ipfs-repo/blocks/O6/CIQOYW2THIZBRGI7IN33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data create mode 100644 test/js-ipfs-repo/blocks/OO/CIQBT4N7PS5IZ5IG2ZOUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data create mode 100644 test/js-ipfs-repo/blocks/PJ/CIQB4F7VKKQDXHMXX6WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data create mode 100644 test/js-ipfs-repo/blocks/PU/CIQJF2E6OPOEYYEVHYML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data create mode 100644 test/js-ipfs-repo/blocks/QF/CIQGPALRQ24P6NS4OWHTQ7R247ZI7KJWP3QWPQYS43LFULQC5ANLQFI.data create mode 100644 test/js-ipfs-repo/blocks/QP/CIQNLGENZXNRUMUHZYGPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data create mode 100644 test/js-ipfs-repo/blocks/QV/CIQOHMGEIKMPYHAUTL57JSEZN64SIJ5OIHSGJG4TJSSJLGI3PBJLQVI.data create mode 100644 test/js-ipfs-repo/blocks/R3/CIQBED3K6YA5I3QQWLJOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data create mode 100644 test/js-ipfs-repo/blocks/S2/CIQPF3CHDB5GQ5ZBISOV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data create mode 100644 test/js-ipfs-repo/blocks/S5/CIQHBGZNZRPWVEFNMTLP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data create mode 100644 test/js-ipfs-repo/blocks/SHARDING create mode 100644 test/js-ipfs-repo/blocks/SW/CIQHPUVCWD6JA6AFUVD6VA64TGWP67KYA3AIMBUMVWGZ5AQN2L2HSWQ.data create mode 100644 test/js-ipfs-repo/blocks/TW/CIQFEAGMNNXXTYKYQSANT6IBNTFN7WR5RPD5F6GN6MBKUUO25DNOTWQ.data create mode 100644 test/js-ipfs-repo/blocks/UM/CIQOGPM2T3ZJMYJKYNNAKDMN4RM26SQETENO5EXBAE5MNXM5DXOBUMI.data create mode 100644 test/js-ipfs-repo/blocks/UN/CIQOMBKARLB7PAITVSNH7VEGIQJRPL6J7FT2XYVKAXT4MQPXXPUYUNY.data create mode 100644 test/js-ipfs-repo/blocks/UW/CIQDVKITASFS55MC2TXCX5XMZLMGTYVODWPEDIW7JYEG7YXBIA7IUWY.data create mode 100644 test/js-ipfs-repo/blocks/VO/CIQGFTQ7FSI2COUXWWLOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data create mode 100644 test/js-ipfs-repo/blocks/X3/CIQFTFEEHEDF6KLBT32BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data create mode 100644 test/js-ipfs-repo/blocks/XO/CIQJGO2B2N75IUEM372FSMG76VV256I4PXBULZZ5ASNLK4FL4EG7XOI.data create mode 100644 test/js-ipfs-repo/blocks/YD/CIQENVCICS44LLYUDQ5KVN6ALXC6QRHK2X4R6EUFRMBB5OSFO2FUYDQ.data create mode 100644 test/js-ipfs-repo/blocks/ZF/CIQLBS5HG4PRCRQ7O4EBXFD5QN6MTI5YBYMCVQJDXPKCOVR6RMLHZFQ.data create mode 100644 test/js-ipfs-repo/blocks/_README create mode 100644 test/js-ipfs-repo/config create mode 100644 test/js-ipfs-repo/datastore/000002.ldb create mode 100644 test/js-ipfs-repo/datastore/000005.ldb create mode 100644 test/js-ipfs-repo/datastore/000010.ldb create mode 100644 test/js-ipfs-repo/datastore/CURRENT create mode 100644 test/js-ipfs-repo/datastore/LOCK create mode 100644 test/js-ipfs-repo/datastore/LOG create mode 100644 test/js-ipfs-repo/datastore/LOG.old create mode 100644 test/js-ipfs-repo/datastore/MANIFEST-000002 create mode 100644 test/js-ipfs-repo/datastore/MANIFEST-000014 create mode 100644 test/js-ipfs-repo/version diff --git a/test/gateway/index.js b/test/gateway/index.js index afa9cb70d9..65d258bee9 100644 --- a/test/gateway/index.js +++ b/test/gateway/index.js @@ -10,8 +10,8 @@ const ncp = require('ncp').ncp const path = require('path') const clean = require('../utils/clean') -describe('HTTP GATEWAY', () => { - const repoExample = path.join(__dirname, '../go-ipfs-repo') +describe('HTTP Gateway', () => { + const repoExample = path.join(__dirname, '../js-ipfs-repo') const repoTests = path.join(__dirname, '../repo-tests-run') let http = {} @@ -67,7 +67,7 @@ describe('HTTP GATEWAY', () => { url: '/ipfs/QmT78zSuBmuS4z925WZfrqQ1qHaJ56DQaTfyMUF7F8ff5o' }, (res) => { expect(res.statusCode).to.equal(200) - expect(res.rawPayload).to.deep.equal(new Buffer('hello world' + '\n')) + expect(res.rawPayload).to.deep.equal(Buffer.from('hello world' + '\n')) expect(res.payload).to.equal('hello world' + '\n') done() }) diff --git a/test/js-ipfs-repo/blocks/2F/CIQEUWUVLBXVFYSYCHHSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data b/test/js-ipfs-repo/blocks/2F/CIQEUWUVLBXVFYSYCHHSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data new file mode 100644 index 0000000000..4145619655 --- /dev/null +++ b/test/js-ipfs-repo/blocks/2F/CIQEUWUVLBXVFYSYCHHSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data @@ -0,0 +1,4 @@ + +ys# js-ipfs-repo +Implementation of the IPFS repo spec (https://github.com/ipfs/specs/tree/master/repo) in JavaScript +s \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/5V/CIQFFRR4O52TS2Z7QLDDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data b/test/js-ipfs-repo/blocks/5V/CIQFFRR4O52TS2Z7QLDDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data new file mode 100644 index 0000000000..951bfe0400 --- /dev/null +++ b/test/js-ipfs-repo/blocks/5V/CIQFFRR4O52TS2Z7QLDDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data @@ -0,0 +1,23 @@ + + IPFS Alpha Security Notes + +We try hard to ensure our system is safe and robust, but all software +has bugs, especially new software. This distribution is meant to be an +alpha preview, don't use it for anything mission critical. + +Please note the following: + +- This is alpha software and has not been audited. It is our goal + to conduct a proper security audit once we close in on a 1.0 release. + +- ipfs is a networked program, and may have serious undiscovered + vulnerabilities. It is written in Go, and we do not execute any + user provided data. But please point any problems out to us in a + github issue, or email security@ipfs.io privately. + +- ipfs uses encryption for all communication, but it's NOT PROVEN SECURE + YET! It may be totally broken. For now, the code is included to make + sure we benchmark our operations with encryption in mind. In the future, + there will be an "unsafe" mode for high performance intranet apps. + If this is a blocking feature for you, please contact us. + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/5X/CIQJ23BL4UHXA2KTI6NLTXZM4PW4VEFWQBJ4ACZQAS37BLGL4HUO5XY.data b/test/js-ipfs-repo/blocks/5X/CIQJ23BL4UHXA2KTI6NLTXZM4PW4VEFWQBJ4ACZQAS37BLGL4HUO5XY.data new file mode 100644 index 0000000000000000000000000000000000000000..b799cf6b2236c91f8224b9fefe43aca16c773490 GIT binary patch literal 309 zcmWgA<5Ch*SgK>j#LTl(=mjz6J*Z#Z`mUeibke1%>*qt`A@ymo*6O-~wOC)CS z3K@XZPnu<5V|lBKJN>}4>2CTNRSd^0_H1qVw~ozKk4Ii%i@p$ha(-S(VseSZ2}U7Z zkd6rls>-Jxlr$AsHtpiyKm2l@?&p6zQA*^T_KfrP>FKF6gjh0Ca|$F5F$tM~ls|2g zoaXRw$uaxIJs~DLpX;8StGsyP<=b&`P8V3!Rm2K~xC=`&le2Y;OA?DpBo6Qj>4S6$ z@n?Nwypkud>C!{F>oTF2;tiY*sWLv16>@vN@a5$jZ-m&2QWH~hQzg!E2$_R41Rb*} zFSX3JZ#rf+Kwo-;;PadOz%uNME$6+{JHE$$U!OUW!sO)in> z4G}T|X$WFClbv7rFrua5=1<0)Q}=J#b|1g=` +- browsers or extensions can learn to use `ipfs://` directly +- hash-addressed content guarantees authenticity + +IPFS is modular: +- connection layer over any network protocol +- routing layer +- uses a routing layer DHT (kademlia/coral) +- uses a path-based naming service +- uses bittorrent-inspired block exchange + +IPFS uses crypto: +- cryptographic-hash content addressing +- block-level deduplication +- file integrity + versioning +- filesystem-level encryption + signing support + +IPFS is p2p: +- worldwide peer-to-peer file transfers +- completely decentralized architecture +- **no** central point of failure + +IPFS is a cdn: +- add a file to the filesystem locally, and it's now available to the world +- caching-friendly (content-hash naming) +- bittorrent-based bandwidth distribution + +IPFS has a name service: +- IPNS, an SFS inspired name system +- global namespace based on PKI +- serves to build trust chains +- compatible with other NSes +- can map DNS, .onion, .bit, etc to IPNS + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/AE/CIQONICFQZH7QVU6IPSIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data b/test/js-ipfs-repo/blocks/AE/CIQONICFQZH7QVU6IPSIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data new file mode 100644 index 0000000000..6860441aa1 --- /dev/null +++ b/test/js-ipfs-repo/blocks/AE/CIQONICFQZH7QVU6IPSIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data @@ -0,0 +1,3 @@ +/ +" gq6\u8~:6~gZ.directT2 +" 6(%݄.Ӿ5(ab recursiveT \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/AP/CIQHAKDLTL5GMIFGN5YVY4BA22FPHUIODJEXS4LCTQDWA275XAJDAPI.data b/test/js-ipfs-repo/blocks/AP/CIQHAKDLTL5GMIFGN5YVY4BA22FPHUIODJEXS4LCTQDWA275XAJDAPI.data new file mode 100644 index 0000000000000000000000000000000000000000..74de75af616ff437b5c7a3a5272281eac2146b6c GIT binary patch literal 309 zcmWgA<5Ch*SgK>j#LTl(=mjz6J*Z#Z`mUeibke1%>*qt`A@ymo*6O-~wOC)CS z3K@XZPnu<5V|lBKJN>}4>2CTNRSd^0_H1qVw~ozKk4Ii%i@p$ha(-S(VseSZ2}U7Z zkd6rls>-Jxlr$AsHtpiyKm2l@?&p6zQA*^T_KfrP>FKF6gjh0Ca|$F5F$tM~lqa2& z*O|D`YWmh`FN(J=HpxnHW4)f}dn)*>RpfVzsIJ{Y+=Zo?$=SNaC5c5P5({~S^g+6W z__Mw-Uda>Kbm^hob(zph@di$ZR2iSh3b{RB`110NH$rShsfj7MsS@Wngv>!2f{xjg zms)1qHyyK_UR9)g=w{aZf~|kMWdrSdqBHNTyCTF_oSIx(lvz@#o0ngbS}f7b!NtVE F2moEFc%uLS literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/blocks/C4/CIQDDZ5EDQK5AP7LRTLZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data b/test/js-ipfs-repo/blocks/C4/CIQDDZ5EDQK5AP7LRTLZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data new file mode 100644 index 0000000000..ecce1053f6 --- /dev/null +++ b/test/js-ipfs-repo/blocks/C4/CIQDDZ5EDQK5AP7LRTLZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data @@ -0,0 +1,4 @@ +5 +" ׾F_uؔlzS?|ڲPc@ js-ipfs-repo + + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/CY/CIQDMKFEUGKSLXMEXO774EZOYCYNHPRVFD53ZSAU7237F67XDSQGCYQ.data b/test/js-ipfs-repo/blocks/CY/CIQDMKFEUGKSLXMEXO774EZOYCYNHPRVFD53ZSAU7237F67XDSQGCYQ.data new file mode 100644 index 0000000000000000000000000000000000000000..bbe6bda78d0a52f6d87da578f452d43718b9aa9f GIT binary patch literal 10807 zcmeI&tqQ_m6vpvsQC>mhU=TmxWeCpZ4H&eDU=nNw5wR?6Yue0K5w>X*qZns5SS{Xw z-hn1RK)egXzn^qC&+{usEMpXsYYQnaL-~-$SEZaY\IzxEElM/fLICENSE1 +" JZXoRX!Fwd87U;SöWw README.md{ + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/HD/CIQDDVW2EZIJF4NQH7WJNESD7XHQSXA5EGJVNTPVHD7444C2KLKXHDI.data b/test/js-ipfs-repo/blocks/HD/CIQDDVW2EZIJF4NQH7WJNESD7XHQSXA5EGJVNTPVHD7444C2KLKXHDI.data new file mode 100644 index 0000000000000000000000000000000000000000..5ea0edda6f82982c276a4a3813a50c0e35121c68 GIT binary patch literal 10765 zcmeIzF%AJy6a~;Rm3)!Nk3xb(CK9$mt(wFR^jf72s6=9D)srdd6*gc6EI~I4+fbbT zRX6Y66Q?1IqgQpGAHRd{ss22Q^=&*UXYbW{zqlNV{wBXJW712aODXvf?XV11kq9)T zAq{CrLmJYMhBTxh4QWV28q$!4G^8O7X-GpF(vXHUq#+GyNJARZkcKp*;U8!iTA|tK KY}4Vho$h|}9P~&4 literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/blocks/IL/CIQJFGRQHQ45VCQLM7AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data b/test/js-ipfs-repo/blocks/IL/CIQJFGRQHQ45VCQLM7AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data new file mode 100644 index 0000000000..62d1c2979b --- /dev/null +++ b/test/js-ipfs-repo/blocks/IL/CIQJFGRQHQ45VCQLM7AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data @@ -0,0 +1,8 @@ + +Come hang out in our IRC chat room if you have any questions. + +Contact the ipfs dev team: +- Bugs: https://github.com/ipfs/go-ipfs/issues +- Help: irc.freenode.org/#ipfs +- Email: dev@ipfs.io + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/IR/CIQERMRAAFXUAUOX3V2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data b/test/js-ipfs-repo/blocks/IR/CIQERMRAAFXUAUOX3V2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data new file mode 100644 index 0000000000..4a26a4ac35 --- /dev/null +++ b/test/js-ipfs-repo/blocks/IR/CIQERMRAAFXUAUOX3V2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data @@ -0,0 +1,3 @@ +/ +" !61صF2$`7#u@1directT2 +" Hz8#3u2ED ƥ*QKMQ recursiveT \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/K5/CIQPW4MAGTUNEBGZCEFZU7XAJL2BMIHGVB5ZR2IOKOSTRMLIKPB6K5I.data b/test/js-ipfs-repo/blocks/K5/CIQPW4MAGTUNEBGZCEFZU7XAJL2BMIHGVB5ZR2IOKOSTRMLIKPB6K5I.data new file mode 100644 index 0000000000000000000000000000000000000000..e1cd3e3e21d1e86994739a1b26b18c77aa98f034 GIT binary patch literal 402 zcmWgA<5Ch*SgK>j#LTl(=mjz6J*Z#Z`mUeibke1%>*qt`A@ymo*6O-~wOC)CS z3K@XZPnu<5V|lBKJN>}4>2CTNRSd^0_H1qVw~ozKk4Ii%i@p$ha(-S(VseSZ2}U6u zkd9Ap+$E|q$`xJa#!6jHIxzjpl!P0h>MB2GPV&4rkBc)?h$SUIxmdzcNEf7f!hx#t zsRt!Z1(r>_`1cRLoTvNwA5WALIj242ynT9l>I@;4jMSV0i9<|6CLrZc+a#wsJX~_j zesNET$-x#mt z32eIbQ0}@+=%siAr$efYPh^GMo-cfP`NkU|wxZO;l-yK_a~wkEAPtfq_t)&|`O0=J z)>&iK^$Wi5P8mg6XI0#KB6hf@e{Q&y5MOa>a%oX!Nu_RHeo1Pv#8gg*cdpLpJ$qZq tsQnzz_w~A`XB0nMvHZ_Q{$={^NqgFq=2+MYv6SSO7D?P^<6`1q1OQn=pS%D7 literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/blocks/LG/CIQJBQD2O6K4CGJVCCTJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data b/test/js-ipfs-repo/blocks/LG/CIQJBQD2O6K4CGJVCCTJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data new file mode 100644 index 0000000000..71be805f1e --- /dev/null +++ b/test/js-ipfs-repo/blocks/LG/CIQJBQD2O6K4CGJVCCTJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data @@ -0,0 +1,9 @@ + +Some helpful resources for finding your way around ipfs: + +- quick-start: a quick show of various ipfs features. +- ipfs commands: a list of all commands +- ipfs --help: every command describes itself +- https://github.com/ipfs/go-ipfs -- the src repository +- #ipfs on irc.freenode.org -- the community irc channel + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/M4/CIQOLBQZSZAODJGGH6RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data b/test/js-ipfs-repo/blocks/M4/CIQOLBQZSZAODJGGH6RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data new file mode 100644 index 0000000000..f2bf4f8b8d --- /dev/null +++ b/test/js-ipfs-repo/blocks/M4/CIQOLBQZSZAODJGGH6RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data @@ -0,0 +1,115 @@ + +  # 0.1 - Quick Start + +This is a set of short examples with minimal explanation. It is meant as +a "quick start". Soon, we'll write a longer tour :-) + + +Add a file to ipfs: + + echo "hello world" >hello + ipfs add hello + + +View it: + + ipfs cat + + +Try a directory: + + mkdir foo + mkdir foo/bar + echo "baz" > foo/baz + echo "baz" > foo/bar/baz + ipfs add -r foo + + +View things: + + ipfs ls + ipfs ls /bar + ipfs cat /baz + ipfs cat /bar/baz + ipfs cat /bar + ipfs ls /baz + + +References: + + ipfs refs + ipfs refs -r + ipfs refs --help + + +Get: + + ipfs get -o foo2 + diff foo foo2 + + +Objects: + + ipfs object get + ipfs object get /foo2 + ipfs object --help + + +Pin + GC: + + ipfs pin add + ipfs repo gc + ipfs ls + ipfs pin rm + ipfs repo gc + + +Daemon: + + ipfs daemon (in another terminal) + ipfs id + + +Network: + + (must be online) + ipfs swarm peers + ipfs id + ipfs cat + + +Mount: + + (warning: fuse is finicky!) + ipfs mount + cd /ipfs/ + ls + + +Tool: + + ipfs version + ipfs update + ipfs commands + ipfs config --help + open http://localhost:5001/webui + + +Browse: + + webui: + + http://localhost:5001/webui + + video: + + http://localhost:8080/ipfs/QmVc6zuAneKJzicnJpfrqCH9gSy6bz54JhcypfJYhGUFQu/play#/ipfs/QmTKZgRNwDNZwHtJSjCp6r5FYefzpULfy37JvMt9DwvXse + + images: + + http://localhost:8080/ipfs/QmZpc3HvfjEXvLWGQPWbHk3AjD5j8NEN4gmFN8Jmrd5g83/cs + + markdown renderer app: + + http://localhost:8080/ipfs/QmX7M9CiYXjVeFnkfVGf3y5ixTZ2ACeSGyL1vBJY1HvQPp/mdown + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/O6/CIQOYW2THIZBRGI7IN33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data b/test/js-ipfs-repo/blocks/O6/CIQOYW2THIZBRGI7IN33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data new file mode 100644 index 0000000000..7b58d6c857 --- /dev/null +++ b/test/js-ipfs-repo/blocks/O6/CIQOYW2THIZBRGI7IN33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data @@ -0,0 +1,3 @@ +/ +" @ԆDgA7directT2 +" ;APY0k}E=p  recursiveT \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/OO/CIQBT4N7PS5IZ5IG2ZOUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data b/test/js-ipfs-repo/blocks/OO/CIQBT4N7PS5IZ5IG2ZOUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data new file mode 100644 index 0000000000..770348274e --- /dev/null +++ b/test/js-ipfs-repo/blocks/OO/CIQBT4N7PS5IZ5IG2ZOUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data @@ -0,0 +1,27 @@ + +  IPFS Alpha Security Notes + +We try hard to ensure our system is safe and robust, but all software +has bugs, especially new software. This distribution is meant to be an +alpha preview, don't use it for anything mission critical. + +Please note the following: + +- This is alpha software and has not been audited. It is our goal + to conduct a proper security audit once we close in on a 1.0 release. + +- ipfs is a networked program, and may have serious undiscovered + vulnerabilities. It is written in Go, and we do not execute any + user provided data. But please point any problems out to us in a + github issue, or email security@ipfs.io privately. + +- security@ipfs.io GPG key: + - 4B9665FB 92636D17 7C7A86D3 50AAE8A9 59B13AF3 + - https://pgp.mit.edu/pks/lookup?op=get&search=0x50AAE8A959B13AF3 + +- ipfs uses encryption for all communication, but it's NOT PROVEN SECURE + YET! It may be totally broken. For now, the code is included to make + sure we benchmark our operations with encryption in mind. In the future, + there will be an "unsafe" mode for high performance intranet apps. + If this is a blocking feature for you, please contact us. + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/PJ/CIQB4F7VKKQDXHMXX6WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data b/test/js-ipfs-repo/blocks/PJ/CIQB4F7VKKQDXHMXX6WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data new file mode 100644 index 0000000000..0335563629 --- /dev/null +++ b/test/js-ipfs-repo/blocks/PJ/CIQB4F7VKKQDXHMXX6WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data @@ -0,0 +1,3 @@ + + Index + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/PU/CIQJF2E6OPOEYYEVHYML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data b/test/js-ipfs-repo/blocks/PU/CIQJF2E6OPOEYYEVHYML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data new file mode 100644 index 0000000000..e6ef304bfd --- /dev/null +++ b/test/js-ipfs-repo/blocks/PU/CIQJF2E6OPOEYYEVHYML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data @@ -0,0 +1,36 @@ + +WIP + +# 0.0 - Introduction + +Welcome to IPFS! This tour will guide you through a few of the +features of this tool, and the most common commands. Then, it will +immerse you into the world of merkledags and the amazing things +you can do with them. + + +This tour has many parts, and can be taken in different sequences. +Different people learn different ways, so choose your own adventure: + + To start with the concepts, try: + - The Merkle DAG + - Data Structures on the Merkle DAG + - Representing Files with unixfs + - add, cat, ls, refs + ... + + + To start with the examples, try: + - add, cat, ls, refs + - Representing Files with unixfs + - Data Structures on the Merkle DAG + - The Merkle DAG + ... + + + To start with the network, try: + - IPFS Nodes + - Running the daemon + - The Swarm + - The Web + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/QF/CIQGPALRQ24P6NS4OWHTQ7R247ZI7KJWP3QWPQYS43LFULQC5ANLQFI.data b/test/js-ipfs-repo/blocks/QF/CIQGPALRQ24P6NS4OWHTQ7R247ZI7KJWP3QWPQYS43LFULQC5ANLQFI.data new file mode 100644 index 0000000000000000000000000000000000000000..a8f98693b7d4797d0a55704b16c15a6aa7d34bd5 GIT binary patch literal 10765 zcmeIzF%AJy6a~Szz&}ww{ J&EYejEr0X~^v?hQ literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/blocks/QP/CIQNLGENZXNRUMUHZYGPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data b/test/js-ipfs-repo/blocks/QP/CIQNLGENZXNRUMUHZYGPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data new file mode 100644 index 0000000000..6636930467 --- /dev/null +++ b/test/js-ipfs-repo/blocks/QP/CIQNLGENZXNRUMUHZYGPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data @@ -0,0 +1,4 @@ +2 +" sL`>P}D +>ڟo_="u' 0.0-intro + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/QV/CIQOHMGEIKMPYHAUTL57JSEZN64SIJ5OIHSGJG4TJSSJLGI3PBJLQVI.data b/test/js-ipfs-repo/blocks/QV/CIQOHMGEIKMPYHAUTL57JSEZN64SIJ5OIHSGJG4TJSSJLGI3PBJLQVI.data new file mode 100644 index 0000000000..e69de29bb2 diff --git a/test/js-ipfs-repo/blocks/R3/CIQBED3K6YA5I3QQWLJOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data b/test/js-ipfs-repo/blocks/R3/CIQBED3K6YA5I3QQWLJOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data new file mode 100644 index 0000000000..389e111776 --- /dev/null +++ b/test/js-ipfs-repo/blocks/R3/CIQBED3K6YA5I3QQWLJOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data @@ -0,0 +1,28 @@ + +Hello and Welcome to IPFS! + +██╗██████╗ ███████╗███████╗ +██║██╔══██╗██╔════╝██╔════╝ +██║██████╔╝█████╗ ███████╗ +██║██╔═══╝ ██╔══╝ ╚════██║ +██║██║ ██║ ███████║ +╚═╝╚═╝ ╚═╝ ╚══════╝ + +If you're seeing this, you have successfully installed +IPFS and are now interfacing with the ipfs merkledag! + + ------------------------------------------------------- +| Warning: | +| This is alpha software. Use at your own discretion! | +| Much is missing or lacking polish. There are bugs. | +| Not yet secure. Read the security notes for more. | + ------------------------------------------------------- + +Check out some of the other files in this directory: + + ./about + ./help + ./quick-start <-- usage examples + ./readme <-- this file + ./security-notes + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/S2/CIQPF3CHDB5GQ5ZBISOV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data b/test/js-ipfs-repo/blocks/S2/CIQPF3CHDB5GQ5ZBISOV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data new file mode 100644 index 0000000000..b137a86405 --- /dev/null +++ b/test/js-ipfs-repo/blocks/S2/CIQPF3CHDB5GQ5ZBISOV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data @@ -0,0 +1,3 @@ +- +" R;fqaU 0 [X@8ӷOindex + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/S5/CIQHBGZNZRPWVEFNMTLP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data b/test/js-ipfs-repo/blocks/S5/CIQHBGZNZRPWVEFNMTLP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data new file mode 100644 index 0000000000..3a99c365f0 --- /dev/null +++ b/test/js-ipfs-repo/blocks/S5/CIQHBGZNZRPWVEFNMTLP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data @@ -0,0 +1,3 @@ +4 +" Y9_)a˹2RmŖke9 js-ipfs-repo + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/SHARDING b/test/js-ipfs-repo/blocks/SHARDING new file mode 100644 index 0000000000..a153331dac --- /dev/null +++ b/test/js-ipfs-repo/blocks/SHARDING @@ -0,0 +1 @@ +/repo/flatfs/shard/v1/next-to-last/2 diff --git a/test/js-ipfs-repo/blocks/SW/CIQHPUVCWD6JA6AFUVD6VA64TGWP67KYA3AIMBUMVWGZ5AQN2L2HSWQ.data b/test/js-ipfs-repo/blocks/SW/CIQHPUVCWD6JA6AFUVD6VA64TGWP67KYA3AIMBUMVWGZ5AQN2L2HSWQ.data new file mode 100644 index 0000000000000000000000000000000000000000..78421c81c29ed8c18921f686ea3292e3e7300668 GIT binary patch literal 10765 zcmeIzAqoOf6a~;x48ym`#~`RM41#V$vuVH`*sVtQAcCXSECW-t8{CI3!R7?FVL1Jz z%e(i)S;%Adsve7DGwPqdpJ%zbPo~%Tdwt$7FNbotEpDrr43g+lN<931^R@s0 literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/blocks/TW/CIQFEAGMNNXXTYKYQSANT6IBNTFN7WR5RPD5F6GN6MBKUUO25DNOTWQ.data b/test/js-ipfs-repo/blocks/TW/CIQFEAGMNNXXTYKYQSANT6IBNTFN7WR5RPD5F6GN6MBKUUO25DNOTWQ.data new file mode 100644 index 0000000000000000000000000000000000000000..10aa2ae4f1291a62edd84980f1c05b09b95f78ff GIT binary patch literal 10891 zcmWgA;8GG&c)a0=(~LhdBC~#fIWaSTr;7SI$0sSXC;Kd!I#aqLXh*0Jg9L-nsLIh$ z8ciXixns0s7%de?YlP9-aI`iYtqn(O!_nGsv^E^A4M%Ik(b{mdHXN-DM{C2;+HkZs z9IXvUYs1mnaI`iYtqn(O!_nGsv^E^A4M%Ik(b{mdHXN-DM{C2;+HkZs9IXvUYs1mn zaI`iYG__&mlom_2c+JFlUxXS??=;#MbToJ7(P`PK4+MoQKg`(u9q~6w(px+ cb8FM| recursiveT \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/VO/CIQGFTQ7FSI2COUXWWLOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data b/test/js-ipfs-repo/blocks/VO/CIQGFTQ7FSI2COUXWWLOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data new file mode 100644 index 0000000000..2dd80560a2 --- /dev/null +++ b/test/js-ipfs-repo/blocks/VO/CIQGFTQ7FSI2COUXWWLOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data @@ -0,0 +1,114 @@ + +  # 0.1 - Quick Start + +This is a set of short examples with minimal explanation. It is meant as +a "quick start". Soon, we'll write a longer tour :-) + + +Add a file to ipfs: + + echo "hello world" >hello + ipfs add hello + + +View it: + + ipfs cat + + +Try a directory: + + mkdir foo + mkdir foo/bar + echo "baz" > foo/baz + echo "baz" > foo/bar/baz + ipfs add -r foo + + +View things: + + ipfs ls + ipfs ls /bar + ipfs cat /baz + ipfs cat /bar/baz + ipfs cat /bar + ipfs ls /baz + + +References: + + ipfs refs + ipfs refs -r + ipfs refs --help + + +Get: + + ipfs get foo2 + diff foo foo2 + + +Objects: + + ipfs object get + ipfs object get /foo2 + ipfs object --help + + +Pin + GC: + + ipfs pin -r + ipfs gc + ipfs ls + ipfs unpin -r + ipfs gc + + +Daemon: + + ipfs daemon (in another terminal) + ipfs id + + +Network: + + (must be online) + ipfs swarm peers + ipfs id + ipfs cat + + +Mount: + + (warning: fuse is finicky!) + ipfs mount + cd /ipfs/< + + +Tool: + + ipfs version + ipfs update + ipfs commands + ipfs config --help + open http://localhost:5001/webui + + +Browse: + + webui: + + http://localhost:5001/webui + + video: + + http://localhost:8080/ipfs/QmVc6zuAneKJzicnJpfrqCH9gSy6bz54JhcypfJYhGUFQu/play#/ipfs/QmTKZgRNwDNZwHtJSjCp6r5FYefzpULfy37JvMt9DwvXse + + images: + + http://localhost:8080/ipfs/QmZpc3HvfjEXvLWGQPWbHk3AjD5j8NEN4gmFN8Jmrd5g83/cs + + markdown renderer app: + + http://localhost:8080/ipfs/QmX7M9CiYXjVeFnkfVGf3y5ixTZ2ACeSGyL1vBJY1HvQPp/mdown + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/X3/CIQFTFEEHEDF6KLBT32BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data b/test/js-ipfs-repo/blocks/X3/CIQFTFEEHEDF6KLBT32BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data new file mode 100644 index 0000000000..9553a942db --- /dev/null +++ b/test/js-ipfs-repo/blocks/X3/CIQFTFEEHEDF6KLBT32BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data @@ -0,0 +1,2 @@ + + \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/XO/CIQJGO2B2N75IUEM372FSMG76VV256I4PXBULZZ5ASNLK4FL4EG7XOI.data b/test/js-ipfs-repo/blocks/XO/CIQJGO2B2N75IUEM372FSMG76VV256I4PXBULZZ5ASNLK4FL4EG7XOI.data new file mode 100644 index 0000000000000000000000000000000000000000..d899663bf7a13bb0563304b1d96985a784c2d515 GIT binary patch literal 10849 zcmWgA;8GG&c)a0=(~LhdBC~#fIWaSTr;7SI$0sSXC;Kd!I#aqLXh*0Jg9L-nsLIh$ z8ciXixns0s7%de?YlP9-aI`iYtqn(O!_nGsv^E^A4M%Ik(b{mdHXN-DM{C2;+HkZs z9IXvUYs1mnaI`iYtqn(O!_nGsv^E^A4M%Ik(b{mdHXN-DM{C2;+HkZs9IXvUYs1mn zaI`iYG__&mlom_2c+JFlUxXS??=;#MbToJ7(P`PK4+MoQKg`(u9-MiqUv4K}orj%_y00+YJV}#XGok z)Gc&_%?H(zbc;FK34aUrCGW}p=j0S(a*4bcz{(GU&M5Dn1~4bcz{(GU&M5Dn1~4bcz{(GU&M5Dn1~ z4bcz{(GU&M5Dn4rZ!}yDE04r@ptViQt$NSPbgyf+T8kZmjdSx~?(hFos3IOo>d{>N G+6cZA+z2}W literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/blocks/_README b/test/js-ipfs-repo/blocks/_README new file mode 100644 index 0000000000..ac3b6034c3 --- /dev/null +++ b/test/js-ipfs-repo/blocks/_README @@ -0,0 +1,22 @@ +This is a repository of IPLD objects. Each IPLD object is in a single file, +named .data. Where is the +"base32" encoding of the CID (as specified in +https://github.com/multiformats/multibase) without the 'B' prefix. +All the object files are placed in a tree of directories, based on a +function of the CID. This is a form of sharding similar to +the objects directory in git repositories. Previously, we used +prefixes, we now use the next-to-last two charters. + func NextToLast(base32cid string) { + nextToLastLen := 2 + offset := len(base32cid) - nextToLastLen - 1 + return str[offset : offset+nextToLastLen] + } +For example, an object with a base58 CIDv1 of + zb2rhYSxw4ZjuzgCnWSt19Q94ERaeFhu9uSqRgjSdx9bsgM6f +has a base32 CIDv1 of + BAFKREIA22FLID5AJ2KU7URG47MDLROZIH6YF2KALU2PWEFPVI37YLKRSCA +and will be placed at + SC/AFKREIA22FLID5AJ2KU7URG47MDLROZIH6YF2KALU2PWEFPVI37YLKRSCA.data +with 'SC' being the last-to-next two characters and the 'B' at the +beginning of the CIDv1 string is the multibase prefix that is not +stored in the filename. diff --git a/test/js-ipfs-repo/config b/test/js-ipfs-repo/config new file mode 100644 index 0000000000..5374e57718 --- /dev/null +++ b/test/js-ipfs-repo/config @@ -0,0 +1,34 @@ +{ + "Addresses": { + "Swarm": [ + "/ip4/0.0.0.0/tcp/4002", + "/ip4/127.0.0.1/tcp/4003/ws" + ], + "API": "/ip4/127.0.0.1/tcp/5002", + "Gateway": "/ip4/127.0.0.1/tcp/9090" + }, + "Discovery": { + "MDNS": { + "Enabled": true, + "Interval": 10 + }, + "webRTCStar": { + "Enabled": true + } + }, + "Bootstrap": [ + "/ip4/104.131.131.82/tcp/4001/ipfs/QmaCpDMGvV2BGHeYERUEnRQAwe3N8SzbUtfsmvsqQLuvuJ", + "/ip4/104.236.176.52/tcp/4001/ipfs/QmSoLnSGccFuZQJzRadHn95W2CrSFmZuTdDWP8HXaHca9z", + "/ip4/104.236.179.241/tcp/4001/ipfs/QmSoLPppuBtQSGwKDZT2M73ULpjvfd3aZ6ha4oFGL1KrGM", + "/ip4/162.243.248.213/tcp/4001/ipfs/QmSoLueR4xBeUbY9WZ9xGUUxunbKWcrNFTDAadQJmocnWm", + "/ip4/128.199.219.111/tcp/4001/ipfs/QmSoLSafTMBsPKadTEgaXctDQVcqN88CNLHXMkTNwMKPnu", + "/ip4/104.236.76.40/tcp/4001/ipfs/QmSoLV4Bbm51jM9C4gDYZQ9Cy3U6aXMJDAbzgu2fzaDs64", + "/ip4/178.62.158.247/tcp/4001/ipfs/QmSoLer265NRgSp2LA3dPaeykiS1J6DifTC88f5uVQKNAd", + "/ip4/178.62.61.185/tcp/4001/ipfs/QmSoLMeWqB7YGVLJN3pNLQpmmEk35v6wYtsMGLzSr5QBU3", + "/ip4/104.236.151.122/tcp/4001/ipfs/QmSoLju6m7xTh3DuokvT3886QRYqxAzb1kShaanJgW36yx" + ], + "Identity": { + "PeerID": "QmWR9foWsbncr89eH7sb6H5XtJSr2DUHyg3pcJWzwCVD5i", + "PrivKey": "CAASpwkwggSjAgEAAoIBAQCVa6fmcaxCpvGBg6RCQoHAJjN3/+ZFDjPt2/hRgNnAKCoA4OfLOKtkh0iVLoFErh/JeL2pd7xzFpzaAYqNkpi8TV2Gu3TOj3dY1jnJ7Kn5jp96rDyPpbWpKLrlGxrZ1lFRQJZnI/TwK9EclTS9wg9sHJMGFX0VrmpnY6XbpToo8N4WyE0hyQwoEOZrOcPTuSNQtXh9lQKzW5MgXh6zuyn4H2FyNWrJmn+0ICduZwVze/BWtbY3zB39WP7xMLg/CKJJSjPNErDpgfSzXloCkRH2eWudfkPiJZQCXQUBaf96DazGjT53WyTsiARuhvgybusm+v+oySitnMPem90tOWCDAgMBAAECggEAJs+64fm+0+VHL+wAu5FI4J9LmSzoJCHHTU5o1srQRMBqAKZqrve+VTNqoA1fWHaeHRf2s9E9KuvBvELQOWvNroOt+nQwqALN7k5e3ltR31ezhwq8BFf3ssO4oBAFTnYD00YiixHk6omHHJSJCY0hsmRNsDPetMYEL2zLxRzK2Z0pQ8cj9O0CuDI8Dp1BL+Qj36e7NN6J8ApResXMXhaxG9XJjH05s4Jmvh3dhe0uzeqtw82k7nUG3w3BrYeLQlCxCEKo2gdBiSiGLWpzbnBhHqrpBlUbFZXUFi7bDxPoEfp5O3J1rReaTO32Ks3gipVT2AZZR0/b9PwP+4b5/5s98QKBgQDSS1bSeR3GMMf+/5payOuI1JvgsdJxdEi7fGhlXAWEzB+DLFS5rnk3FGR5CtssP1SokG/2y4zYZwmfReJElcKi8/97+cz8qaOSVibNJQgo4YFoieL6yN2T+PTERrP1wKHluY/VDVEip0f+Dy27YWFzyfkRftJauH1EJhHB03prCwKBgQC15VVcdeonjhdg43HRD6fYapIwRu3SKKW6+cuZX1z9i4pd6leF4659v0J5+56qZwlQP7tprALMEzXP9Qlp0CnmeayxKv13P0gCa2dj07ev0I7tgfyu+VsanAxdNpVjITkzMZFvA+iaPcl9exBXQyqibZ9F+ouySf/6j7ISOn4LaQKBgQDNJBdXDDViT9XZSAiQBS3W9ef1giIS8Oe+lTbVobX0fJdFA6rG04+FjowQk+mIOWNhaD0AmEIWHDWDvM1kMF+FTNZuXxl7ZBSfvRDuz/AsyQbPLqtQD0AviA1lpr8Ivdhw29qWb/40+tA84xuujeld5anlSjzzqMLyBt0NojnqTwKBgBrfQJciTG9/jHCh6WSrXW6Zf/Nl5yiPHLzcvnlwHzrN7KknP7rmAhdNtDJ1O+GMy1FxuOxJnzgq9l48oyFgwW/tNdgOs9d0H91LpMCYvKBavWLVO1FWQWFTAHic4cD9XYf0oLm36BsxWZ0SGz3pDFlmvIsIEq9KQ2ju5a3PumNRAoGAPt2oRZ+qqNI7rC+GEzYCxQC1W0ehblyTSOfg8m8YbolUwmKWVJ36KQWsnOR63A1QS2XMbs3lQRK4sMajIKbHscwR4aDvIp7C2jslFhcDfZE8tcJSwLg4fBDH8E+eNz5BWndqP0L+qWdGikOlO9PJtjZi1rmlXbEmHHTpkN5W2rE=" + } +} \ No newline at end of file diff --git a/test/js-ipfs-repo/datastore/000002.ldb b/test/js-ipfs-repo/datastore/000002.ldb new file mode 100644 index 0000000000000000000000000000000000000000..fc04d660e9ab26bb5d27b3b6b33d212bd1e34f0b GIT binary patch literal 211 zcmZROP}0xIPfpCyPs_|nEiTH>FJWY200SX~$SEzBZ1I|j^S%f*oZe})FX(9Q%%jt? zQy&NlS$>$Y`#n34QbA^3F(V^P!?OjhZT>&P<~cuU%i;RLe(ysy@1_qD4Q&dq4R$uX zbPxoo11sKKRu>QF9Qd?w0|O_M0pouNU?_u2-8}VndNoG}qZmkp00{rx(5+JHej5PO CoIXSV literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/datastore/000005.ldb b/test/js-ipfs-repo/datastore/000005.ldb new file mode 100644 index 0000000000000000000000000000000000000000..63d9d260b936415ee42b5aeda6c79fb941664398 GIT binary patch literal 1120 zcmZQDSjViNS&&z(FQg!8*qs;0^V3RMWIveo0t|Brg#j3VRkisv@!=|IgJvNC_dNx9y@%e`@arzW`upR#0^RR3Lu1P3 zgny1+s9JOHtH$|N^V2^pIaU+*`j0Vdf^UEA=Y(L*NL7==e45ubJUDbI<}5@1nZ5i@ zU-f@onB_6=BHQa-tVc_VRZiT0AST5ib<){%%1r}PkJe@8+6(<3{`CFeMP?prC*c znwwdgM`+7O$B&?~72w>77RV zf{y0SJUT5q^?{&}<%b!&-?Q^56=dcWGqS)myonCBGLo1n?_9oP$D!cvsdqIm%dJT< zUeCYn&kfrf>=lfOBoQtW6k&$7W$0~c7gOL@IORq4N$7y_CFLmFs-1D4yKkEePa;Yx&V3}9e z6J;wFzbL(7XX!B6*Rdrw=+4dxeU2>!j1QaGe>~sK(>yOETD?v4ss5$s3VXC14~0|~ z@|s=JZrGS)y@`hY)f%XfBnQv@ZgL4too7C-W6;9@lKgp&)Y9xu#YGC zq+Q(oPkf)J&zo^7bJ>~t)Xqs~*IU@HQqP#^{d0FztkiZ*y~k%OjhiRCYiY>$1@7+Z zzc{J2k%`0O*2zuI)9>zn)wWdsh9cXQc|w1;OnM=;)%wQi`6UM;Jtuj8n*Q?qk;X3q zZaf<6!(B7;zLu#kRKLAQM`r(FbFXM+CT2zkqS6H@{24)n$QSJzIOo8pg&P<+nGG2K hLjXe)2VAZvSyQK-gJTMZI7kRT2>;#Cty1cK8vstu!bkuB literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/datastore/000010.ldb b/test/js-ipfs-repo/datastore/000010.ldb new file mode 100644 index 0000000000000000000000000000000000000000..92077582ca4c1094d81e1c35f8703550afefa5f7 GIT binary patch literal 1143 zcmZQDV9?Jj$Sc+tQjj$4&Wq#uX{9W(pHXd=&LNx2cdi6ocRIV%d~TcDw2tFO0@nWL9`h-bL3Yk+%*afr8@TbPk^RJchgVHA9N_5eTxD41QxO=J9Z^{56JZ); z7-4K;T3DFokyM;!W}Z>zXzZ6K)WGC&@4ao?>Nz(8|F>suXs_zIpd7w&-MpLjzTew7 zw;bnNa{uX~ml~7*<|J4d{FVA*zBjGA&O}w!=F`N7tE3H@g#_I96y)_DdRxPzt0SfDVr1iIeMXL&AqQ0=U2^7|FGm(P2B50#;ggx{k5MHf;A&mO%C&EUfb~C z(50BO4E<;J@;iOi|8-%O$GnSduXnK?Eh$zxasPps6ob@BXVWP+4NN^+mzirX^ndu{ z#t*;Sx0dNGNfK9{8ZLg&Y3{=GymywrWge8Ax zb8_8$u}O-_UoIvnSEU^1zHqkmzk}Jo{lB**@XgG7Tp+=qBxht`XsT;yq-$sxVrXh* zVrgY+u4iCoXlQ9}W@#R!!6>CfIN}*~849wANg!O1*k!T*#RNJcTULFAv{=p%R0p5|p&LRGejGWK}qNY!jl|rkT zI2eSMF&Q*5DjD#yacZ@Bw0-AgWMpJzU~XdMX8?+GF*PwVGHml$WiN9uvO;p{6~^*7 zjnC_)?mL!yo-^-fogiH<)#V;6^U8XnY{lXir8n#>9VYuaw!{YA*;%2_v890VVH5k0 z=ev2D=Y>S8w`o4rzw}&TkCx-1kjg?{vrD=>%@*e)9$i}<^>(k?*#eGjDX!_SpO^_A zd~u&uKT_JeV$DC^DKqPN`vna4@g$$Li@X1c@ALF|GfrhLJ5!(9IqB?r3;R{-856yK z?v9F;+ODbh_-v(d^JI4|4f(#n-Cg|`C$%;*ad_N1xygC@-QBO+mg?V7WV_2So6|Ky~%*a4g zWP`$=5k#;l^#s5<2R<#_z`)6D!1x~m7-n+7<=&pJcTULFAv{=p%R0p5|p&LRGe zjGPcN*cn(K5N|UhD@2zb$Y#P0Wn_a{!gQHYLO&-zIWb2+Ei)&zxF|orgpmm%D-4w= M$jmEdWQ2$T0KrCJEdT%j literal 0 HcmV?d00001 diff --git a/test/js-ipfs-repo/version b/test/js-ipfs-repo/version new file mode 100644 index 0000000000..7813681f5b --- /dev/null +++ b/test/js-ipfs-repo/version @@ -0,0 +1 @@ +5 \ No newline at end of file From e2076d42ba2f90ff5e129993912945dfa9a55e78 Mon Sep 17 00:00:00 2001 From: Yahya Date: Mon, 4 Sep 2017 17:25:52 +0200 Subject: [PATCH 09/14] Using unix-fs to detect dirs, replacing object.get with DAG.get, CID checks License: MIT Signed-off-by: Yahya --- src/http/gateway/resolver.js | 30 +++++++++++++++--------------- 1 file changed, 15 insertions(+), 15 deletions(-) diff --git a/src/http/gateway/resolver.js b/src/http/gateway/resolver.js index bc9ec1d5a0..f2dc4a07be 100644 --- a/src/http/gateway/resolver.js +++ b/src/http/gateway/resolver.js @@ -3,10 +3,11 @@ const mh = require('multihashes') const promisify = require('promisify-es6') const eachOfSeries = require('async/eachOfSeries') +const CID = require('cids') +const Unixfs = require('ipfs-unixfs') const debug = require('debug') const log = debug('jsipfs:http-gateway:resolver') log.error = debug('jsipfs:http-gateway:resolver:error') - const html = require('./utils/html') const PathUtil = require('./utils/path') @@ -41,32 +42,31 @@ const resolveMultihash = promisify((ipfs, path, callback) => { const partsLength = parts.length let currentMultihash = parts[0] - + let currentCid eachOfSeries(parts, (multihash, currentIndex, next) => { - // throws error when invalid multihash is passed - mh.validate(mh.fromB58String(currentMultihash)) + // throws error when invalid CID is passed + try { + currentCid = new CID(mh.fromB58String(currentMultihash)) + } catch (e) { + if (e) throw e + } + log('currentMultihash: ', currentMultihash) log('currentIndex: ', currentIndex, '/', partsLength) - ipfs.object.get(currentMultihash, { enc: 'base58' }, (err, dagNode) => { + ipfs.dag.get(currentCid, (err, result) => { if (err) { return next(err) } + let dagNode = result.value if (currentIndex === partsLength - 1) { - // leaf node - log('leaf node: ', currentMultihash) - - // TODO: Check if it is a directory by using Unixfs Type, right now - // it won't detect empty dirs - if (dagNode.links && - dagNode.links.length > 0 && - dagNode.links[0].name.length > 0) { - // this is a directory. - + let dagDataObj = Unixfs.unmarshal(dagNode.data) + if (dagDataObj.type === 'directory') { let isDirErr = new Error('This dag node is a directory') // add currentMultihash as a fileName so it can be used by resolveDirectory isDirErr.fileName = currentMultihash return next(isDirErr) } + return next() } From 439cf1b5c12d86ec33255590d95c2d99e3ce2275 Mon Sep 17 00:00:00 2001 From: Yahya Date: Mon, 4 Sep 2017 18:29:27 +0200 Subject: [PATCH 10/14] rename checkHash -> checkCID License: MIT Signed-off-by: Yahya --- src/http/gateway/resources/gateway.js | 6 +++--- src/http/gateway/routes/gateway.js | 4 ++-- 2 files changed, 5 insertions(+), 5 deletions(-) diff --git a/src/http/gateway/resources/gateway.js b/src/http/gateway/resources/gateway.js index 37ccba22b4..5c472d1492 100644 --- a/src/http/gateway/resources/gateway.js +++ b/src/http/gateway/resources/gateway.js @@ -12,8 +12,8 @@ const PathUtils = require('../utils/path') const Stream = require('stream') module.exports = { - checkHash: (request, reply) => { - if (!request.params.hash) { + checkCID: (request, reply) => { + if (!request.params.cid) { return reply({ Message: 'Path Resolve error: path must contain at least one component', Code: 0 @@ -21,7 +21,7 @@ module.exports = { } return reply({ - ref: `/ipfs/${request.params.hash}` + ref: `/ipfs/${request.params.cid}` }) }, handler: (request, reply) => { diff --git a/src/http/gateway/routes/gateway.js b/src/http/gateway/routes/gateway.js index e1c0f3222f..5eddeb7c21 100644 --- a/src/http/gateway/routes/gateway.js +++ b/src/http/gateway/routes/gateway.js @@ -7,10 +7,10 @@ module.exports = (server) => { gateway.route({ method: '*', - path: '/ipfs/{hash*}', + path: '/ipfs/{cid*}', config: { pre: [ - { method: resources.gateway.checkHash, assign: 'args' } + { method: resources.gateway.checkCID, assign: 'args' } ], handler: resources.gateway.handler } From 75c537ff8de9901c09abcc3743a095ff155dd399 Mon Sep 17 00:00:00 2001 From: Yahya Date: Tue, 5 Sep 2017 12:46:30 +0200 Subject: [PATCH 11/14] gateway tests: init a fresh repo --- test/gateway/index.js | 19 +-- ...HSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data | 4 - ...DDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data | 23 ---- ...NLTXZM4PW4VEFWQBJ4ACZQAS37BLGL4HUO5XY.data | Bin 309 -> 0 bytes ...B5FJNHZPTSVA7IB6OHXSQ2XSVEEKMKK6RT75I.data | Bin 101 -> 0 bytes ...7E32LQAL6236OUKZTMHPQSFIXPWXNZHQOV7JQ.data | 55 --------- ...SIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data | 3 - ...YVY4BA22FPHUIODJEXS4LCTQDWA275XAJDAPI.data | Bin 309 -> 0 bytes ...LZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data | 4 - ...774EZOYCYNHPRVFD53ZSAU7237F67XDSQGCYQ.data | Bin 10807 -> 0 bytes ...RTG5JS3JCUJDQM2DRB5RVF33DCUUOFJNGVDUI.data | Bin 10849 -> 0 bytes ...YNRNMP2VDKWBWGAEDDEJDACM3SGG3VIANDDXI.data | Bin 10765 -> 0 bytes ...LJNXLLHZOPGSL2PBC65D4UIVWM6TI5F5TAFNI.data | 24 ---- ...VXLWEU4FWPVGT24VJT7DUZPTNLF25N25IGGQA.data | 4 - ...WJNESD7XHQSXA5EGJVNTPVHD7444C2KLKXHDI.data | Bin 10765 -> 0 bytes ...AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data | 8 -- ...2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data | 3 - ...FZU7XAJL2BMIHGVB5ZR2IOKOSTRMLIKPB6K5I.data | Bin 402 -> 0 bytes ...TJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data | 9 -- ...RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data | 115 ------------------ ...33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data | 3 - ...OUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data | 27 ---- ...WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data | 3 - ...ML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data | 36 ------ ...HTQ7R247ZI7KJWP3QWPQYS43LFULQC5ANLQFI.data | Bin 10765 -> 0 bytes ...GPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data | 4 - ...57JSEZN64SIJ5OIHSGJG4TJSSJLGI3PBJLQVI.data | 0 ...JOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data | 28 ----- ...OV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data | 3 - ...LP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data | 3 - test/js-ipfs-repo/blocks/SHARDING | 1 - ...D6VA64TGWP67KYA3AIMBUMVWGZ5AQN2L2HSWQ.data | Bin 10765 -> 0 bytes ...ANT6IBNTFN7WR5RPD5F6GN6MBKUUO25DNOTWQ.data | Bin 10891 -> 0 bytes ...NAKDMN4RM26SQETENO5EXBAE5MNXM5DXOBUMI.data | 3 - ...NH7VEGIQJRPL6J7FT2XYVKAXT4MQPXXPUYUNY.data | Bin 10765 -> 0 bytes ...XCX5XMZLMGTYVODWPEDIW7JYEG7YXBIA7IUWY.data | 3 - ...LOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data | 114 ----------------- ...2BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data | 2 - ...2FSMG76VV256I4PXBULZZ5ASNLK4FL4EG7XOI.data | Bin 10849 -> 0 bytes ...5KVN6ALXC6QRHK2X4R6EUFRMBB5OSFO2FUYDQ.data | 3 - ...EBXFD5QN6MTI5YBYMCVQJDXPKCOVR6RMLHZFQ.data | Bin 10807 -> 0 bytes test/js-ipfs-repo/blocks/_README | 22 ---- test/js-ipfs-repo/config | 34 ------ test/js-ipfs-repo/datastore/000002.ldb | Bin 211 -> 0 bytes test/js-ipfs-repo/datastore/000005.ldb | Bin 1120 -> 0 bytes test/js-ipfs-repo/datastore/000010.ldb | Bin 1143 -> 0 bytes test/js-ipfs-repo/datastore/CURRENT | 1 - test/js-ipfs-repo/datastore/LOCK | 0 test/js-ipfs-repo/datastore/LOG | 29 ----- test/js-ipfs-repo/datastore/LOG.old | 10 -- test/js-ipfs-repo/datastore/MANIFEST-000002 | Bin 50 -> 0 bytes test/js-ipfs-repo/datastore/MANIFEST-000014 | Bin 314 -> 0 bytes test/js-ipfs-repo/version | 1 - 53 files changed, 4 insertions(+), 597 deletions(-) delete mode 100644 test/js-ipfs-repo/blocks/2F/CIQEUWUVLBXVFYSYCHHSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data delete mode 100644 test/js-ipfs-repo/blocks/5V/CIQFFRR4O52TS2Z7QLDDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data delete mode 100644 test/js-ipfs-repo/blocks/5X/CIQJ23BL4UHXA2KTI6NLTXZM4PW4VEFWQBJ4ACZQAS37BLGL4HUO5XY.data delete mode 100644 test/js-ipfs-repo/blocks/75/CIQMB7DLJFKD267QJ2B5FJNHZPTSVA7IB6OHXSQ2XSVEEKMKK6RT75I.data delete mode 100644 test/js-ipfs-repo/blocks/7J/CIQKKLBWAIBQZOIS5X7E32LQAL6236OUKZTMHPQSFIXPWXNZHQOV7JQ.data delete mode 100644 test/js-ipfs-repo/blocks/AE/CIQONICFQZH7QVU6IPSIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data delete mode 100644 test/js-ipfs-repo/blocks/AP/CIQHAKDLTL5GMIFGN5YVY4BA22FPHUIODJEXS4LCTQDWA275XAJDAPI.data delete mode 100644 test/js-ipfs-repo/blocks/C4/CIQDDZ5EDQK5AP7LRTLZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data delete mode 100644 test/js-ipfs-repo/blocks/CY/CIQDMKFEUGKSLXMEXO774EZOYCYNHPRVFD53ZSAU7237F67XDSQGCYQ.data delete mode 100644 test/js-ipfs-repo/blocks/DU/CIQLBK52T5EHVHZY5URTG5JS3JCUJDQM2DRB5RVF33DCUUOFJNGVDUI.data delete mode 100644 test/js-ipfs-repo/blocks/DX/CIQPDQJBGYDZNMOAGGYNRNMP2VDKWBWGAEDDEJDACM3SGG3VIANDDXI.data delete mode 100644 test/js-ipfs-repo/blocks/FN/CIQIXBZMUTXFC5QIGMLJNXLLHZOPGSL2PBC65D4UIVWM6TI5F5TAFNI.data delete mode 100644 test/js-ipfs-repo/blocks/GQ/CIQH7OEYWXL34RWYL7VXLWEU4FWPVGT24VJT7DUZPTNLF25N25IGGQA.data delete mode 100644 test/js-ipfs-repo/blocks/HD/CIQDDVW2EZIJF4NQH7WJNESD7XHQSXA5EGJVNTPVHD7444C2KLKXHDI.data delete mode 100644 test/js-ipfs-repo/blocks/IL/CIQJFGRQHQ45VCQLM7AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data delete mode 100644 test/js-ipfs-repo/blocks/IR/CIQERMRAAFXUAUOX3V2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data delete mode 100644 test/js-ipfs-repo/blocks/K5/CIQPW4MAGTUNEBGZCEFZU7XAJL2BMIHGVB5ZR2IOKOSTRMLIKPB6K5I.data delete mode 100644 test/js-ipfs-repo/blocks/LG/CIQJBQD2O6K4CGJVCCTJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data delete mode 100644 test/js-ipfs-repo/blocks/M4/CIQOLBQZSZAODJGGH6RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data delete mode 100644 test/js-ipfs-repo/blocks/O6/CIQOYW2THIZBRGI7IN33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data delete mode 100644 test/js-ipfs-repo/blocks/OO/CIQBT4N7PS5IZ5IG2ZOUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data delete mode 100644 test/js-ipfs-repo/blocks/PJ/CIQB4F7VKKQDXHMXX6WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data delete mode 100644 test/js-ipfs-repo/blocks/PU/CIQJF2E6OPOEYYEVHYML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data delete mode 100644 test/js-ipfs-repo/blocks/QF/CIQGPALRQ24P6NS4OWHTQ7R247ZI7KJWP3QWPQYS43LFULQC5ANLQFI.data delete mode 100644 test/js-ipfs-repo/blocks/QP/CIQNLGENZXNRUMUHZYGPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data delete mode 100644 test/js-ipfs-repo/blocks/QV/CIQOHMGEIKMPYHAUTL57JSEZN64SIJ5OIHSGJG4TJSSJLGI3PBJLQVI.data delete mode 100644 test/js-ipfs-repo/blocks/R3/CIQBED3K6YA5I3QQWLJOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data delete mode 100644 test/js-ipfs-repo/blocks/S2/CIQPF3CHDB5GQ5ZBISOV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data delete mode 100644 test/js-ipfs-repo/blocks/S5/CIQHBGZNZRPWVEFNMTLP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data delete mode 100644 test/js-ipfs-repo/blocks/SHARDING delete mode 100644 test/js-ipfs-repo/blocks/SW/CIQHPUVCWD6JA6AFUVD6VA64TGWP67KYA3AIMBUMVWGZ5AQN2L2HSWQ.data delete mode 100644 test/js-ipfs-repo/blocks/TW/CIQFEAGMNNXXTYKYQSANT6IBNTFN7WR5RPD5F6GN6MBKUUO25DNOTWQ.data delete mode 100644 test/js-ipfs-repo/blocks/UM/CIQOGPM2T3ZJMYJKYNNAKDMN4RM26SQETENO5EXBAE5MNXM5DXOBUMI.data delete mode 100644 test/js-ipfs-repo/blocks/UN/CIQOMBKARLB7PAITVSNH7VEGIQJRPL6J7FT2XYVKAXT4MQPXXPUYUNY.data delete mode 100644 test/js-ipfs-repo/blocks/UW/CIQDVKITASFS55MC2TXCX5XMZLMGTYVODWPEDIW7JYEG7YXBIA7IUWY.data delete mode 100644 test/js-ipfs-repo/blocks/VO/CIQGFTQ7FSI2COUXWWLOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data delete mode 100644 test/js-ipfs-repo/blocks/X3/CIQFTFEEHEDF6KLBT32BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data delete mode 100644 test/js-ipfs-repo/blocks/XO/CIQJGO2B2N75IUEM372FSMG76VV256I4PXBULZZ5ASNLK4FL4EG7XOI.data delete mode 100644 test/js-ipfs-repo/blocks/YD/CIQENVCICS44LLYUDQ5KVN6ALXC6QRHK2X4R6EUFRMBB5OSFO2FUYDQ.data delete mode 100644 test/js-ipfs-repo/blocks/ZF/CIQLBS5HG4PRCRQ7O4EBXFD5QN6MTI5YBYMCVQJDXPKCOVR6RMLHZFQ.data delete mode 100644 test/js-ipfs-repo/blocks/_README delete mode 100644 test/js-ipfs-repo/config delete mode 100644 test/js-ipfs-repo/datastore/000002.ldb delete mode 100644 test/js-ipfs-repo/datastore/000005.ldb delete mode 100644 test/js-ipfs-repo/datastore/000010.ldb delete mode 100644 test/js-ipfs-repo/datastore/CURRENT delete mode 100644 test/js-ipfs-repo/datastore/LOCK delete mode 100644 test/js-ipfs-repo/datastore/LOG delete mode 100644 test/js-ipfs-repo/datastore/LOG.old delete mode 100644 test/js-ipfs-repo/datastore/MANIFEST-000002 delete mode 100644 test/js-ipfs-repo/datastore/MANIFEST-000014 delete mode 100644 test/js-ipfs-repo/version diff --git a/test/gateway/index.js b/test/gateway/index.js index 65d258bee9..9a77f9903c 100644 --- a/test/gateway/index.js +++ b/test/gateway/index.js @@ -6,34 +6,23 @@ const dirtyChai = require('dirty-chai') const expect = chai.expect chai.use(dirtyChai) const API = require('../../src/http') -const ncp = require('ncp').ncp -const path = require('path') -const clean = require('../utils/clean') describe('HTTP Gateway', () => { - const repoExample = path.join(__dirname, '../js-ipfs-repo') - const repoTests = path.join(__dirname, '../repo-tests-run') - let http = {} let gateway before((done) => { - http.api = new API(repoTests) - - ncp(repoExample, repoTests, (err) => { - expect(err).to.not.exist() + http.api = new API() - http.api.start(false, () => { - gateway = http.api.server.select('Gateway') - done() - }) + http.api.start(true, () => { + gateway = http.api.server.select('Gateway') + done() }) }) after((done) => { http.api.stop((err) => { expect(err).to.not.exist() - clean(repoTests) done() }) }) diff --git a/test/js-ipfs-repo/blocks/2F/CIQEUWUVLBXVFYSYCHHSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data b/test/js-ipfs-repo/blocks/2F/CIQEUWUVLBXVFYSYCHHSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data deleted file mode 100644 index 4145619655..0000000000 --- a/test/js-ipfs-repo/blocks/2F/CIQEUWUVLBXVFYSYCHHSCRTXCYHGIOBXKWUMKFR3UPAFHQ5WK5362FQ.data +++ /dev/null @@ -1,4 +0,0 @@ - -ys# js-ipfs-repo -Implementation of the IPFS repo spec (https://github.com/ipfs/specs/tree/master/repo) in JavaScript -s \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/5V/CIQFFRR4O52TS2Z7QLDDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data b/test/js-ipfs-repo/blocks/5V/CIQFFRR4O52TS2Z7QLDDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data deleted file mode 100644 index 951bfe0400..0000000000 --- a/test/js-ipfs-repo/blocks/5V/CIQFFRR4O52TS2Z7QLDDTF32OIR4FWLKT5YLL7MLDVIT7DC3NHOK5VA.data +++ /dev/null @@ -1,23 +0,0 @@ - - IPFS Alpha Security Notes - -We try hard to ensure our system is safe and robust, but all software -has bugs, especially new software. This distribution is meant to be an -alpha preview, don't use it for anything mission critical. - -Please note the following: - -- This is alpha software and has not been audited. It is our goal - to conduct a proper security audit once we close in on a 1.0 release. - -- ipfs is a networked program, and may have serious undiscovered - vulnerabilities. It is written in Go, and we do not execute any - user provided data. But please point any problems out to us in a - github issue, or email security@ipfs.io privately. - -- ipfs uses encryption for all communication, but it's NOT PROVEN SECURE - YET! It may be totally broken. For now, the code is included to make - sure we benchmark our operations with encryption in mind. In the future, - there will be an "unsafe" mode for high performance intranet apps. - If this is a blocking feature for you, please contact us. - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/5X/CIQJ23BL4UHXA2KTI6NLTXZM4PW4VEFWQBJ4ACZQAS37BLGL4HUO5XY.data b/test/js-ipfs-repo/blocks/5X/CIQJ23BL4UHXA2KTI6NLTXZM4PW4VEFWQBJ4ACZQAS37BLGL4HUO5XY.data deleted file mode 100644 index b799cf6b2236c91f8224b9fefe43aca16c773490..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 309 zcmWgA<5Ch*SgK>j#LTl(=mjz6J*Z#Z`mUeibke1%>*qt`A@ymo*6O-~wOC)CS z3K@XZPnu<5V|lBKJN>}4>2CTNRSd^0_H1qVw~ozKk4Ii%i@p$ha(-S(VseSZ2}U7Z zkd6rls>-Jxlr$AsHtpiyKm2l@?&p6zQA*^T_KfrP>FKF6gjh0Ca|$F5F$tM~ls|2g zoaXRw$uaxIJs~DLpX;8StGsyP<=b&`P8V3!Rm2K~xC=`&le2Y;OA?DpBo6Qj>4S6$ z@n?Nwypkud>C!{F>oTF2;tiY*sWLv16>@vN@a5$jZ-m&2QWH~hQzg!E2$_R41Rb*} zFSX3JZ#rf+Kwo-;;PadOz%uNME$6+{JHE$$U!OUW!sO)in> z4G}T|X$WFClbv7rFrua5=1<0)Q}=J#b|1g=` -- browsers or extensions can learn to use `ipfs://` directly -- hash-addressed content guarantees authenticity - -IPFS is modular: -- connection layer over any network protocol -- routing layer -- uses a routing layer DHT (kademlia/coral) -- uses a path-based naming service -- uses bittorrent-inspired block exchange - -IPFS uses crypto: -- cryptographic-hash content addressing -- block-level deduplication -- file integrity + versioning -- filesystem-level encryption + signing support - -IPFS is p2p: -- worldwide peer-to-peer file transfers -- completely decentralized architecture -- **no** central point of failure - -IPFS is a cdn: -- add a file to the filesystem locally, and it's now available to the world -- caching-friendly (content-hash naming) -- bittorrent-based bandwidth distribution - -IPFS has a name service: -- IPNS, an SFS inspired name system -- global namespace based on PKI -- serves to build trust chains -- compatible with other NSes -- can map DNS, .onion, .bit, etc to IPNS - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/AE/CIQONICFQZH7QVU6IPSIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data b/test/js-ipfs-repo/blocks/AE/CIQONICFQZH7QVU6IPSIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data deleted file mode 100644 index 6860441aa1..0000000000 --- a/test/js-ipfs-repo/blocks/AE/CIQONICFQZH7QVU6IPSIM3AK7AD554D3BWZPAGEAQYQOWMFZQDUUAEI.data +++ /dev/null @@ -1,3 +0,0 @@ -/ -" gq6\u8~:6~gZ.directT2 -" 6(%݄.Ӿ5(ab recursiveT \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/AP/CIQHAKDLTL5GMIFGN5YVY4BA22FPHUIODJEXS4LCTQDWA275XAJDAPI.data b/test/js-ipfs-repo/blocks/AP/CIQHAKDLTL5GMIFGN5YVY4BA22FPHUIODJEXS4LCTQDWA275XAJDAPI.data deleted file mode 100644 index 74de75af616ff437b5c7a3a5272281eac2146b6c..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 309 zcmWgA<5Ch*SgK>j#LTl(=mjz6J*Z#Z`mUeibke1%>*qt`A@ymo*6O-~wOC)CS z3K@XZPnu<5V|lBKJN>}4>2CTNRSd^0_H1qVw~ozKk4Ii%i@p$ha(-S(VseSZ2}U7Z zkd6rls>-Jxlr$AsHtpiyKm2l@?&p6zQA*^T_KfrP>FKF6gjh0Ca|$F5F$tM~lqa2& z*O|D`YWmh`FN(J=HpxnHW4)f}dn)*>RpfVzsIJ{Y+=Zo?$=SNaC5c5P5({~S^g+6W z__Mw-Uda>Kbm^hob(zph@di$ZR2iSh3b{RB`110NH$rShsfj7MsS@Wngv>!2f{xjg zms)1qHyyK_UR9)g=w{aZf~|kMWdrSdqBHNTyCTF_oSIx(lvz@#o0ngbS}f7b!NtVE F2moEFc%uLS diff --git a/test/js-ipfs-repo/blocks/C4/CIQDDZ5EDQK5AP7LRTLZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data b/test/js-ipfs-repo/blocks/C4/CIQDDZ5EDQK5AP7LRTLZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data deleted file mode 100644 index ecce1053f6..0000000000 --- a/test/js-ipfs-repo/blocks/C4/CIQDDZ5EDQK5AP7LRTLZHQZUR2R3GECRFV3WPKNL7PL2SKFIL2LXC4Y.data +++ /dev/null @@ -1,4 +0,0 @@ -5 -" ׾F_uؔlzS?|ڲPc@ js-ipfs-repo - - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/CY/CIQDMKFEUGKSLXMEXO774EZOYCYNHPRVFD53ZSAU7237F67XDSQGCYQ.data b/test/js-ipfs-repo/blocks/CY/CIQDMKFEUGKSLXMEXO774EZOYCYNHPRVFD53ZSAU7237F67XDSQGCYQ.data deleted file mode 100644 index bbe6bda78d0a52f6d87da578f452d43718b9aa9f..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10807 zcmeI&tqQ_m6vpvsQC>mhU=TmxWeCpZ4H&eDU=nNw5wR?6Yue0K5w>X*qZns5SS{Xw z-hn1RK)egXzn^qC&+{usEMpXsYYQnaL-~-$SEZaY\IzxEElM/fLICENSE1 -" JZXoRX!Fwd87U;SöWw README.md{ - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/HD/CIQDDVW2EZIJF4NQH7WJNESD7XHQSXA5EGJVNTPVHD7444C2KLKXHDI.data b/test/js-ipfs-repo/blocks/HD/CIQDDVW2EZIJF4NQH7WJNESD7XHQSXA5EGJVNTPVHD7444C2KLKXHDI.data deleted file mode 100644 index 5ea0edda6f82982c276a4a3813a50c0e35121c68..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10765 zcmeIzF%AJy6a~;Rm3)!Nk3xb(CK9$mt(wFR^jf72s6=9D)srdd6*gc6EI~I4+fbbT zRX6Y66Q?1IqgQpGAHRd{ss22Q^=&*UXYbW{zqlNV{wBXJW712aODXvf?XV11kq9)T zAq{CrLmJYMhBTxh4QWV28q$!4G^8O7X-GpF(vXHUq#+GyNJARZkcKp*;U8!iTA|tK KY}4Vho$h|}9P~&4 diff --git a/test/js-ipfs-repo/blocks/IL/CIQJFGRQHQ45VCQLM7AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data b/test/js-ipfs-repo/blocks/IL/CIQJFGRQHQ45VCQLM7AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data deleted file mode 100644 index 62d1c2979b..0000000000 --- a/test/js-ipfs-repo/blocks/IL/CIQJFGRQHQ45VCQLM7AJNF2GF5UHUAGGHC6LLAH6VYDEKLQMD4QLILY.data +++ /dev/null @@ -1,8 +0,0 @@ - -Come hang out in our IRC chat room if you have any questions. - -Contact the ipfs dev team: -- Bugs: https://github.com/ipfs/go-ipfs/issues -- Help: irc.freenode.org/#ipfs -- Email: dev@ipfs.io - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/IR/CIQERMRAAFXUAUOX3V2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data b/test/js-ipfs-repo/blocks/IR/CIQERMRAAFXUAUOX3V2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data deleted file mode 100644 index 4a26a4ac35..0000000000 --- a/test/js-ipfs-repo/blocks/IR/CIQERMRAAFXUAUOX3V2DCW7R77FRIVHQ3V5OIPPS3XQBX34KRPNOIRQ.data +++ /dev/null @@ -1,3 +0,0 @@ -/ -" !61صF2$`7#u@1directT2 -" Hz8#3u2ED ƥ*QKMQ recursiveT \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/K5/CIQPW4MAGTUNEBGZCEFZU7XAJL2BMIHGVB5ZR2IOKOSTRMLIKPB6K5I.data b/test/js-ipfs-repo/blocks/K5/CIQPW4MAGTUNEBGZCEFZU7XAJL2BMIHGVB5ZR2IOKOSTRMLIKPB6K5I.data deleted file mode 100644 index e1cd3e3e21d1e86994739a1b26b18c77aa98f034..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 402 zcmWgA<5Ch*SgK>j#LTl(=mjz6J*Z#Z`mUeibke1%>*qt`A@ymo*6O-~wOC)CS z3K@XZPnu<5V|lBKJN>}4>2CTNRSd^0_H1qVw~ozKk4Ii%i@p$ha(-S(VseSZ2}U6u zkd9Ap+$E|q$`xJa#!6jHIxzjpl!P0h>MB2GPV&4rkBc)?h$SUIxmdzcNEf7f!hx#t zsRt!Z1(r>_`1cRLoTvNwA5WALIj242ynT9l>I@;4jMSV0i9<|6CLrZc+a#wsJX~_j zesNET$-x#mt z32eIbQ0}@+=%siAr$efYPh^GMo-cfP`NkU|wxZO;l-yK_a~wkEAPtfq_t)&|`O0=J z)>&iK^$Wi5P8mg6XI0#KB6hf@e{Q&y5MOa>a%oX!Nu_RHeo1Pv#8gg*cdpLpJ$qZq tsQnzz_w~A`XB0nMvHZ_Q{$={^NqgFq=2+MYv6SSO7D?P^<6`1q1OQn=pS%D7 diff --git a/test/js-ipfs-repo/blocks/LG/CIQJBQD2O6K4CGJVCCTJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data b/test/js-ipfs-repo/blocks/LG/CIQJBQD2O6K4CGJVCCTJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data deleted file mode 100644 index 71be805f1e..0000000000 --- a/test/js-ipfs-repo/blocks/LG/CIQJBQD2O6K4CGJVCCTJNUP57QHR4SKHZ74OIITBBGLOMCO3ZOLWLGA.data +++ /dev/null @@ -1,9 +0,0 @@ - -Some helpful resources for finding your way around ipfs: - -- quick-start: a quick show of various ipfs features. -- ipfs commands: a list of all commands -- ipfs --help: every command describes itself -- https://github.com/ipfs/go-ipfs -- the src repository -- #ipfs on irc.freenode.org -- the community irc channel - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/M4/CIQOLBQZSZAODJGGH6RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data b/test/js-ipfs-repo/blocks/M4/CIQOLBQZSZAODJGGH6RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data deleted file mode 100644 index f2bf4f8b8d..0000000000 --- a/test/js-ipfs-repo/blocks/M4/CIQOLBQZSZAODJGGH6RYYVBUXHTS3SM5EORZDU63LYPEFUAFE4SBM4I.data +++ /dev/null @@ -1,115 +0,0 @@ - -  # 0.1 - Quick Start - -This is a set of short examples with minimal explanation. It is meant as -a "quick start". Soon, we'll write a longer tour :-) - - -Add a file to ipfs: - - echo "hello world" >hello - ipfs add hello - - -View it: - - ipfs cat - - -Try a directory: - - mkdir foo - mkdir foo/bar - echo "baz" > foo/baz - echo "baz" > foo/bar/baz - ipfs add -r foo - - -View things: - - ipfs ls - ipfs ls /bar - ipfs cat /baz - ipfs cat /bar/baz - ipfs cat /bar - ipfs ls /baz - - -References: - - ipfs refs - ipfs refs -r - ipfs refs --help - - -Get: - - ipfs get -o foo2 - diff foo foo2 - - -Objects: - - ipfs object get - ipfs object get /foo2 - ipfs object --help - - -Pin + GC: - - ipfs pin add - ipfs repo gc - ipfs ls - ipfs pin rm - ipfs repo gc - - -Daemon: - - ipfs daemon (in another terminal) - ipfs id - - -Network: - - (must be online) - ipfs swarm peers - ipfs id - ipfs cat - - -Mount: - - (warning: fuse is finicky!) - ipfs mount - cd /ipfs/ - ls - - -Tool: - - ipfs version - ipfs update - ipfs commands - ipfs config --help - open http://localhost:5001/webui - - -Browse: - - webui: - - http://localhost:5001/webui - - video: - - http://localhost:8080/ipfs/QmVc6zuAneKJzicnJpfrqCH9gSy6bz54JhcypfJYhGUFQu/play#/ipfs/QmTKZgRNwDNZwHtJSjCp6r5FYefzpULfy37JvMt9DwvXse - - images: - - http://localhost:8080/ipfs/QmZpc3HvfjEXvLWGQPWbHk3AjD5j8NEN4gmFN8Jmrd5g83/cs - - markdown renderer app: - - http://localhost:8080/ipfs/QmX7M9CiYXjVeFnkfVGf3y5ixTZ2ACeSGyL1vBJY1HvQPp/mdown - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/O6/CIQOYW2THIZBRGI7IN33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data b/test/js-ipfs-repo/blocks/O6/CIQOYW2THIZBRGI7IN33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data deleted file mode 100644 index 7b58d6c857..0000000000 --- a/test/js-ipfs-repo/blocks/O6/CIQOYW2THIZBRGI7IN33ROGCKOFZLXJJ2MPKYZBTV4H3N7GYHXMAO6A.data +++ /dev/null @@ -1,3 +0,0 @@ -/ -" @ԆDgA7directT2 -" ;APY0k}E=p  recursiveT \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/OO/CIQBT4N7PS5IZ5IG2ZOUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data b/test/js-ipfs-repo/blocks/OO/CIQBT4N7PS5IZ5IG2ZOUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data deleted file mode 100644 index 770348274e..0000000000 --- a/test/js-ipfs-repo/blocks/OO/CIQBT4N7PS5IZ5IG2ZOUGKFK27IE33WKGJNDW2TY3LSBNQ34R6OVOOQ.data +++ /dev/null @@ -1,27 +0,0 @@ - -  IPFS Alpha Security Notes - -We try hard to ensure our system is safe and robust, but all software -has bugs, especially new software. This distribution is meant to be an -alpha preview, don't use it for anything mission critical. - -Please note the following: - -- This is alpha software and has not been audited. It is our goal - to conduct a proper security audit once we close in on a 1.0 release. - -- ipfs is a networked program, and may have serious undiscovered - vulnerabilities. It is written in Go, and we do not execute any - user provided data. But please point any problems out to us in a - github issue, or email security@ipfs.io privately. - -- security@ipfs.io GPG key: - - 4B9665FB 92636D17 7C7A86D3 50AAE8A9 59B13AF3 - - https://pgp.mit.edu/pks/lookup?op=get&search=0x50AAE8A959B13AF3 - -- ipfs uses encryption for all communication, but it's NOT PROVEN SECURE - YET! It may be totally broken. For now, the code is included to make - sure we benchmark our operations with encryption in mind. In the future, - there will be an "unsafe" mode for high performance intranet apps. - If this is a blocking feature for you, please contact us. - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/PJ/CIQB4F7VKKQDXHMXX6WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data b/test/js-ipfs-repo/blocks/PJ/CIQB4F7VKKQDXHMXX6WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data deleted file mode 100644 index 0335563629..0000000000 --- a/test/js-ipfs-repo/blocks/PJ/CIQB4F7VKKQDXHMXX6WYQZTRR5QVLP7VBQYAYW2Y5BAPOOGTW5H2PJQ.data +++ /dev/null @@ -1,3 +0,0 @@ - - Index - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/PU/CIQJF2E6OPOEYYEVHYML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data b/test/js-ipfs-repo/blocks/PU/CIQJF2E6OPOEYYEVHYML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data deleted file mode 100644 index e6ef304bfd..0000000000 --- a/test/js-ipfs-repo/blocks/PU/CIQJF2E6OPOEYYEVHYML4UD5A3R4QRAVBI7NVH3PL64D3IJCWR2SPUQ.data +++ /dev/null @@ -1,36 +0,0 @@ - -WIP - -# 0.0 - Introduction - -Welcome to IPFS! This tour will guide you through a few of the -features of this tool, and the most common commands. Then, it will -immerse you into the world of merkledags and the amazing things -you can do with them. - - -This tour has many parts, and can be taken in different sequences. -Different people learn different ways, so choose your own adventure: - - To start with the concepts, try: - - The Merkle DAG - - Data Structures on the Merkle DAG - - Representing Files with unixfs - - add, cat, ls, refs - ... - - - To start with the examples, try: - - add, cat, ls, refs - - Representing Files with unixfs - - Data Structures on the Merkle DAG - - The Merkle DAG - ... - - - To start with the network, try: - - IPFS Nodes - - Running the daemon - - The Swarm - - The Web - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/QF/CIQGPALRQ24P6NS4OWHTQ7R247ZI7KJWP3QWPQYS43LFULQC5ANLQFI.data b/test/js-ipfs-repo/blocks/QF/CIQGPALRQ24P6NS4OWHTQ7R247ZI7KJWP3QWPQYS43LFULQC5ANLQFI.data deleted file mode 100644 index a8f98693b7d4797d0a55704b16c15a6aa7d34bd5..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10765 zcmeIzF%AJy6a~Szz&}ww{ J&EYejEr0X~^v?hQ diff --git a/test/js-ipfs-repo/blocks/QP/CIQNLGENZXNRUMUHZYGPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data b/test/js-ipfs-repo/blocks/QP/CIQNLGENZXNRUMUHZYGPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data deleted file mode 100644 index 6636930467..0000000000 --- a/test/js-ipfs-repo/blocks/QP/CIQNLGENZXNRUMUHZYGPPLZNZOMHHZVIU76LCD5GF5DWFPEGEKODQPI.data +++ /dev/null @@ -1,4 +0,0 @@ -2 -" sL`>P}D ->ڟo_="u' 0.0-intro - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/QV/CIQOHMGEIKMPYHAUTL57JSEZN64SIJ5OIHSGJG4TJSSJLGI3PBJLQVI.data b/test/js-ipfs-repo/blocks/QV/CIQOHMGEIKMPYHAUTL57JSEZN64SIJ5OIHSGJG4TJSSJLGI3PBJLQVI.data deleted file mode 100644 index e69de29bb2..0000000000 diff --git a/test/js-ipfs-repo/blocks/R3/CIQBED3K6YA5I3QQWLJOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data b/test/js-ipfs-repo/blocks/R3/CIQBED3K6YA5I3QQWLJOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data deleted file mode 100644 index 389e111776..0000000000 --- a/test/js-ipfs-repo/blocks/R3/CIQBED3K6YA5I3QQWLJOCHWXDRK5EXZQILBCKAPEDUJENZ5B5HJ5R3A.data +++ /dev/null @@ -1,28 +0,0 @@ - -Hello and Welcome to IPFS! - -██╗██████╗ ███████╗███████╗ -██║██╔══██╗██╔════╝██╔════╝ -██║██████╔╝█████╗ ███████╗ -██║██╔═══╝ ██╔══╝ ╚════██║ -██║██║ ██║ ███████║ -╚═╝╚═╝ ╚═╝ ╚══════╝ - -If you're seeing this, you have successfully installed -IPFS and are now interfacing with the ipfs merkledag! - - ------------------------------------------------------- -| Warning: | -| This is alpha software. Use at your own discretion! | -| Much is missing or lacking polish. There are bugs. | -| Not yet secure. Read the security notes for more. | - ------------------------------------------------------- - -Check out some of the other files in this directory: - - ./about - ./help - ./quick-start <-- usage examples - ./readme <-- this file - ./security-notes - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/S2/CIQPF3CHDB5GQ5ZBISOV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data b/test/js-ipfs-repo/blocks/S2/CIQPF3CHDB5GQ5ZBISOV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data deleted file mode 100644 index b137a86405..0000000000 --- a/test/js-ipfs-repo/blocks/S2/CIQPF3CHDB5GQ5ZBISOV2GWVMLAJPVEUMDMFKJZE7CMZESO6TYFAS2I.data +++ /dev/null @@ -1,3 +0,0 @@ -- -" R;fqaU 0 [X@8ӷOindex - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/S5/CIQHBGZNZRPWVEFNMTLP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data b/test/js-ipfs-repo/blocks/S5/CIQHBGZNZRPWVEFNMTLP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data deleted file mode 100644 index 3a99c365f0..0000000000 --- a/test/js-ipfs-repo/blocks/S5/CIQHBGZNZRPWVEFNMTLP4OS5EAVHFMCX2HD7FZUC2B3WUU3D4LGKS5A.data +++ /dev/null @@ -1,3 +0,0 @@ -4 -" Y9_)a˹2RmŖke9 js-ipfs-repo - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/SHARDING b/test/js-ipfs-repo/blocks/SHARDING deleted file mode 100644 index a153331dac..0000000000 --- a/test/js-ipfs-repo/blocks/SHARDING +++ /dev/null @@ -1 +0,0 @@ -/repo/flatfs/shard/v1/next-to-last/2 diff --git a/test/js-ipfs-repo/blocks/SW/CIQHPUVCWD6JA6AFUVD6VA64TGWP67KYA3AIMBUMVWGZ5AQN2L2HSWQ.data b/test/js-ipfs-repo/blocks/SW/CIQHPUVCWD6JA6AFUVD6VA64TGWP67KYA3AIMBUMVWGZ5AQN2L2HSWQ.data deleted file mode 100644 index 78421c81c29ed8c18921f686ea3292e3e7300668..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10765 zcmeIzAqoOf6a~;x48ym`#~`RM41#V$vuVH`*sVtQAcCXSECW-t8{CI3!R7?FVL1Jz z%e(i)S;%Adsve7DGwPqdpJ%zbPo~%Tdwt$7FNbotEpDrr43g+lN<931^R@s0 diff --git a/test/js-ipfs-repo/blocks/TW/CIQFEAGMNNXXTYKYQSANT6IBNTFN7WR5RPD5F6GN6MBKUUO25DNOTWQ.data b/test/js-ipfs-repo/blocks/TW/CIQFEAGMNNXXTYKYQSANT6IBNTFN7WR5RPD5F6GN6MBKUUO25DNOTWQ.data deleted file mode 100644 index 10aa2ae4f1291a62edd84980f1c05b09b95f78ff..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10891 zcmWgA;8GG&c)a0=(~LhdBC~#fIWaSTr;7SI$0sSXC;Kd!I#aqLXh*0Jg9L-nsLIh$ z8ciXixns0s7%de?YlP9-aI`iYtqn(O!_nGsv^E^A4M%Ik(b{mdHXN-DM{C2;+HkZs z9IXvUYs1mnaI`iYtqn(O!_nGsv^E^A4M%Ik(b{mdHXN-DM{C2;+HkZs9IXvUYs1mn zaI`iYG__&mlom_2c+JFlUxXS??=;#MbToJ7(P`PK4+MoQKg`(u9q~6w(px+ cb8FM| recursiveT \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/VO/CIQGFTQ7FSI2COUXWWLOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data b/test/js-ipfs-repo/blocks/VO/CIQGFTQ7FSI2COUXWWLOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data deleted file mode 100644 index 2dd80560a2..0000000000 --- a/test/js-ipfs-repo/blocks/VO/CIQGFTQ7FSI2COUXWWLOQ45VUM2GUZCGAXLWCTOKKPGTUWPXHBNIVOY.data +++ /dev/null @@ -1,114 +0,0 @@ - -  # 0.1 - Quick Start - -This is a set of short examples with minimal explanation. It is meant as -a "quick start". Soon, we'll write a longer tour :-) - - -Add a file to ipfs: - - echo "hello world" >hello - ipfs add hello - - -View it: - - ipfs cat - - -Try a directory: - - mkdir foo - mkdir foo/bar - echo "baz" > foo/baz - echo "baz" > foo/bar/baz - ipfs add -r foo - - -View things: - - ipfs ls - ipfs ls /bar - ipfs cat /baz - ipfs cat /bar/baz - ipfs cat /bar - ipfs ls /baz - - -References: - - ipfs refs - ipfs refs -r - ipfs refs --help - - -Get: - - ipfs get foo2 - diff foo foo2 - - -Objects: - - ipfs object get - ipfs object get /foo2 - ipfs object --help - - -Pin + GC: - - ipfs pin -r - ipfs gc - ipfs ls - ipfs unpin -r - ipfs gc - - -Daemon: - - ipfs daemon (in another terminal) - ipfs id - - -Network: - - (must be online) - ipfs swarm peers - ipfs id - ipfs cat - - -Mount: - - (warning: fuse is finicky!) - ipfs mount - cd /ipfs/< - - -Tool: - - ipfs version - ipfs update - ipfs commands - ipfs config --help - open http://localhost:5001/webui - - -Browse: - - webui: - - http://localhost:5001/webui - - video: - - http://localhost:8080/ipfs/QmVc6zuAneKJzicnJpfrqCH9gSy6bz54JhcypfJYhGUFQu/play#/ipfs/QmTKZgRNwDNZwHtJSjCp6r5FYefzpULfy37JvMt9DwvXse - - images: - - http://localhost:8080/ipfs/QmZpc3HvfjEXvLWGQPWbHk3AjD5j8NEN4gmFN8Jmrd5g83/cs - - markdown renderer app: - - http://localhost:8080/ipfs/QmX7M9CiYXjVeFnkfVGf3y5ixTZ2ACeSGyL1vBJY1HvQPp/mdown - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/X3/CIQFTFEEHEDF6KLBT32BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data b/test/js-ipfs-repo/blocks/X3/CIQFTFEEHEDF6KLBT32BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data deleted file mode 100644 index 9553a942db..0000000000 --- a/test/js-ipfs-repo/blocks/X3/CIQFTFEEHEDF6KLBT32BFAGLXEZL4UWFNWM4LFTLMXQBCERZ6CMLX3Y.data +++ /dev/null @@ -1,2 +0,0 @@ - - \ No newline at end of file diff --git a/test/js-ipfs-repo/blocks/XO/CIQJGO2B2N75IUEM372FSMG76VV256I4PXBULZZ5ASNLK4FL4EG7XOI.data b/test/js-ipfs-repo/blocks/XO/CIQJGO2B2N75IUEM372FSMG76VV256I4PXBULZZ5ASNLK4FL4EG7XOI.data deleted file mode 100644 index d899663bf7a13bb0563304b1d96985a784c2d515..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 10849 zcmWgA;8GG&c)a0=(~LhdBC~#fIWaSTr;7SI$0sSXC;Kd!I#aqLXh*0Jg9L-nsLIh$ z8ciXixns0s7%de?YlP9-aI`iYtqn(O!_nGsv^E^A4M%Ik(b{mdHXN-DM{C2;+HkZs z9IXvUYs1mnaI`iYtqn(O!_nGsv^E^A4M%Ik(b{mdHXN-DM{C2;+HkZs9IXvUYs1mn zaI`iYG__&mlom_2c+JFlUxXS??=;#MbToJ7(P`PK4+MoQKg`(u9-MiqUv4K}orj%_y00+YJV}#XGok z)Gc&_%?H(zbc;FK34aUrCGW}p=j0S(a*4bcz{(GU&M5Dn1~4bcz{(GU&M5Dn1~4bcz{(GU&M5Dn1~ z4bcz{(GU&M5Dn4rZ!}yDE04r@ptViQt$NSPbgyf+T8kZmjdSx~?(hFos3IOo>d{>N G+6cZA+z2}W diff --git a/test/js-ipfs-repo/blocks/_README b/test/js-ipfs-repo/blocks/_README deleted file mode 100644 index ac3b6034c3..0000000000 --- a/test/js-ipfs-repo/blocks/_README +++ /dev/null @@ -1,22 +0,0 @@ -This is a repository of IPLD objects. Each IPLD object is in a single file, -named .data. Where is the -"base32" encoding of the CID (as specified in -https://github.com/multiformats/multibase) without the 'B' prefix. -All the object files are placed in a tree of directories, based on a -function of the CID. This is a form of sharding similar to -the objects directory in git repositories. Previously, we used -prefixes, we now use the next-to-last two charters. - func NextToLast(base32cid string) { - nextToLastLen := 2 - offset := len(base32cid) - nextToLastLen - 1 - return str[offset : offset+nextToLastLen] - } -For example, an object with a base58 CIDv1 of - zb2rhYSxw4ZjuzgCnWSt19Q94ERaeFhu9uSqRgjSdx9bsgM6f -has a base32 CIDv1 of - BAFKREIA22FLID5AJ2KU7URG47MDLROZIH6YF2KALU2PWEFPVI37YLKRSCA -and will be placed at - SC/AFKREIA22FLID5AJ2KU7URG47MDLROZIH6YF2KALU2PWEFPVI37YLKRSCA.data -with 'SC' being the last-to-next two characters and the 'B' at the -beginning of the CIDv1 string is the multibase prefix that is not -stored in the filename. diff --git a/test/js-ipfs-repo/config b/test/js-ipfs-repo/config deleted file mode 100644 index 5374e57718..0000000000 --- a/test/js-ipfs-repo/config +++ /dev/null @@ -1,34 +0,0 @@ -{ - "Addresses": { - "Swarm": [ - "/ip4/0.0.0.0/tcp/4002", - "/ip4/127.0.0.1/tcp/4003/ws" - ], - "API": "/ip4/127.0.0.1/tcp/5002", - "Gateway": "/ip4/127.0.0.1/tcp/9090" - }, - "Discovery": { - "MDNS": { - "Enabled": true, - "Interval": 10 - }, - "webRTCStar": { - "Enabled": true - } - }, - "Bootstrap": [ - "/ip4/104.131.131.82/tcp/4001/ipfs/QmaCpDMGvV2BGHeYERUEnRQAwe3N8SzbUtfsmvsqQLuvuJ", - "/ip4/104.236.176.52/tcp/4001/ipfs/QmSoLnSGccFuZQJzRadHn95W2CrSFmZuTdDWP8HXaHca9z", - "/ip4/104.236.179.241/tcp/4001/ipfs/QmSoLPppuBtQSGwKDZT2M73ULpjvfd3aZ6ha4oFGL1KrGM", - "/ip4/162.243.248.213/tcp/4001/ipfs/QmSoLueR4xBeUbY9WZ9xGUUxunbKWcrNFTDAadQJmocnWm", - "/ip4/128.199.219.111/tcp/4001/ipfs/QmSoLSafTMBsPKadTEgaXctDQVcqN88CNLHXMkTNwMKPnu", - "/ip4/104.236.76.40/tcp/4001/ipfs/QmSoLV4Bbm51jM9C4gDYZQ9Cy3U6aXMJDAbzgu2fzaDs64", - "/ip4/178.62.158.247/tcp/4001/ipfs/QmSoLer265NRgSp2LA3dPaeykiS1J6DifTC88f5uVQKNAd", - "/ip4/178.62.61.185/tcp/4001/ipfs/QmSoLMeWqB7YGVLJN3pNLQpmmEk35v6wYtsMGLzSr5QBU3", - "/ip4/104.236.151.122/tcp/4001/ipfs/QmSoLju6m7xTh3DuokvT3886QRYqxAzb1kShaanJgW36yx" - ], - "Identity": { - "PeerID": "QmWR9foWsbncr89eH7sb6H5XtJSr2DUHyg3pcJWzwCVD5i", - "PrivKey": "CAASpwkwggSjAgEAAoIBAQCVa6fmcaxCpvGBg6RCQoHAJjN3/+ZFDjPt2/hRgNnAKCoA4OfLOKtkh0iVLoFErh/JeL2pd7xzFpzaAYqNkpi8TV2Gu3TOj3dY1jnJ7Kn5jp96rDyPpbWpKLrlGxrZ1lFRQJZnI/TwK9EclTS9wg9sHJMGFX0VrmpnY6XbpToo8N4WyE0hyQwoEOZrOcPTuSNQtXh9lQKzW5MgXh6zuyn4H2FyNWrJmn+0ICduZwVze/BWtbY3zB39WP7xMLg/CKJJSjPNErDpgfSzXloCkRH2eWudfkPiJZQCXQUBaf96DazGjT53WyTsiARuhvgybusm+v+oySitnMPem90tOWCDAgMBAAECggEAJs+64fm+0+VHL+wAu5FI4J9LmSzoJCHHTU5o1srQRMBqAKZqrve+VTNqoA1fWHaeHRf2s9E9KuvBvELQOWvNroOt+nQwqALN7k5e3ltR31ezhwq8BFf3ssO4oBAFTnYD00YiixHk6omHHJSJCY0hsmRNsDPetMYEL2zLxRzK2Z0pQ8cj9O0CuDI8Dp1BL+Qj36e7NN6J8ApResXMXhaxG9XJjH05s4Jmvh3dhe0uzeqtw82k7nUG3w3BrYeLQlCxCEKo2gdBiSiGLWpzbnBhHqrpBlUbFZXUFi7bDxPoEfp5O3J1rReaTO32Ks3gipVT2AZZR0/b9PwP+4b5/5s98QKBgQDSS1bSeR3GMMf+/5payOuI1JvgsdJxdEi7fGhlXAWEzB+DLFS5rnk3FGR5CtssP1SokG/2y4zYZwmfReJElcKi8/97+cz8qaOSVibNJQgo4YFoieL6yN2T+PTERrP1wKHluY/VDVEip0f+Dy27YWFzyfkRftJauH1EJhHB03prCwKBgQC15VVcdeonjhdg43HRD6fYapIwRu3SKKW6+cuZX1z9i4pd6leF4659v0J5+56qZwlQP7tprALMEzXP9Qlp0CnmeayxKv13P0gCa2dj07ev0I7tgfyu+VsanAxdNpVjITkzMZFvA+iaPcl9exBXQyqibZ9F+ouySf/6j7ISOn4LaQKBgQDNJBdXDDViT9XZSAiQBS3W9ef1giIS8Oe+lTbVobX0fJdFA6rG04+FjowQk+mIOWNhaD0AmEIWHDWDvM1kMF+FTNZuXxl7ZBSfvRDuz/AsyQbPLqtQD0AviA1lpr8Ivdhw29qWb/40+tA84xuujeld5anlSjzzqMLyBt0NojnqTwKBgBrfQJciTG9/jHCh6WSrXW6Zf/Nl5yiPHLzcvnlwHzrN7KknP7rmAhdNtDJ1O+GMy1FxuOxJnzgq9l48oyFgwW/tNdgOs9d0H91LpMCYvKBavWLVO1FWQWFTAHic4cD9XYf0oLm36BsxWZ0SGz3pDFlmvIsIEq9KQ2ju5a3PumNRAoGAPt2oRZ+qqNI7rC+GEzYCxQC1W0ehblyTSOfg8m8YbolUwmKWVJ36KQWsnOR63A1QS2XMbs3lQRK4sMajIKbHscwR4aDvIp7C2jslFhcDfZE8tcJSwLg4fBDH8E+eNz5BWndqP0L+qWdGikOlO9PJtjZi1rmlXbEmHHTpkN5W2rE=" - } -} \ No newline at end of file diff --git a/test/js-ipfs-repo/datastore/000002.ldb b/test/js-ipfs-repo/datastore/000002.ldb deleted file mode 100644 index fc04d660e9ab26bb5d27b3b6b33d212bd1e34f0b..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 211 zcmZROP}0xIPfpCyPs_|nEiTH>FJWY200SX~$SEzBZ1I|j^S%f*oZe})FX(9Q%%jt? zQy&NlS$>$Y`#n34QbA^3F(V^P!?OjhZT>&P<~cuU%i;RLe(ysy@1_qD4Q&dq4R$uX zbPxoo11sKKRu>QF9Qd?w0|O_M0pouNU?_u2-8}VndNoG}qZmkp00{rx(5+JHej5PO CoIXSV diff --git a/test/js-ipfs-repo/datastore/000005.ldb b/test/js-ipfs-repo/datastore/000005.ldb deleted file mode 100644 index 63d9d260b936415ee42b5aeda6c79fb941664398..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1120 zcmZQDSjViNS&&z(FQg!8*qs;0^V3RMWIveo0t|Brg#j3VRkisv@!=|IgJvNC_dNx9y@%e`@arzW`upR#0^RR3Lu1P3 zgny1+s9JOHtH$|N^V2^pIaU+*`j0Vdf^UEA=Y(L*NL7==e45ubJUDbI<}5@1nZ5i@ zU-f@onB_6=BHQa-tVc_VRZiT0AST5ib<){%%1r}PkJe@8+6(<3{`CFeMP?prC*c znwwdgM`+7O$B&?~72w>77RV zf{y0SJUT5q^?{&}<%b!&-?Q^56=dcWGqS)myonCBGLo1n?_9oP$D!cvsdqIm%dJT< zUeCYn&kfrf>=lfOBoQtW6k&$7W$0~c7gOL@IORq4N$7y_CFLmFs-1D4yKkEePa;Yx&V3}9e z6J;wFzbL(7XX!B6*Rdrw=+4dxeU2>!j1QaGe>~sK(>yOETD?v4ss5$s3VXC14~0|~ z@|s=JZrGS)y@`hY)f%XfBnQv@ZgL4too7C-W6;9@lKgp&)Y9xu#YGC zq+Q(oPkf)J&zo^7bJ>~t)Xqs~*IU@HQqP#^{d0FztkiZ*y~k%OjhiRCYiY>$1@7+Z zzc{J2k%`0O*2zuI)9>zn)wWdsh9cXQc|w1;OnM=;)%wQi`6UM;Jtuj8n*Q?qk;X3q zZaf<6!(B7;zLu#kRKLAQM`r(FbFXM+CT2zkqS6H@{24)n$QSJzIOo8pg&P<+nGG2K hLjXe)2VAZvSyQK-gJTMZI7kRT2>;#Cty1cK8vstu!bkuB diff --git a/test/js-ipfs-repo/datastore/000010.ldb b/test/js-ipfs-repo/datastore/000010.ldb deleted file mode 100644 index 92077582ca4c1094d81e1c35f8703550afefa5f7..0000000000000000000000000000000000000000 GIT binary patch literal 0 HcmV?d00001 literal 1143 zcmZQDV9?Jj$Sc+tQjj$4&Wq#uX{9W(pHXd=&LNx2cdi6ocRIV%d~TcDw2tFO0@nWL9`h-bL3Yk+%*afr8@TbPk^RJchgVHA9N_5eTxD41QxO=J9Z^{56JZ); z7-4K;T3DFokyM;!W}Z>zXzZ6K)WGC&@4ao?>Nz(8|F>suXs_zIpd7w&-MpLjzTew7 zw;bnNa{uX~ml~7*<|J4d{FVA*zBjGA&O}w!=F`N7tE3H@g#_I96y)_DdRxPzt0SfDVr1iIeMXL&AqQ0=U2^7|FGm(P2B50#;ggx{k5MHf;A&mO%C&EUfb~C z(50BO4E<;J@;iOi|8-%O$GnSduXnK?Eh$zxasPps6ob@BXVWP+4NN^+mzirX^ndu{ z#t*;Sx0dNGNfK9{8ZLg&Y3{=GymywrWge8Ax zb8_8$u}O-_UoIvnSEU^1zHqkmzk}Jo{lB**@XgG7Tp+=qBxht`XsT;yq-$sxVrXh* zVrgY+u4iCoXlQ9}W@#R!!6>CfIN}*~849wANg!O1*k!T*#RNJcTULFAv{=p%R0p5|p&LRGejGWK}qNY!jl|rkT zI2eSMF&Q*5DjD#yacZ@Bw0-AgWMpJzU~XdMX8?+GF*PwVGHml$WiN9uvO;p{6~^*7 zjnC_)?mL!yo-^-fogiH<)#V;6^U8XnY{lXir8n#>9VYuaw!{YA*;%2_v890VVH5k0 z=ev2D=Y>S8w`o4rzw}&TkCx-1kjg?{vrD=>%@*e)9$i}<^>(k?*#eGjDX!_SpO^_A zd~u&uKT_JeV$DC^DKqPN`vna4@g$$Li@X1c@ALF|GfrhLJ5!(9IqB?r3;R{-856yK z?v9F;+ODbh_-v(d^JI4|4f(#n-Cg|`C$%;*ad_N1xygC@-QBO+mg?V7WV_2So6|Ky~%*a4g zWP`$=5k#;l^#s5<2R<#_z`)6D!1x~m7-n+7<=&pJcTULFAv{=p%R0p5|p&LRGe zjGPcN*cn(K5N|UhD@2zb$Y#P0Wn_a{!gQHYLO&-zIWb2+Ei)&zxF|orgpmm%D-4w= M$jmEdWQ2$T0KrCJEdT%j diff --git a/test/js-ipfs-repo/version b/test/js-ipfs-repo/version deleted file mode 100644 index 7813681f5b..0000000000 --- a/test/js-ipfs-repo/version +++ /dev/null @@ -1 +0,0 @@ -5 \ No newline at end of file From 2349f1e95f0cf7d9369686934882822ae26476f9 Mon Sep 17 00:00:00 2001 From: Yahya Date: Tue, 5 Sep 2017 12:50:32 +0200 Subject: [PATCH 12/14] Changes required for #968 feat/Gateway (#989) * feat: add gateway route to daemon * feat: adding Gateway endeavour - improve codebase * clean up, remove commented out lines. (#971) * clean up, remove commented out lines. * cleaning code, removing commented out blocks. * gateway initial tests. * clean up , working tests on node v8.4.0 License: MIT Signed-off-by: Yahya * fix using js-ipfs-repo in gateway tests. License: MIT Signed-off-by: Yahya * Using unix-fs to detect dirs, replacing object.get with DAG.get, CID checks License: MIT Signed-off-by: Yahya * rename checkHash -> checkCID License: MIT Signed-off-by: Yahya * gateway tests: init a fresh repo --- src/http/gateway/resolver.js | 30 +++++++++++++-------------- src/http/gateway/resources/gateway.js | 6 +++--- src/http/gateway/routes/gateway.js | 4 ++-- test/gateway/index.js | 23 ++++++-------------- 4 files changed, 26 insertions(+), 37 deletions(-) diff --git a/src/http/gateway/resolver.js b/src/http/gateway/resolver.js index bc9ec1d5a0..f2dc4a07be 100644 --- a/src/http/gateway/resolver.js +++ b/src/http/gateway/resolver.js @@ -3,10 +3,11 @@ const mh = require('multihashes') const promisify = require('promisify-es6') const eachOfSeries = require('async/eachOfSeries') +const CID = require('cids') +const Unixfs = require('ipfs-unixfs') const debug = require('debug') const log = debug('jsipfs:http-gateway:resolver') log.error = debug('jsipfs:http-gateway:resolver:error') - const html = require('./utils/html') const PathUtil = require('./utils/path') @@ -41,32 +42,31 @@ const resolveMultihash = promisify((ipfs, path, callback) => { const partsLength = parts.length let currentMultihash = parts[0] - + let currentCid eachOfSeries(parts, (multihash, currentIndex, next) => { - // throws error when invalid multihash is passed - mh.validate(mh.fromB58String(currentMultihash)) + // throws error when invalid CID is passed + try { + currentCid = new CID(mh.fromB58String(currentMultihash)) + } catch (e) { + if (e) throw e + } + log('currentMultihash: ', currentMultihash) log('currentIndex: ', currentIndex, '/', partsLength) - ipfs.object.get(currentMultihash, { enc: 'base58' }, (err, dagNode) => { + ipfs.dag.get(currentCid, (err, result) => { if (err) { return next(err) } + let dagNode = result.value if (currentIndex === partsLength - 1) { - // leaf node - log('leaf node: ', currentMultihash) - - // TODO: Check if it is a directory by using Unixfs Type, right now - // it won't detect empty dirs - if (dagNode.links && - dagNode.links.length > 0 && - dagNode.links[0].name.length > 0) { - // this is a directory. - + let dagDataObj = Unixfs.unmarshal(dagNode.data) + if (dagDataObj.type === 'directory') { let isDirErr = new Error('This dag node is a directory') // add currentMultihash as a fileName so it can be used by resolveDirectory isDirErr.fileName = currentMultihash return next(isDirErr) } + return next() } diff --git a/src/http/gateway/resources/gateway.js b/src/http/gateway/resources/gateway.js index 37ccba22b4..5c472d1492 100644 --- a/src/http/gateway/resources/gateway.js +++ b/src/http/gateway/resources/gateway.js @@ -12,8 +12,8 @@ const PathUtils = require('../utils/path') const Stream = require('stream') module.exports = { - checkHash: (request, reply) => { - if (!request.params.hash) { + checkCID: (request, reply) => { + if (!request.params.cid) { return reply({ Message: 'Path Resolve error: path must contain at least one component', Code: 0 @@ -21,7 +21,7 @@ module.exports = { } return reply({ - ref: `/ipfs/${request.params.hash}` + ref: `/ipfs/${request.params.cid}` }) }, handler: (request, reply) => { diff --git a/src/http/gateway/routes/gateway.js b/src/http/gateway/routes/gateway.js index e1c0f3222f..5eddeb7c21 100644 --- a/src/http/gateway/routes/gateway.js +++ b/src/http/gateway/routes/gateway.js @@ -7,10 +7,10 @@ module.exports = (server) => { gateway.route({ method: '*', - path: '/ipfs/{hash*}', + path: '/ipfs/{cid*}', config: { pre: [ - { method: resources.gateway.checkHash, assign: 'args' } + { method: resources.gateway.checkCID, assign: 'args' } ], handler: resources.gateway.handler } diff --git a/test/gateway/index.js b/test/gateway/index.js index afa9cb70d9..9a77f9903c 100644 --- a/test/gateway/index.js +++ b/test/gateway/index.js @@ -6,34 +6,23 @@ const dirtyChai = require('dirty-chai') const expect = chai.expect chai.use(dirtyChai) const API = require('../../src/http') -const ncp = require('ncp').ncp -const path = require('path') -const clean = require('../utils/clean') - -describe('HTTP GATEWAY', () => { - const repoExample = path.join(__dirname, '../go-ipfs-repo') - const repoTests = path.join(__dirname, '../repo-tests-run') +describe('HTTP Gateway', () => { let http = {} let gateway before((done) => { - http.api = new API(repoTests) - - ncp(repoExample, repoTests, (err) => { - expect(err).to.not.exist() + http.api = new API() - http.api.start(false, () => { - gateway = http.api.server.select('Gateway') - done() - }) + http.api.start(true, () => { + gateway = http.api.server.select('Gateway') + done() }) }) after((done) => { http.api.stop((err) => { expect(err).to.not.exist() - clean(repoTests) done() }) }) @@ -67,7 +56,7 @@ describe('HTTP GATEWAY', () => { url: '/ipfs/QmT78zSuBmuS4z925WZfrqQ1qHaJ56DQaTfyMUF7F8ff5o' }, (res) => { expect(res.statusCode).to.equal(200) - expect(res.rawPayload).to.deep.equal(new Buffer('hello world' + '\n')) + expect(res.rawPayload).to.deep.equal(Buffer.from('hello world' + '\n')) expect(res.payload).to.equal('hello world' + '\n') done() }) From d3f34f5b28474338169513989be2ce0866e3b354 Mon Sep 17 00:00:00 2001 From: Yahya Date: Tue, 5 Sep 2017 23:22:11 +0200 Subject: [PATCH 13/14] test: serve big file via gateway test License: MIT Signed-off-by: Yahya --- test/gateway/index.js | 22 +++++++++++++++++++++- 1 file changed, 21 insertions(+), 1 deletion(-) diff --git a/test/gateway/index.js b/test/gateway/index.js index 9a77f9903c..12ab020c34 100644 --- a/test/gateway/index.js +++ b/test/gateway/index.js @@ -6,17 +6,26 @@ const dirtyChai = require('dirty-chai') const expect = chai.expect chai.use(dirtyChai) const API = require('../../src/http') +const loadFixture = require('aegir/fixtures') +const bigFile = loadFixture(__dirname, '../../node_modules/interface-ipfs-core/test/fixtures/15mb.random', 'ipfs') describe('HTTP Gateway', () => { let http = {} let gateway + let bigFileHash = 'Qme79tX2bViL26vNjPsF3DP1R9rMKMvnPYJiKTTKPrXJjq' before((done) => { http.api = new API() http.api.start(true, () => { gateway = http.api.server.select('Gateway') - done() + http.api.node.files.add(bigFile, (err, files) => { + if (err) throw err + expect(files).to.exist() + expect(files[0].hash).to.deep.equal(bigFileHash) + + done() + }) }) }) @@ -61,5 +70,16 @@ describe('HTTP Gateway', () => { done() }) }) + + it('stream a large file', (done) => { + gateway.inject({ + method: 'GET', + url: '/ipfs/' + bigFileHash + }, (res) => { + expect(res.statusCode).to.equal(200) + expect(res.rawPayload).to.deep.equal(bigFile) + done() + }) + }) }) }) From 0bd6c6778cae23f1a02a8e29bf8c0aa3c75788b9 Mon Sep 17 00:00:00 2001 From: Yahya Date: Wed, 6 Sep 2017 15:55:05 +0200 Subject: [PATCH 14/14] initial gateway tests WIP License: MIT Signed-off-by: Yahya --- test/gateway/index.js | 127 +++++++++++++++++- test/gateway/interface/block.js | 20 +++ test/gateway/interface/config.js | 20 +++ test/gateway/interface/files.js | 20 +++ test/gateway/interface/object.js | 20 +++ test/gateway/interface/pubsub.js | 23 ++++ test/gateway/interface/swarm.js | 20 +++ test/gateway/test-folder/index.html | 10 ++ .../test-folder/nested-folder/hello.txt | 1 + .../test-folder/nested-folder/ipfs.txt | 1 + .../test-folder/nested-folder/nested.html | 10 ++ 11 files changed, 265 insertions(+), 7 deletions(-) create mode 100644 test/gateway/interface/block.js create mode 100644 test/gateway/interface/config.js create mode 100644 test/gateway/interface/files.js create mode 100644 test/gateway/interface/object.js create mode 100644 test/gateway/interface/pubsub.js create mode 100644 test/gateway/interface/swarm.js create mode 100644 test/gateway/test-folder/index.html create mode 100644 test/gateway/test-folder/nested-folder/hello.txt create mode 100644 test/gateway/test-folder/nested-folder/ipfs.txt create mode 100644 test/gateway/test-folder/nested-folder/nested.html diff --git a/test/gateway/index.js b/test/gateway/index.js index 12ab020c34..13759a1205 100644 --- a/test/gateway/index.js +++ b/test/gateway/index.js @@ -1,6 +1,8 @@ /* eslint-env mocha */ 'use strict' +const fs = require('fs') +const path = require('path') const chai = require('chai') const dirtyChai = require('dirty-chai') const expect = chai.expect @@ -8,22 +10,48 @@ chai.use(dirtyChai) const API = require('../../src/http') const loadFixture = require('aegir/fixtures') const bigFile = loadFixture(__dirname, '../../node_modules/interface-ipfs-core/test/fixtures/15mb.random', 'ipfs') +const directoryContent = { + 'index.html': loadFixture(__dirname, './test-folder/index.html', 'ipfs'), + 'nested-folder/hello.txt': loadFixture(__dirname, './test-folder/nested-folder/hello.txt', 'ipfs'), + 'nested-folder/ipfs.txt': loadFixture(__dirname, './test-folder/nested-folder/ipfs.txt', 'ipfs'), + 'nested-folder/nested.html': loadFixture(__dirname, './test-folder/nested-folder/nested.html', 'ipfs') +} describe('HTTP Gateway', () => { let http = {} let gateway - let bigFileHash = 'Qme79tX2bViL26vNjPsF3DP1R9rMKMvnPYJiKTTKPrXJjq' before((done) => { http.api = new API() http.api.start(true, () => { - gateway = http.api.server.select('Gateway') - http.api.node.files.add(bigFile, (err, files) => { - if (err) throw err - expect(files).to.exist() - expect(files[0].hash).to.deep.equal(bigFileHash) + const content = (name) => ({ + path: `test-folder/${name}`, + content: directoryContent[name] + }) + + const emptyDir = (name) => ({ + path: `test-folder/${name}` + }) + + const expectedRootMultihash = 'QmbQD7EMEL1zeebwBsWEfA3ndgSS6F7S6iTuwuqasPgVRi' + const dirs = [ + content('index.html'), + emptyDir('empty-folder'), + content('nested-folder/hello.txt'), + content('nested-folder/ipfs.txt'), + content('nested-folder/nested.html'), + emptyDir('nested-folder/empty') + ] + + http.api.node.files.add(dirs, (err, res) => { + expect(err).to.not.exist() + const root = res[res.length - 1] + + expect(root.path).to.equal('test-folder') + expect(root.hash).to.equal(expectedRootMultihash) + gateway = http.api.server.select('Gateway') done() }) }) @@ -36,7 +64,12 @@ describe('HTTP Gateway', () => { }) }) - describe('/ipfs/* route', () => { + describe('## interface tests', () => { + fs.readdirSync(path.join(__dirname, '/interface')) + .forEach((file) => require('./interface/' + file)) + }) + + describe('## HTTP Gateway', () => { it('returns 400 for request without argument', (done) => { gateway.inject({ method: 'GET', @@ -72,6 +105,8 @@ describe('HTTP Gateway', () => { }) it('stream a large file', (done) => { + let bigFileHash = 'Qme79tX2bViL26vNjPsF3DP1R9rMKMvnPYJiKTTKPrXJjq' + gateway.inject({ method: 'GET', url: '/ipfs/' + bigFileHash @@ -81,5 +116,83 @@ describe('HTTP Gateway', () => { done() }) }) + + it('load a non text file', (done) => { + let kitty = 'QmW2WQi7j6c7UgJTarActp7tDNikE4B2qXtFCfLPdsgaTQ/cat.jpg' + + gateway.inject({ + method: 'GET', + url: '/ipfs/' + kitty + }, (res) => { + expect(res.statusCode).to.equal(200) + expect(res.headers['content-type']).to.equal('image/jpeg') + done() + }) + }) + + it('load a directory', (done) => { + let dir = 'QmW2WQi7j6c7UgJTarActp7tDNikE4B2qXtFCfLPdsgaTQ/' + + gateway.inject({ + method: 'GET', + url: '/ipfs/' + dir + }, (res) => { + expect(res.statusCode).to.equal(200) + expect(res.headers['content-type']).to.equal('text/html; charset=utf-8') + done() + }) + }) + + it('load a webpage index.html', (done) => { + let dir = 'QmbQD7EMEL1zeebwBsWEfA3ndgSS6F7S6iTuwuqasPgVRi/index.html' + + gateway.inject({ + method: 'GET', + url: '/ipfs/' + dir + }, (res) => { + expect(res.statusCode).to.equal(200) + expect(res.rawPayload).to.deep.equal(directoryContent['index.html']) + done() + }) + }) + + it('load a webpage {hash}/nested-folder/nested.html', (done) => { + let dir = 'QmbQD7EMEL1zeebwBsWEfA3ndgSS6F7S6iTuwuqasPgVRi/nested-folder/nested.html' + + gateway.inject({ + method: 'GET', + url: '/ipfs/' + dir + }, (res) => { + expect(res.statusCode).to.equal(200) + expect(res.rawPayload).to.deep.equal(directoryContent['nested-folder/nested.html']) + done() + }) + }) + + it('redirect to generated index', (done) => { + let dir = 'QmW2WQi7j6c7UgJTarActp7tDNikE4B2qXtFCfLPdsgaTQ' + + gateway.inject({ + method: 'GET', + url: '/ipfs/' + dir + }, (res) => { + expect(res.statusCode).to.equal(301) + expect(res.headers['location']).to.equal('/ipfs/QmW2WQi7j6c7UgJTarActp7tDNikE4B2qXtFCfLPdsgaTQ/') + done() + }) + }) + + it('redirect to webpage index.html', (done) => { + let dir = 'QmbQD7EMEL1zeebwBsWEfA3ndgSS6F7S6iTuwuqasPgVRi/' + + gateway.inject({ + method: 'GET', + url: '/ipfs/' + dir + }, (res) => { + expect(res.statusCode).to.equal(302) + expect(res.headers['location']).to.equal('/ipfs/QmbQD7EMEL1zeebwBsWEfA3ndgSS6F7S6iTuwuqasPgVRi/index.html') + done() + }) + }) }) }) diff --git a/test/gateway/interface/block.js b/test/gateway/interface/block.js new file mode 100644 index 0000000000..db39d378c6 --- /dev/null +++ b/test/gateway/interface/block.js @@ -0,0 +1,20 @@ +/* eslint-env mocha */ + +'use strict' + +const test = require('interface-ipfs-core') +const FactoryClient = require('./../../utils/ipfs-factory-daemon') + +let fc + +const common = { + setup: function (callback) { + fc = new FactoryClient() + callback(null, fc) + }, + teardown: function (callback) { + fc.dismantle(callback) + } +} + +test.block(common) diff --git a/test/gateway/interface/config.js b/test/gateway/interface/config.js new file mode 100644 index 0000000000..30837f8b86 --- /dev/null +++ b/test/gateway/interface/config.js @@ -0,0 +1,20 @@ +/* eslint-env mocha */ + +'use strict' + +const test = require('interface-ipfs-core') +const FactoryClient = require('./../../utils/ipfs-factory-daemon') + +let fc + +const common = { + setup: function (callback) { + fc = new FactoryClient() + callback(null, fc) + }, + teardown: function (callback) { + fc.dismantle(callback) + } +} + +test.config(common) diff --git a/test/gateway/interface/files.js b/test/gateway/interface/files.js new file mode 100644 index 0000000000..84b150f79f --- /dev/null +++ b/test/gateway/interface/files.js @@ -0,0 +1,20 @@ +/* eslint-env mocha */ + +'use strict' + +const test = require('interface-ipfs-core') +const FactoryClient = require('./../../utils/ipfs-factory-daemon') + +let fc + +const common = { + setup: function (callback) { + fc = new FactoryClient() + callback(null, fc) + }, + teardown: function (callback) { + fc.dismantle(callback) + } +} + +test.files(common) diff --git a/test/gateway/interface/object.js b/test/gateway/interface/object.js new file mode 100644 index 0000000000..e00df72564 --- /dev/null +++ b/test/gateway/interface/object.js @@ -0,0 +1,20 @@ +/* eslint-env mocha */ + +'use strict' + +const test = require('interface-ipfs-core') +const FactoryClient = require('./../../utils/ipfs-factory-daemon') + +let fc + +const common = { + setup: function (callback) { + fc = new FactoryClient() + callback(null, fc) + }, + teardown: function (callback) { + fc.dismantle(callback) + } +} + +test.object(common) diff --git a/test/gateway/interface/pubsub.js b/test/gateway/interface/pubsub.js new file mode 100644 index 0000000000..968227c83b --- /dev/null +++ b/test/gateway/interface/pubsub.js @@ -0,0 +1,23 @@ +/* eslint-env mocha */ + +'use strict' + +// TODO needs: https://github.com/ipfs/js-ipfs-api/pull/493 +/* +const test = require('interface-ipfs-core') +const FactoryClient = require('./../../utils/ipfs-factory-daemon') + +let fc + +const common = { + setup: function (callback) { + fc = new FactoryClient() + callback(null, fc) + }, + teardown: function (callback) { + fc.dismantle(callback) + } +} + +test.pubsub(common) +*/ diff --git a/test/gateway/interface/swarm.js b/test/gateway/interface/swarm.js new file mode 100644 index 0000000000..eeca079124 --- /dev/null +++ b/test/gateway/interface/swarm.js @@ -0,0 +1,20 @@ +/* eslint-env mocha */ + +'use strict' + +const test = require('interface-ipfs-core') +const FactoryClient = require('./../../utils/ipfs-factory-daemon') + +let fc + +const common = { + setup: function (callback) { + fc = new FactoryClient() + callback(null, fc) + }, + teardown: function (callback) { + fc.dismantle(callback) + } +} + +test.swarm(common) diff --git a/test/gateway/test-folder/index.html b/test/gateway/test-folder/index.html new file mode 100644 index 0000000000..a83cca1dfe --- /dev/null +++ b/test/gateway/test-folder/index.html @@ -0,0 +1,10 @@ + + + + + IPFS test index.html + + + index.html + + diff --git a/test/gateway/test-folder/nested-folder/hello.txt b/test/gateway/test-folder/nested-folder/hello.txt new file mode 100644 index 0000000000..e965047ad7 --- /dev/null +++ b/test/gateway/test-folder/nested-folder/hello.txt @@ -0,0 +1 @@ +Hello diff --git a/test/gateway/test-folder/nested-folder/ipfs.txt b/test/gateway/test-folder/nested-folder/ipfs.txt new file mode 100644 index 0000000000..95a3116521 --- /dev/null +++ b/test/gateway/test-folder/nested-folder/ipfs.txt @@ -0,0 +1 @@ +IPFS diff --git a/test/gateway/test-folder/nested-folder/nested.html b/test/gateway/test-folder/nested-folder/nested.html new file mode 100644 index 0000000000..6f81153fcd --- /dev/null +++ b/test/gateway/test-folder/nested-folder/nested.html @@ -0,0 +1,10 @@ + + + + + IPFS test nested.html + + + nested.html + +