mirror of
https://github.com/openstf/stf
synced 2025-10-03 09:49:17 +02:00
APKs can now be drag & dropped to the device screen, which will install the application. Still lacking UI, and the app doesn't get launched.
This commit is contained in:
parent
5771507fc9
commit
dfe3d97de3
12 changed files with 280 additions and 8 deletions
|
@ -21,7 +21,8 @@
|
||||||
"requirejs": "~2.1.11",
|
"requirejs": "~2.1.11",
|
||||||
"stf-graphics": "git@ghe.amb.ca.local:stf/stf-graphics.git",
|
"stf-graphics": "git@ghe.amb.ca.local:stf/stf-graphics.git",
|
||||||
"angular-bootstrap": "~0.10.0",
|
"angular-bootstrap": "~0.10.0",
|
||||||
"angular-dialog-service": "~3.1.0"
|
"angular-dialog-service": "~3.1.0",
|
||||||
|
"ng-file-upload": "~1.2.9"
|
||||||
},
|
},
|
||||||
"private": true,
|
"private": true,
|
||||||
"resolutions": {
|
"resolutions": {
|
||||||
|
|
37
lib/cli.js
37
lib/cli.js
|
@ -374,6 +374,9 @@ program
|
||||||
.option('-a, --auth-url <url>'
|
.option('-a, --auth-url <url>'
|
||||||
, 'URL to auth client'
|
, 'URL to auth client'
|
||||||
, String)
|
, String)
|
||||||
|
.option('-r, --storage-url <url>'
|
||||||
|
, 'URL to storage client'
|
||||||
|
, String)
|
||||||
.option('-u, --connect-sub <endpoint>'
|
.option('-u, --connect-sub <endpoint>'
|
||||||
, 'sub endpoint'
|
, 'sub endpoint'
|
||||||
, cliutil.list)
|
, cliutil.list)
|
||||||
|
@ -389,6 +392,9 @@ program
|
||||||
if (!options.authUrl) {
|
if (!options.authUrl) {
|
||||||
this.missingArgument('--auth-url')
|
this.missingArgument('--auth-url')
|
||||||
}
|
}
|
||||||
|
if (!options.storageUrl) {
|
||||||
|
this.missingArgument('--storage-url')
|
||||||
|
}
|
||||||
if (!options.connectSub) {
|
if (!options.connectSub) {
|
||||||
this.missingArgument('--connect-sub')
|
this.missingArgument('--connect-sub')
|
||||||
}
|
}
|
||||||
|
@ -401,6 +407,7 @@ program
|
||||||
, secret: options.secret
|
, secret: options.secret
|
||||||
, ssid: options.ssid
|
, ssid: options.ssid
|
||||||
, authUrl: options.authUrl
|
, authUrl: options.authUrl
|
||||||
|
, storageUrl: options.storageUrl
|
||||||
, groupTimeout: 600 * 1000
|
, groupTimeout: 600 * 1000
|
||||||
, endpoints: {
|
, endpoints: {
|
||||||
sub: options.connectSub
|
sub: options.connectSub
|
||||||
|
@ -410,6 +417,24 @@ program
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
program
|
||||||
|
.command('storage-temp')
|
||||||
|
.description('start temp storage')
|
||||||
|
.option('-p, --port <port>'
|
||||||
|
, 'port (or $PORT)'
|
||||||
|
, Number
|
||||||
|
, process.env.PORT || 7100)
|
||||||
|
.option('--public-ip <ip>'
|
||||||
|
, 'public ip for global access'
|
||||||
|
, String
|
||||||
|
, ip())
|
||||||
|
.action(function(options) {
|
||||||
|
require('./roles/storage/temp')({
|
||||||
|
port: options.port
|
||||||
|
, publicIp: options.publicIp
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
program
|
program
|
||||||
.command('local [serial..]')
|
.command('local [serial..]')
|
||||||
.description('start everything locally')
|
.description('start everything locally')
|
||||||
|
@ -449,6 +474,10 @@ program
|
||||||
, 'app port'
|
, 'app port'
|
||||||
, Number
|
, Number
|
||||||
, 7100)
|
, 7100)
|
||||||
|
.option('--storage-port <port>'
|
||||||
|
, 'storage port'
|
||||||
|
, Number
|
||||||
|
, 7102)
|
||||||
.option('--provider <name>'
|
.option('--provider <name>'
|
||||||
, 'provider name (or os.hostname())'
|
, 'provider name (or os.hostname())'
|
||||||
, String
|
, String
|
||||||
|
@ -518,6 +547,8 @@ program
|
||||||
, '--port', options.appPort
|
, '--port', options.appPort
|
||||||
, '--secret', options.authSecret
|
, '--secret', options.authSecret
|
||||||
, '--auth-url', util.format('http://localhost:%d/', options.authPort)
|
, '--auth-url', util.format('http://localhost:%d/', options.authPort)
|
||||||
|
, '--storage-url'
|
||||||
|
, util.format('http://localhost:%d/', options.storagePort)
|
||||||
, '--connect-sub', options.bindAppPub
|
, '--connect-sub', options.bindAppPub
|
||||||
, '--connect-push', options.bindAppPull
|
, '--connect-push', options.bindAppPull
|
||||||
].concat((function() {
|
].concat((function() {
|
||||||
|
@ -527,6 +558,12 @@ program
|
||||||
}
|
}
|
||||||
return extra
|
return extra
|
||||||
})()))
|
})()))
|
||||||
|
|
||||||
|
// storage
|
||||||
|
, procutil.fork(__filename, [
|
||||||
|
'storage-temp'
|
||||||
|
, '--port', options.storagePort
|
||||||
|
])
|
||||||
]
|
]
|
||||||
|
|
||||||
function shutdown() {
|
function shutdown() {
|
||||||
|
|
|
@ -7,6 +7,7 @@ var validator = require('express-validator')
|
||||||
var socketio = require('socket.io')
|
var socketio = require('socket.io')
|
||||||
var zmq = require('zmq')
|
var zmq = require('zmq')
|
||||||
var Promise = require('bluebird')
|
var Promise = require('bluebird')
|
||||||
|
var httpProxy = require('http-proxy')
|
||||||
var _ = require('lodash')
|
var _ = require('lodash')
|
||||||
|
|
||||||
var logger = require('../util/logger')
|
var logger = require('../util/logger')
|
||||||
|
@ -20,14 +21,17 @@ var datautil = require('../util/datautil')
|
||||||
var auth = require('../middleware/auth')
|
var auth = require('../middleware/auth')
|
||||||
var webpack = require('../middleware/webpack')
|
var webpack = require('../middleware/webpack')
|
||||||
|
|
||||||
var cors = require('cors')
|
|
||||||
|
|
||||||
module.exports = function(options) {
|
module.exports = function(options) {
|
||||||
var log = logger.createLogger('app')
|
var log = logger.createLogger('app')
|
||||||
, app = express()
|
, app = express()
|
||||||
, server = http.createServer(app)
|
, server = http.createServer(app)
|
||||||
, io = socketio.listen(server)
|
, io = socketio.listen(server)
|
||||||
, channelRouter = new events.EventEmitter()
|
, channelRouter = new events.EventEmitter()
|
||||||
|
, proxy = httpProxy.createProxyServer()
|
||||||
|
|
||||||
|
proxy.on('error', function(err) {
|
||||||
|
log.error('Proxy had an error', err.stack)
|
||||||
|
})
|
||||||
|
|
||||||
app.set('view engine', 'jade')
|
app.set('view engine', 'jade')
|
||||||
app.set('views', pathutil.resource('app/views'))
|
app.set('views', pathutil.resource('app/views'))
|
||||||
|
@ -38,8 +42,6 @@ module.exports = function(options) {
|
||||||
io.set('log level', 1)
|
io.set('log level', 1)
|
||||||
io.set('browser client', false)
|
io.set('browser client', false)
|
||||||
|
|
||||||
app.use(cors())
|
|
||||||
|
|
||||||
app.use('/static/bower_components',
|
app.use('/static/bower_components',
|
||||||
express.static(pathutil.resource('bower_components')))
|
express.static(pathutil.resource('bower_components')))
|
||||||
app.use('/static/data', express.static(pathutil.resource('data')))
|
app.use('/static/data', express.static(pathutil.resource('data')))
|
||||||
|
@ -181,6 +183,18 @@ module.exports = function(options) {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
app.post('/api/v1/resources', function(req, res) {
|
||||||
|
proxy.web(req, res, {
|
||||||
|
target: options.storageUrl
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
app.get('/api/v1/resources/:id', function(req, res) {
|
||||||
|
proxy.web(req, res, {
|
||||||
|
target: options.storageUrl
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
io.set('authorization', (function() {
|
io.set('authorization', (function() {
|
||||||
var parse = Promise.promisify(express.cookieParser(options.secret))
|
var parse = Promise.promisify(express.cookieParser(options.secret))
|
||||||
return function(handshake, accept) {
|
return function(handshake, accept) {
|
||||||
|
@ -387,6 +401,16 @@ module.exports = function(options) {
|
||||||
, wireutil.envelope(new wire.ShellKeepAliveMessage(data))
|
, wireutil.envelope(new wire.ShellKeepAliveMessage(data))
|
||||||
])
|
])
|
||||||
})
|
})
|
||||||
|
.on('device.install', function(channel, responseChannel, data) {
|
||||||
|
joinChannel(responseChannel)
|
||||||
|
push.send([
|
||||||
|
channel
|
||||||
|
, wireutil.transaction(
|
||||||
|
responseChannel
|
||||||
|
, new wire.InstallMessage(data)
|
||||||
|
)
|
||||||
|
])
|
||||||
|
})
|
||||||
})
|
})
|
||||||
.finally(function() {
|
.finally(function() {
|
||||||
// Clean up all listeners and subscriptions
|
// Clean up all listeners and subscriptions
|
||||||
|
|
|
@ -21,6 +21,7 @@ module.exports = function(options) {
|
||||||
.dependency(require('./device/plugins/logcat'))
|
.dependency(require('./device/plugins/logcat'))
|
||||||
.dependency(require('./device/plugins/shell'))
|
.dependency(require('./device/plugins/shell'))
|
||||||
.dependency(require('./device/plugins/touch'))
|
.dependency(require('./device/plugins/touch'))
|
||||||
|
.dependency(require('./device/plugins/install'))
|
||||||
.dependency(require('./device/plugins/owner'))
|
.dependency(require('./device/plugins/owner'))
|
||||||
.define(function(options, solo) {
|
.define(function(options, solo) {
|
||||||
if (process.send) {
|
if (process.send) {
|
||||||
|
|
46
lib/roles/device/plugins/install.js
Normal file
46
lib/roles/device/plugins/install.js
Normal file
|
@ -0,0 +1,46 @@
|
||||||
|
var stream = require('stream')
|
||||||
|
|
||||||
|
var syrup = require('syrup')
|
||||||
|
var request = require('request')
|
||||||
|
|
||||||
|
var logger = require('../../../util/logger')
|
||||||
|
var wire = require('../../../wire')
|
||||||
|
var wireutil = require('../../../wire/util')
|
||||||
|
|
||||||
|
module.exports = syrup.serial()
|
||||||
|
.dependency(require('../support/adb'))
|
||||||
|
.dependency(require('../support/router'))
|
||||||
|
.dependency(require('../support/push'))
|
||||||
|
.define(function(options, adb, router, push) {
|
||||||
|
var log = logger.createLogger('device:plugins:install')
|
||||||
|
|
||||||
|
router.on(wire.InstallMessage, function(channel, message) {
|
||||||
|
log.info('Installing "%s"', message.url)
|
||||||
|
var source = new stream.Readable().wrap(request(message.url))
|
||||||
|
var seq = 0
|
||||||
|
adb.install(options.serial, source)
|
||||||
|
.then(function() {
|
||||||
|
log.info('Installed "%s"', message.url)
|
||||||
|
push.send([
|
||||||
|
channel
|
||||||
|
, wireutil.envelope(new wire.TransactionDoneMessage(
|
||||||
|
options.serial
|
||||||
|
, seq++
|
||||||
|
, true
|
||||||
|
))
|
||||||
|
])
|
||||||
|
})
|
||||||
|
.catch(function(err) {
|
||||||
|
log.error('Installation failed', err.stack)
|
||||||
|
push.send([
|
||||||
|
channel
|
||||||
|
, wireutil.envelope(new wire.TransactionDoneMessage(
|
||||||
|
options.serial
|
||||||
|
, seq++
|
||||||
|
, false
|
||||||
|
, err.message
|
||||||
|
))
|
||||||
|
])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
71
lib/roles/storage/temp.js
Normal file
71
lib/roles/storage/temp.js
Normal file
|
@ -0,0 +1,71 @@
|
||||||
|
var http = require('http')
|
||||||
|
var util = require('util')
|
||||||
|
|
||||||
|
var express = require('express')
|
||||||
|
var formidable = require('formidable')
|
||||||
|
var Promise = require('bluebird')
|
||||||
|
|
||||||
|
var logger = require('../../util/logger')
|
||||||
|
var Storage = require('../../util/storage')
|
||||||
|
|
||||||
|
module.exports = function(options) {
|
||||||
|
var log = logger.createLogger('storage-temp')
|
||||||
|
, app = express()
|
||||||
|
, server = http.createServer(app)
|
||||||
|
, storage = new Storage()
|
||||||
|
|
||||||
|
app.set('strict routing', true)
|
||||||
|
app.set('case sensitive routing', true)
|
||||||
|
app.set('trust proxy', true)
|
||||||
|
|
||||||
|
app.use(express.json())
|
||||||
|
app.use(express.urlencoded())
|
||||||
|
|
||||||
|
storage.on('timeout', function(id) {
|
||||||
|
log.info('Cleaning up inactive resource "%s"', id)
|
||||||
|
})
|
||||||
|
|
||||||
|
app.post('/api/v1/resources', function(req, res) {
|
||||||
|
var form = Promise.promisifyAll(new formidable.IncomingForm())
|
||||||
|
form.parseAsync(req)
|
||||||
|
.spread(function(fields, files) {
|
||||||
|
if (files.file) {
|
||||||
|
var id = storage.store(files.file)
|
||||||
|
res.json(201, {
|
||||||
|
success: true
|
||||||
|
, url: util.format(
|
||||||
|
'http://%s:%s/api/v1/resources/%s'
|
||||||
|
, options.publicIp
|
||||||
|
, options.port
|
||||||
|
, id
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
res.json(400, {
|
||||||
|
success: false
|
||||||
|
})
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(function(err) {
|
||||||
|
log.error('Failed to save resource: ', err.stack)
|
||||||
|
res.json(500, {
|
||||||
|
success: false
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
app.get('/api/v1/resources/:id', function(req, res) {
|
||||||
|
var file = storage.retrieve(req.params.id)
|
||||||
|
if (file) {
|
||||||
|
res.set('Content-Type', file.type)
|
||||||
|
res.sendfile(file.path)
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
res.send(404)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
server.listen(options.port)
|
||||||
|
log.info('Listening on port %d', options.port)
|
||||||
|
}
|
62
lib/util/storage.js
Normal file
62
lib/util/storage.js
Normal file
|
@ -0,0 +1,62 @@
|
||||||
|
var events = require('events')
|
||||||
|
var util = require('util')
|
||||||
|
var fs = require('fs')
|
||||||
|
|
||||||
|
var uuid = require('node-uuid')
|
||||||
|
|
||||||
|
function Storage() {
|
||||||
|
events.EventEmitter.call(this)
|
||||||
|
this.files = Object.create(null)
|
||||||
|
this.timer = setInterval(this.check.bind(this), 60000)
|
||||||
|
}
|
||||||
|
|
||||||
|
util.inherits(Storage, events.EventEmitter)
|
||||||
|
|
||||||
|
Storage.prototype.store = function(file) {
|
||||||
|
var id = uuid.v4()
|
||||||
|
|
||||||
|
this.files[id] = {
|
||||||
|
timeout: 600000
|
||||||
|
, lastActivity: Date.now()
|
||||||
|
, data: file
|
||||||
|
}
|
||||||
|
|
||||||
|
return id
|
||||||
|
}
|
||||||
|
|
||||||
|
Storage.prototype.remove = function(id) {
|
||||||
|
var file = this.files[id]
|
||||||
|
if (file) {
|
||||||
|
delete this.files[id]
|
||||||
|
fs.unlink(file.data.path, function() {})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Storage.prototype.retrieve = function(id) {
|
||||||
|
var file = this.files[id]
|
||||||
|
if (file) {
|
||||||
|
file.lastActivity = Date.now()
|
||||||
|
return file.data
|
||||||
|
}
|
||||||
|
return null
|
||||||
|
}
|
||||||
|
|
||||||
|
Storage.prototype.check = function() {
|
||||||
|
var now = Date.now()
|
||||||
|
|
||||||
|
Object.keys(this.files).forEach(function(id) {
|
||||||
|
var file = this.files[id]
|
||||||
|
, inactivePeriod = now - file.lastActivity
|
||||||
|
|
||||||
|
if (inactivePeriod >= file.timeout) {
|
||||||
|
this.remove(id)
|
||||||
|
this.emit('timeout', id, file.data)
|
||||||
|
}
|
||||||
|
}, this)
|
||||||
|
}
|
||||||
|
|
||||||
|
Storage.prototype.stop = function() {
|
||||||
|
clearInterval(this.timer)
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = Storage
|
|
@ -12,6 +12,7 @@ enum MessageType {
|
||||||
DeviceRegisteredMessage = 8;
|
DeviceRegisteredMessage = 8;
|
||||||
DeviceStatusMessage = 9;
|
DeviceStatusMessage = 9;
|
||||||
GroupMessage = 10;
|
GroupMessage = 10;
|
||||||
|
InstallMessage = 30;
|
||||||
PhysicalIdentifyMessage = 29;
|
PhysicalIdentifyMessage = 29;
|
||||||
JoinGroupMessage = 11;
|
JoinGroupMessage = 11;
|
||||||
KeyDownMessage = 12;
|
KeyDownMessage = 12;
|
||||||
|
@ -289,3 +290,7 @@ message ShellCommandMessage {
|
||||||
message ShellKeepAliveMessage {
|
message ShellKeepAliveMessage {
|
||||||
required uint32 timeout = 1;
|
required uint32 timeout = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
message InstallMessage {
|
||||||
|
required string url = 1;
|
||||||
|
}
|
||||||
|
|
|
@ -2,10 +2,12 @@ require('angular')
|
||||||
require('angular-route')
|
require('angular-route')
|
||||||
|
|
||||||
require('angular-gettext')
|
require('angular-gettext')
|
||||||
|
require('ng-file-upload')
|
||||||
|
|
||||||
angular.module('app', [
|
angular.module('app', [
|
||||||
'ngRoute',
|
'ngRoute',
|
||||||
'gettext',
|
'gettext',
|
||||||
|
'angularFileUpload',
|
||||||
require('./layout').name,
|
require('./layout').name,
|
||||||
require('./device-list').name,
|
require('./device-list').name,
|
||||||
require('./device-control').name,
|
require('./device-control').name,
|
||||||
|
|
|
@ -1,4 +1,4 @@
|
||||||
module.exports = function ControlServiceFactory($rootScope, socket, TransactionService) {
|
module.exports = function ControlServiceFactory($rootScope, $upload, socket, TransactionService) {
|
||||||
var controlService = {
|
var controlService = {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -87,6 +87,28 @@ module.exports = function ControlServiceFactory($rootScope, socket, TransactionS
|
||||||
socket.emit('device.identify', channel, tx.channel)
|
socket.emit('device.identify', channel, tx.channel)
|
||||||
return tx
|
return tx
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.install = function(files) {
|
||||||
|
return $upload.upload({
|
||||||
|
url: '/api/v1/resources'
|
||||||
|
, method: 'POST'
|
||||||
|
, file: files[0]
|
||||||
|
})
|
||||||
|
.success(function(data) {
|
||||||
|
console.log('success', arguments)
|
||||||
|
var tx = TransactionService.create(devices)
|
||||||
|
socket.emit('device.install', channel, tx.channel, {
|
||||||
|
url: data.url
|
||||||
|
})
|
||||||
|
return tx
|
||||||
|
})
|
||||||
|
.error(function() {
|
||||||
|
console.log('error', arguments)
|
||||||
|
})
|
||||||
|
.progress(function() {
|
||||||
|
console.log('progress', arguments)
|
||||||
|
})
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
controlService.forOne = function(device, channel) {
|
controlService.forOne = function(device, channel) {
|
||||||
|
|
|
@ -3,7 +3,7 @@ h1 {{ device.serial }} {{ device.present ? 'present' : 'absent' }}
|
||||||
button(ng-click='showScreen = !showScreen') Show/Hide
|
button(ng-click='showScreen = !showScreen') Show/Hide
|
||||||
button(ng-click='control.identify()') Identify
|
button(ng-click='control.identify()') Identify
|
||||||
|
|
||||||
div(ng-controller='DeviceScreenCtrl')
|
div(ng-controller='DeviceScreenCtrl', ng-file-drop='control.install($files)')
|
||||||
device-screen(style='width: 400px; height: 600px; background: gray')
|
device-screen(style='width: 400px; height: 600px; background: gray')
|
||||||
|
|
||||||
button(ng-click='control.menu()') Menu
|
button(ng-click='control.menu()') Menu
|
||||||
|
|
|
@ -20,7 +20,8 @@ module.exports = {
|
||||||
'angular-bootstrap': 'angular-bootstrap/ui-bootstrap-tpls',
|
'angular-bootstrap': 'angular-bootstrap/ui-bootstrap-tpls',
|
||||||
'localforage': 'localforage/dist/localforage.js',
|
'localforage': 'localforage/dist/localforage.js',
|
||||||
'socket.io': 'socket.io-client/dist/socket.io',
|
'socket.io': 'socket.io-client/dist/socket.io',
|
||||||
'oboe': 'oboe/dist/oboe-browser'
|
'oboe': 'oboe/dist/oboe-browser',
|
||||||
|
'ng-file-upload': 'ng-file-upload/angular-file-upload'
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
module: {
|
module: {
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue