Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

Add ipfs view & ipfs propagate #406

Merged
merged 4 commits into from
Mar 23, 2019
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
148 changes: 138 additions & 10 deletions Specification.md
Original file line number Diff line number Diff line change
@@ -1,17 +1,145 @@
## Gas limit
# Specification

By default `web3.js` uses `web3.eth.Contract.method.myMethod().estimateGas()`, but this value can differ from the
actual gas that will be used, if the contract you are calling depends on the blockhash, blocknumber or any other source of
randomness making the gas cost nondeterministic.
The CLI should package several "core" extensions.
The API these extensions expose is aimed at `node` environments.
The aim is to provide convenience to power-users & devs.

E.g.: the [`MiniMeToken`](https://github.com/aragon/aragon-apps/blob/master/shared/minime/contracts/MiniMeToken.sol) contract which snapshots balances at certain block numbers.
## IPFS `@aragon/ipfs-utils`
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

For extensions related to the cli, perhaps it would make sense to prefix them with cli (so this would become @aragon/cli-ipfs-utils)?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Hmm, makes sense.
What about dao, apm, etc? I think we can publish those without the utils suffix, e.g.: @aragon/cli-dao.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I like @aragon/cli-<package> format


Considering the above-mentioned behavior, the CLI should calculate the recommended gas limit as follows:
- `recommendedGas = estimatedGas`, if `estimatedGas > upperGasLimit` or
Goals:

- installing & starting a daemon with the right configuration
- interacting with local/remote nodes: authentication, configuration, pinning, etc.
- help setup a "production" node (?)
- automatically pin anything published to an APM repository (?)
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I wouldn't really consider these last two part of the flow; for production you're likely to set up your own infra and the last item is very abusable (we actually used to do this but have since moved away to curating the pinning set).


### IPFS Binaries

Local node:

- ✔️ `aragon ipfs` - Alias for `aragon ipfs start`
Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I've added a bunch of potential changes here after brainstorming a bit about the ipfs and devchain commands. Would love some feedback! cc: @0xGabi @sohkai @izqui

I think people usually "optimize" their process and run ipfs and devchain in a separate terminal, so that aragon run and other commands are faster. This is why I think we should not stop IPFS/Devchain as part of these commands, but instead expose commands to do so.

Example

IPFS not started, do you wanna start it? [yes/no]
Remember to do aragon ipfs stop afterwards.

And that ipfs start should finish and leave the daemon running in the background.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I agree, having management commands for the daemon would be great! It'd also solve things like the race condition in #383.

- `aragon ipfs start` - Start the **IPFS Daemon**
- Should start in the background and then finish
- Should start with the recommended configuration from `$HOMEDIR/.aragon/ipfsconfig.json`. E.g.:

```json
{
"daemonArgs": [
"--migrate",
"--enable-namesys-pubsub"
],
"logsLocation": "$HOMEDIR/.aragon/ipfs-logs"
}
```

- Should save `stdout` and `stderr` to files, e.g.: `stdout-${number}.log` in the `logsLocation`
- Should warn if it's already running
- Should error (exit code 1) if it cannot start (missing libs/ports taken)
- Should inform about where the logs are saved and how to stop it
- Should run `aragon ipfs status` and finish afterwards (optional)
- `aragon ipfs stop` - Stop the **IPFS Daemon**
- Should warn if the node was already stopped
- `aragon ipfs enable-startup` - Start the **IPFS Daemon** automatically at start-up
Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

May also make sense to add the ability to clear the IPFS db. Is the plan for us to use our own DB (rather than one the user may already have from a previous IPFS installation)?

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

May also make sense to add the ability to clear the IPFS db.

👍

Is the plan for us to use our own DB (rather than one the user may already have from a previous IPFS installation)?

Hmm, we could totally do this, but also allow it to be the default directory. I wonder what should be the default.. 🤔


Local/remote node:

- `aragon ipfs status` - Check the configuration of the **IPFS Daemon**
- Should print if CORS is enabled
- Should print which Aragon artifacts are pinned
- Should print number of peers, repository size
- Should print which ports are used
- Should print local ip and public ip
- Should print bootstrapped nodes
- If it's a local node:
- Should print whether 'Run at startup' is enabled
- If it's not configured correctly:
- Should ask whether to run `aragon ipfs configure`
- or Should error (exit code 1)
- `aragon ipfs configure` - Configure the **IPFS Daemon**
- Should configure CORS
- Should pin Aragon artifacts (from ipfs with a fallback to http)
- Should inform the user about advanced configurations with `ipfs config`
- ✔️ `aragon ipfs view <cid>` - Display metadata about the content, such as size, links, etc.
- ✔️ `aragon ipfs propagate <cid>` - Request the content and its links at several gateways, making the files
more distributed within the network

### IPFS API

Connection:

- ✔️ `ensureConnection`
- throws if it cannot be established
- 🚧 auth headers support
- `start`

Installation:

- `isInstalled`
- `install`

Configuration:

- `hasCORSEnabled`
- `enableCORS`
- `hasAragonArtifacts`
- `listAragonArtifacts`
- `pinAragonArtifacts`
- `hasAuthEnabled`
- `enableAuth`

Data:

- ✔️ `getMerkleDAG`
- ✔️ `extractCIDsFromMerkleDAG`
- ✔️ `propagateFiles`

Data viz:

- ✔️ `stringifyMerkleDAG`

## Devchain `@aragon/devchain-utils`

- Pre-bundled because (is tiny?)

### Devchain Binaries

- `aragon devchain` - start ganache with our aragen-generated db

### Devchain API

- `ensureConnection`
- `hasAragonDeployements`
- `deployAragon`
- `start` -- should save stdout and stderr in some files to be outputed by `aragon devchain output`

## Web3 `@aragon/web3-utils`

Note: perhaps this is better suited for `aragonAPI`.

### Web3 API

- `getRecommendedGasLimit`

By default `web3.js` uses `web3.eth.Contract.method.myMethod().estimateGas()`, but this value can
differ from the actual gas that will be used, if the contract you are calling depends on the
blockhash, blocknumber or any other source of randomness making the gas cost nondeterministic.

E.g.: the [`MiniMeToken`](https://github.com/aragon/aragon-apps/blob/master/shared/minime/contracts/MiniMeToken.sol)
contract which snapshots balances at certain block numbers.

Considering the above-mentioned behavior, the CLI should calculate the recommended gas limit
as follows:

- `recommendedGas = estimatedGas` (if `estimatedGas > upperGasLimit`) or
- `recommendedGas = estimatedGas * DEFAULT_GAS_FUZZ_FACTOR` with a maximum value of `upperGasLimit`

Where `upperGasLimit = latestBlock.gasLimit * LAST_BLOCK_GAS_LIMIT_FACTOR`, `DEFAULT_GAS_FUZZ_FACTOR = 1.5` and `LAST_BLOCK_GAS_LIMIT_FACTOR = 0.95`.
Where:

- `upperGasLimit = latestBlock.gasLimit * LAST_BLOCK_GAS_LIMIT_FACTOR`
- `LAST_BLOCK_GAS_LIMIT_FACTOR = 0.95`
- `DEFAULT_GAS_FUZZ_FACTOR = 1.5`

See `src/util.js#getRecommendedGasLimit`. (This should probably be its own library)
See `src/util.js#getRecommendedGasLimit`.
(This should probably be its own library since `aragon.js` uses it as well)

The CLI should ignore the `gas` property of the network from truffle config.
Note: The CLI should ignore the `gas` property of the network from truffle config.
4 changes: 4 additions & 0 deletions packages/aragon-cli/package.json
Original file line number Diff line number Diff line change
Expand Up @@ -46,6 +46,7 @@
"@babel/polyfill": "^7.0.0",
"ajv": "^6.6.2",
"bignumber.js": "^7.1.0",
"byte-size": "^5.0.1",
"chalk": "^2.1.0",
"cli-table": "^0.3.1",
"colors": "^1.2.4",
Expand All @@ -62,6 +63,7 @@
"go-ipfs": "0.4.17",
"ignore": "^3.3.7",
"ipfs-api": "^21.0.0",
"ipfs-http-client": "^30.1.0",
"js-sha3": "^0.7.0",
"listr": "^0.13.0",
"listr-input": "0.1.3",
Expand All @@ -70,10 +72,12 @@
"listr-verbose-renderer": "^0.4.1",
"mkdirp": "^0.5.1",
"ncp": "^2.0.0",
"node-fetch": "^2.3.0",
"opn": "^5.3.0",
"rimraf": "^2.6.2",
"semver": "^5.4.1",
"source-map-support": "^0.5.11",
"stringify-tree": "^1.0.2",
"tmp-promise": "^1.0.4",
"truffle": "4.1.14",
"truffle-flattener": "^1.2.9",
Expand Down
2 changes: 1 addition & 1 deletion packages/aragon-cli/src/commands/apm_cmds/publish.js
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ const execa = require('execa')
const { compileContracts } = require('../../helpers/truffle-runner')
const web3Utils = require('web3-utils')
const deploy = require('../deploy')
const startIPFS = require('../ipfs')
const startIPFS = require('../ipfs_cmds/start')
const getRepoTask = require('../dao_cmds/utils/getRepoTask')
const listrOpts = require('../../helpers/listr-options')

Expand Down
2 changes: 1 addition & 1 deletion packages/aragon-cli/src/commands/dao_cmds/new.js
Original file line number Diff line number Diff line change
Expand Up @@ -6,7 +6,7 @@ const chalk = require('chalk')
const { getContract } = require('../../util')
const getRepoTask = require('./utils/getRepoTask')
const listrOpts = require('../../helpers/listr-options')
const startIPFS = require('../ipfs')
const startIPFS = require('../ipfs_cmds/start')
const { getRecommendedGasLimit } = require('../../util')

exports.BARE_KIT = defaultAPMName('bare-kit')
Expand Down
92 changes: 6 additions & 86 deletions packages/aragon-cli/src/commands/ipfs.js
Original file line number Diff line number Diff line change
@@ -1,89 +1,9 @@
const path = require('path')
const TaskList = require('listr')
const {
startIPFSDaemon,
isIPFSCORS,
setIPFSCORS,
isIPFSRunning,
} = require('../helpers/ipfs-daemon')
const startCommand = require('./ipfs_cmds/start')

const IPFS = require('ipfs-api')
const listrOpts = require('../helpers/listr-options')

exports.command = 'ipfs'

exports.describe = 'Start IPFS daemon configured to work with Aragon'

exports.task = ({ apmOptions, silent, debug }) => {
return new TaskList(
[
{
title: 'Start IPFS',
task: async (ctx, task) => {
// If the dev manually set their IPFS node, skip install and running check
if (apmOptions.ipfs.rpc.default) {
const running = await isIPFSRunning(apmOptions.ipfs.rpc)
if (!running) {
task.output = 'Starting IPFS at port: ' + apmOptions.ipfs.rpc.port
await startIPFSDaemon()
ctx.started = true
await setIPFSCORS(apmOptions.ipfs.rpc)
} else {
task.output = 'IPFS is started, checking CORS config'
await setIPFSCORS(apmOptions.ipfs.rpc)
return (
'Connected to IPFS daemon at port: ' + apmOptions.ipfs.rpc.port
)
}
} else {
await isIPFSCORS(apmOptions.ipfs.rpc)
return 'Connecting to provided IPFS daemon'
}
},
},
{
title: 'Add local files',
task: ctx => {
const ipfs = IPFS('localhost', '5001', { protocol: 'http' })
const files = path.resolve(
require.resolve('@aragon/aragen'),
'../ipfs-cache'
)

return new Promise((resolve, reject) => {
ipfs.util.addFromFs(
files,
{ recursive: true, ignore: 'node_modules' },
(err, files) => {
if (err) return reject(err)
resolve(files)
}
)
})
},
},
],
listrOpts(silent, debug)
)
exports.builder = function(yargs) {
return yargs.commandDir('ipfs_cmds')
}

exports.handler = function({ reporter, apm: apmOptions }) {
const task = exports.task({ apmOptions })

task
.run()
.then(ctx => {
if (ctx.started) {
reporter.info(
'IPFS daemon is now running. Stopping this process will stop IPFS'
)
} else {
reporter.warning('Didnt start IPFS, port busy')
process.exit()
}
})
.catch(err => {
reporter.error(err)
process.exit(1)
})
}
exports.command = 'ipfs'
exports.describe = 'Shortcut for aragon ipfs start'
exports.handler = startCommand.handler
81 changes: 81 additions & 0 deletions packages/aragon-cli/src/commands/ipfs_cmds/propagate.js
Original file line number Diff line number Diff line change
@@ -0,0 +1,81 @@
import TaskList from 'listr'
//
import {
ensureConnection,
getMerkleDAG,
extractCIDsFromMerkleDAG,
propagateFiles,
} from '../../lib/ipfs'
import listrOpts from '../../helpers/listr-options'

exports.command = 'propagate <cid>'
exports.describe =
'Request the content and its links at several gateways, making the files more distributed within the network.'

exports.builder = yargs => {
return yargs.positional('cid', {
description: 'A self-describing content-addressed identifier',
})
}

exports.task = ({ apmOptions, silent, debug, cid }) => {
return new TaskList(
[
{
title: 'Connect to IPFS',
task: async ctx => {
ctx.ipfs = await ensureConnection(apmOptions.ipfs.rpc)
},
},
{
title: 'Fetch the links',
task: async ctx => {
ctx.data = await getMerkleDAG(ctx.ipfs.client, cid, {
recursive: true,
})
},
},
{
title: 'Query gateways',
task: async (ctx, task) => {
ctx.CIDs = extractCIDsFromMerkleDAG(ctx.data, {
recursive: true,
})

const logger = text => (task.output = text)
ctx.result = await propagateFiles(ctx.CIDs, logger)
},
},
],
listrOpts(silent, debug)
)
}

exports.handler = async function({
reporter,
apm: apmOptions,
cid,
debug,
silent,
}) {
const task = await exports.task({
reporter,
apmOptions,
cid,
debug,
silent,
})

const ctx = await task.run()

reporter.info(
`Queried ${ctx.CIDs.length} CIDs at ${ctx.result.gateways.length} gateways`
)
reporter.info(`Requests succeeded: ${ctx.result.succeeded}`)
reporter.info(`Requests failed: ${ctx.result.failed}`)
reporter.debug(`Gateways: ${ctx.result.gateways.join(', ')}`)
reporter.debug(
`Errors: \n${ctx.result.errors.map(JSON.stringify).join('\n')}`
)
// TODO add your own gateways
}
Loading