Skip to content
This repository has been archived by the owner on Feb 12, 2024. It is now read-only.

Commit

Permalink
fix: make progress bar work again when adding files (#2386)
Browse files Browse the repository at this point in the history
Pulled out of #2379.

BREAKING CHANGE: Order of output from the CLI command `ipfs add` may change between invocations given the same files since it is not longer buffered and sorted.

Previously, js-ipfs buffered all hashes of added files and sorted them before outputting to the terminal, this might be because the `glob` module does not return stable results.  This gives a poor user experience as you see nothing then everything, compared to `go-ipfs` which prints out the hashes as they become available.  The tradeoff is the order might be slightly different between invocations, though the hashes are always the same as we sort DAGNode links before serializing so it doesn't matter what order you import them in.
  • Loading branch information
achingbrain authored and alanshaw committed Aug 27, 2019
1 parent 44045b0 commit f6dcb0f
Show file tree
Hide file tree
Showing 2 changed files with 60 additions and 41 deletions.
91 changes: 52 additions & 39 deletions src/cli/commands/add.js
Original file line number Diff line number Diff line change
@@ -1,6 +1,8 @@
'use strict'

const pull = require('pull-stream')
const pull = require('pull-stream/pull')
const through = require('pull-stream/throughs/through')
const end = require('pull-stream/sinks/on-end')
const promisify = require('promisify-es6')
const getFolderSize = promisify(require('get-folder-size'))
const byteman = require('byteman')
Expand All @@ -11,17 +13,35 @@ const { createProgressBar } = require('../utils')
const { cidToString } = require('../../utils/cid')
const globSource = require('../../utils/files/glob-source')

async function getTotalBytes (paths, cb) {
async function getTotalBytes (paths) {
const sizes = await Promise.all(paths.map(p => getFolderSize(p)))
return sizes.reduce((total, size) => total + size, 0)
}

function addPipeline (source, addStream, options) {
function addPipeline (source, addStream, options, log) {
let finalHash

return new Promise((resolve, reject) => {
pull(
source,
addStream,
pull.collect((err, added) => {
through((file) => {
const cid = finalHash = cidToString(file.hash, { base: options.cidBase })

if (options.silent || options.quieter) {
return
}

let message = cid

if (!options.quiet) {
// print the hash twice if we are piping from stdin
message = `added ${cid} ${options.file ? file.path || '' : cid}`.trim()
}

log(message)
}),
end((err) => {
if (err) {
// Tweak the error message and add more relevant infor for the CLI
if (err.code === 'ERR_DIR_NON_RECURSIVE') {
Expand All @@ -30,32 +50,10 @@ function addPipeline (source, addStream, options) {
return reject(err)
}

if (options.silent) return resolve()

if (options.quieter) {
options.print(added.pop().hash)
return resolve()
log(finalHash)
}

added
.sort((a, b) => {
if (a.path > b.path) {
return 1
}
if (a.path < b.path) {
return -1
}
return 0
})
.reverse()
.map((file) => {
const log = options.quiet ? [] : ['added']
log.push(cidToString(file.hash, { base: options.cidBase }))
if (!options.quiet && file.path.length > 0) log.push(file.path)
return log.join(' ')
})
.forEach((msg) => options.print(msg))

resolve()
})
)
Expand Down Expand Up @@ -179,25 +177,40 @@ module.exports = {
throw new Error('Error: Enabling the sharding experiment should be done on the daemon')
}

let bar
let log = argv.print

if (argv.quieter || argv.quiet || argv.silent) {
argv.progress = false
}

if (argv.progress && argv.file) {
const totalBytes = await getTotalBytes(argv.file)
bar = createProgressBar(totalBytes)

if (process.stdout.isTTY) {
// bar.interrupt uses clearLine and cursorTo methods that are only on TTYs
log = bar.interrupt.bind(bar)
}

options.progress = byteLength => {
bar.update(byteLength / totalBytes, { progress: byteman(byteLength, 2, 'MB') })
}
}

const source = argv.file
? globSource(...argv.file, { recursive: argv.recursive })
: toPull.source(process.stdin) // Pipe directly to ipfs.add

const adder = ipfs.addPullStream(options)

// No progress or piping directly to ipfs.add: no need to getTotalBytes
if (!argv.progress || !argv.file) {
return addPipeline(source, adder, argv)
}

const totalBytes = await getTotalBytes(argv.file)
const bar = createProgressBar(totalBytes)

options.progress = byteLength => {
bar.update(byteLength / totalBytes, { progress: byteman(byteLength, 2, 'MB') })
try {
await addPipeline(source, adder, argv, log)
} finally {
if (bar) {
bar.terminate()
}
}

return addPipeline(source, adder, argv)
})())
}
}
10 changes: 8 additions & 2 deletions test/cli/files.js
Original file line number Diff line number Diff line change
Expand Up @@ -170,7 +170,10 @@ describe('files', () => runOnAndOff((thing) => {

return ipfs('add -r test/fixtures/test-data/recursive-get-dir')
.then((out) => {
expect(out).to.eql(recursiveGetDirResults.join('\n') + '\n')
// glob module does not return stable matches
recursiveGetDirResults.forEach(line => { // eslint-disable-line max-nested-callbacks
expect(out).to.include(line)
})
})
})

Expand All @@ -179,7 +182,10 @@ describe('files', () => runOnAndOff((thing) => {

return ipfs('add -r test/fixtures/test-data/recursive-get-dir/')
.then((out) => {
expect(out).to.eql(recursiveGetDirResults.join('\n') + '\n')
// glob module does not return stable matches
recursiveGetDirResults.forEach(line => { // eslint-disable-line max-nested-callbacks
expect(out).to.include(line)
})
})
})

Expand Down

0 comments on commit f6dcb0f

Please sign in to comment.