2015-05-05 20:32:44 +00:00
|
|
|
capitano = require('capitano')
|
2015-02-04 19:08:38 +00:00
|
|
|
_ = require('lodash-contrib')
|
2015-02-03 17:31:16 +00:00
|
|
|
os = require('os')
|
2014-12-02 15:08:22 +00:00
|
|
|
async = require('async')
|
|
|
|
path = require('path')
|
|
|
|
mkdirp = require('mkdirp')
|
2015-01-08 12:04:37 +00:00
|
|
|
resin = require('resin-sdk')
|
2015-05-06 14:58:06 +00:00
|
|
|
image = require('resin-image')
|
2015-01-21 13:50:19 +00:00
|
|
|
visuals = require('resin-cli-visuals')
|
2015-05-13 18:37:27 +00:00
|
|
|
selfupdate = require('selfupdate')
|
2015-02-03 17:31:16 +00:00
|
|
|
commandOptions = require('./command-options')
|
2015-03-11 12:49:26 +00:00
|
|
|
packageJSON = require('../../package.json')
|
2015-04-06 20:43:31 +00:00
|
|
|
elevate = require('../elevate')
|
2014-12-02 15:08:22 +00:00
|
|
|
|
2015-01-15 17:10:14 +00:00
|
|
|
exports.download =
|
2015-04-20 13:06:40 +00:00
|
|
|
signature: 'os download <name>'
|
2015-01-15 17:10:14 +00:00
|
|
|
description: 'download device OS'
|
|
|
|
help: '''
|
|
|
|
Use this command to download the device OS configured to a specific network.
|
2015-01-08 13:55:01 +00:00
|
|
|
|
2015-01-15 17:10:14 +00:00
|
|
|
Ethernet:
|
|
|
|
You can setup the device OS to use ethernet by setting the `--network` option to "ethernet".
|
2015-01-08 13:55:01 +00:00
|
|
|
|
2015-01-15 17:10:14 +00:00
|
|
|
Wifi:
|
|
|
|
You can setup the device OS to use wifi by setting the `--network` option to "wifi".
|
|
|
|
If you set "network" to "wifi", you will need to specify the `--ssid` and `--key` option as well.
|
2014-12-02 15:08:22 +00:00
|
|
|
|
2015-03-09 23:49:07 +00:00
|
|
|
Alternatively, you can omit all kind of network configuration options to configure interactively.
|
|
|
|
|
|
|
|
You have to specify an output location with the `--output` option.
|
2014-12-02 15:08:22 +00:00
|
|
|
|
2015-01-15 17:10:14 +00:00
|
|
|
Examples:
|
2015-03-03 14:14:16 +00:00
|
|
|
|
2015-04-20 13:06:40 +00:00
|
|
|
$ resin os download MyApp --output ~/MyResinOS.zip
|
|
|
|
$ resin os download MyApp --network ethernet --output ~/MyResinOS.zip
|
|
|
|
$ resin os download MyApp --network wifi --ssid MyNetwork --key secreykey123 --output ~/MyResinOS.zip
|
|
|
|
$ resin os download MyApp --network ethernet --output ~/MyResinOS.zip
|
2015-01-15 17:10:14 +00:00
|
|
|
'''
|
|
|
|
options: [
|
2015-02-03 18:52:17 +00:00
|
|
|
commandOptions.network
|
|
|
|
commandOptions.wifiSsid
|
|
|
|
commandOptions.wifiKey
|
|
|
|
|
2015-01-15 17:10:14 +00:00
|
|
|
{
|
|
|
|
signature: 'output'
|
|
|
|
parameter: 'output'
|
|
|
|
description: 'output file'
|
|
|
|
alias: 'o'
|
2015-03-09 16:38:37 +00:00
|
|
|
required: 'You need to specify an output file'
|
2015-01-15 17:10:14 +00:00
|
|
|
}
|
|
|
|
]
|
2015-01-16 12:34:59 +00:00
|
|
|
permission: 'user'
|
2015-01-15 17:10:14 +00:00
|
|
|
action: (params, options, done) ->
|
2015-04-20 13:06:40 +00:00
|
|
|
resin.models.application.get params.name, (error, application) ->
|
2015-01-15 17:10:14 +00:00
|
|
|
return done(error) if error?
|
2015-04-20 13:06:40 +00:00
|
|
|
|
|
|
|
osParams =
|
|
|
|
network: options.network
|
|
|
|
wifiSsid: options.ssid
|
|
|
|
wifiKey: options.key
|
|
|
|
appId: application.id
|
|
|
|
|
|
|
|
async.waterfall [
|
|
|
|
|
|
|
|
(callback) ->
|
|
|
|
return callback() if osParams.network?
|
|
|
|
visuals.patterns.selectNetworkParameters (error, parameters) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
_.extend(osParams, parameters)
|
|
|
|
return callback()
|
|
|
|
|
|
|
|
(callback) ->
|
|
|
|
|
|
|
|
# We need to ensure this directory exists
|
|
|
|
mkdirp(path.dirname(options.output), _.unary(callback))
|
|
|
|
|
|
|
|
(callback) ->
|
|
|
|
console.info("Destination file: #{options.output}\n")
|
|
|
|
|
|
|
|
bar = new visuals.widgets.Progress('Downloading Device OS')
|
|
|
|
spinner = new visuals.widgets.Spinner('Downloading Device OS (size unknown)')
|
|
|
|
|
|
|
|
resin.models.os.download osParams, options.output, (error) ->
|
|
|
|
spinner.stop()
|
2015-05-11 17:23:34 +00:00
|
|
|
return callback(error)
|
2015-04-20 13:06:40 +00:00
|
|
|
, (state) ->
|
|
|
|
if state?
|
|
|
|
bar.update(state)
|
|
|
|
else
|
|
|
|
spinner.start()
|
|
|
|
|
|
|
|
], (error) ->
|
|
|
|
return done(error) if error?
|
|
|
|
console.info("\nFinished downloading #{options.output}")
|
|
|
|
return done(null, options.output)
|
2015-02-03 17:31:16 +00:00
|
|
|
|
|
|
|
exports.install =
|
|
|
|
signature: 'os install <image> [device]'
|
|
|
|
description: 'write an operating system image to a device'
|
|
|
|
help: '''
|
|
|
|
Use this command to write an operating system image to a device.
|
|
|
|
|
|
|
|
Note that this command requires admin privileges.
|
|
|
|
|
|
|
|
If `device` is omitted, you will be prompted to select a device interactively.
|
|
|
|
|
|
|
|
Notice this command asks for confirmation interactively.
|
|
|
|
You can avoid this by passing the `--yes` boolean option.
|
|
|
|
|
|
|
|
You can quiet the progress bar by passing the `--quiet` boolean option.
|
|
|
|
|
|
|
|
Examples:
|
2015-03-03 14:14:16 +00:00
|
|
|
|
2015-02-03 17:31:16 +00:00
|
|
|
$ resin os install rpi.iso /dev/disk2
|
|
|
|
'''
|
|
|
|
options: [ commandOptions.yes ]
|
|
|
|
permission: 'user'
|
|
|
|
action: (params, options, done) ->
|
2015-04-06 20:57:53 +00:00
|
|
|
|
2015-02-03 17:31:16 +00:00
|
|
|
async.waterfall [
|
|
|
|
|
2015-03-11 12:49:26 +00:00
|
|
|
(callback) ->
|
2015-05-13 18:37:27 +00:00
|
|
|
selfupdate.isUpdated(packageJSON, callback)
|
2015-03-11 12:49:26 +00:00
|
|
|
|
|
|
|
(isUpdated, callback) ->
|
|
|
|
return callback() if isUpdated
|
|
|
|
|
|
|
|
console.info '''
|
|
|
|
Resin CLI is outdated.
|
|
|
|
|
|
|
|
In order to avoid device compatibility issues, this command
|
|
|
|
requires that you have the Resin CLI updated.
|
|
|
|
|
|
|
|
Updating now...
|
|
|
|
'''
|
|
|
|
|
2015-05-05 20:32:44 +00:00
|
|
|
capitano.run('update', _.unary(callback))
|
2015-03-11 12:49:26 +00:00
|
|
|
|
2015-02-03 17:31:16 +00:00
|
|
|
(callback) ->
|
|
|
|
return callback(null, params.device) if params.device?
|
2015-02-25 15:12:58 +00:00
|
|
|
|
|
|
|
# TODO: See if we can reuse the drives action somehow here
|
|
|
|
visuals.patterns.selectDrive (error, device) ->
|
|
|
|
return callback(error) if error?
|
|
|
|
|
|
|
|
if not device?
|
|
|
|
return callback(new Error('No removable devices available'))
|
|
|
|
|
|
|
|
return callback(null, device)
|
2015-02-03 17:31:16 +00:00
|
|
|
|
|
|
|
(device, callback) ->
|
|
|
|
params.device = device
|
2015-02-06 15:11:38 +00:00
|
|
|
message = "This will completely erase #{params.device}. Are you sure you want to continue?"
|
|
|
|
visuals.patterns.confirm(options.yes, message, callback)
|
2015-02-03 17:31:16 +00:00
|
|
|
|
|
|
|
(confirmed, callback) ->
|
|
|
|
return done() if not confirmed
|
2015-04-06 20:57:53 +00:00
|
|
|
bar = new visuals.widgets.Progress('Writing Device OS')
|
2015-04-17 14:11:02 +00:00
|
|
|
params.progress = _.bind(bar.update, bar)
|
2015-05-06 14:58:06 +00:00
|
|
|
image.write(params, callback)
|
2015-02-03 17:31:16 +00:00
|
|
|
|
|
|
|
], (error) ->
|
2015-03-03 16:38:45 +00:00
|
|
|
return done() if not error?
|
|
|
|
|
2015-04-06 20:43:31 +00:00
|
|
|
if elevate.shouldElevate(error) and not options.fromScript
|
2015-02-03 17:31:16 +00:00
|
|
|
|
2015-02-05 12:56:52 +00:00
|
|
|
# Need to escape every path to avoid errors
|
2015-02-03 17:31:16 +00:00
|
|
|
resinWritePath = "\"#{path.join(__dirname, '..', '..', 'bin', 'resin-write')}\""
|
2015-04-06 20:43:31 +00:00
|
|
|
elevate.run("\"#{process.argv[0]}\" #{resinWritePath} \"#{params.image}\" \"#{params.device}\"")
|
2015-02-03 17:31:16 +00:00
|
|
|
else
|
|
|
|
return done(error)
|