1
0
Fork 0
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:
Simo Kinnunen 2014-03-20 22:00:01 +09:00
parent 5771507fc9
commit dfe3d97de3
12 changed files with 280 additions and 8 deletions

View file

@ -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": {

View file

@ -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() {

View file

@ -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

View file

@ -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) {

View 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
View 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
View 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

View file

@ -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;
}

View file

@ -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,

View file

@ -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) {

View file

@ -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

View file

@ -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: {