mirror of
https://github.com/openstf/stf
synced 2025-10-04 18:29:17 +02:00
Screenshots are more or less working (server-side). Resize not implemented yet. Temporarily breaks APK uploads.
This commit is contained in:
parent
5be2b8f7d5
commit
48726669dc
9 changed files with 435 additions and 185 deletions
44
lib/cli.js
44
lib/cli.js
|
@ -43,6 +43,9 @@ program
|
||||||
, 'group timeout'
|
, 'group timeout'
|
||||||
, Number
|
, Number
|
||||||
, 600)
|
, 600)
|
||||||
|
.option('-r, --storage-url <url>'
|
||||||
|
, 'URL to storage client'
|
||||||
|
, String)
|
||||||
.action(function() {
|
.action(function() {
|
||||||
var serials = cliutil.allUnknownArgs(arguments)
|
var serials = cliutil.allUnknownArgs(arguments)
|
||||||
, options = cliutil.lastArg(arguments)
|
, options = cliutil.lastArg(arguments)
|
||||||
|
@ -53,6 +56,9 @@ program
|
||||||
if (!options.connectPush) {
|
if (!options.connectPush) {
|
||||||
this.missingArgument('--connect-push')
|
this.missingArgument('--connect-push')
|
||||||
}
|
}
|
||||||
|
if (!options.storageUrl) {
|
||||||
|
this.missingArgument('--storage-url')
|
||||||
|
}
|
||||||
|
|
||||||
require('./roles/provider')({
|
require('./roles/provider')({
|
||||||
name: options.name
|
name: options.name
|
||||||
|
@ -71,6 +77,7 @@ program
|
||||||
, '--ports', ports.join(',')
|
, '--ports', ports.join(',')
|
||||||
, '--public-ip', options.publicIp
|
, '--public-ip', options.publicIp
|
||||||
, '--group-timeout', options.groupTimeout
|
, '--group-timeout', options.groupTimeout
|
||||||
|
, '--storage-url', options.storageUrl
|
||||||
])
|
])
|
||||||
}
|
}
|
||||||
, endpoints: {
|
, endpoints: {
|
||||||
|
@ -107,6 +114,9 @@ program
|
||||||
, 'group timeout'
|
, 'group timeout'
|
||||||
, Number
|
, Number
|
||||||
, 600)
|
, 600)
|
||||||
|
.option('-r, --storage-url <url>'
|
||||||
|
, 'URL to storage client'
|
||||||
|
, String)
|
||||||
.action(function(serial, options) {
|
.action(function(serial, options) {
|
||||||
if (!options.connectSub) {
|
if (!options.connectSub) {
|
||||||
this.missingArgument('--connect-sub')
|
this.missingArgument('--connect-sub')
|
||||||
|
@ -120,6 +130,9 @@ program
|
||||||
if (!options.ports) {
|
if (!options.ports) {
|
||||||
this.missingArgument('--ports')
|
this.missingArgument('--ports')
|
||||||
}
|
}
|
||||||
|
if (!options.storageUrl) {
|
||||||
|
this.missingArgument('--storage-url')
|
||||||
|
}
|
||||||
|
|
||||||
require('./roles/device')({
|
require('./roles/device')({
|
||||||
serial: serial
|
serial: serial
|
||||||
|
@ -132,6 +145,7 @@ program
|
||||||
}
|
}
|
||||||
, heartbeatInterval: options.heartbeatInterval
|
, heartbeatInterval: options.heartbeatInterval
|
||||||
, groupTimeout: options.groupTimeout * 1000 // change to ms
|
, groupTimeout: options.groupTimeout * 1000 // change to ms
|
||||||
|
, storageUrl: options.storageUrl
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -426,8 +440,8 @@ program
|
||||||
})
|
})
|
||||||
|
|
||||||
program
|
program
|
||||||
.command('storage-temp')
|
.command('cache-apk')
|
||||||
.description('start temp storage')
|
.description('apk cache')
|
||||||
.option('-p, --port <port>'
|
.option('-p, --port <port>'
|
||||||
, 'port (or $PORT)'
|
, 'port (or $PORT)'
|
||||||
, Number
|
, Number
|
||||||
|
@ -463,6 +477,29 @@ 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())
|
||||||
|
.option('--save-dir <dir>'
|
||||||
|
, 'where to save files'
|
||||||
|
, String
|
||||||
|
, os.tmpdir())
|
||||||
|
.action(function(options) {
|
||||||
|
require('./roles/storage/temp')({
|
||||||
|
port: options.port
|
||||||
|
, publicIp: options.publicIp
|
||||||
|
, saveDir: options.saveDir
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
program
|
program
|
||||||
.command('migrate')
|
.command('migrate')
|
||||||
.description('migrates the database to the latest version')
|
.description('migrates the database to the latest version')
|
||||||
|
@ -593,6 +630,8 @@ program
|
||||||
, '--connect-push', options.bindDevPull
|
, '--connect-push', options.bindDevPull
|
||||||
, '--group-timeout', options.groupTimeout
|
, '--group-timeout', options.groupTimeout
|
||||||
, '--public-ip', options.publicIp
|
, '--public-ip', options.publicIp
|
||||||
|
, '--storage-url'
|
||||||
|
, util.format('http://localhost:%d/', options.storagePort)
|
||||||
].concat(cliutil.allUnknownArgs(args)))
|
].concat(cliutil.allUnknownArgs(args)))
|
||||||
|
|
||||||
// auth-mock
|
// auth-mock
|
||||||
|
@ -625,7 +664,6 @@ program
|
||||||
, procutil.fork(__filename, [
|
, procutil.fork(__filename, [
|
||||||
'storage-temp'
|
'storage-temp'
|
||||||
, '--port', options.storagePort
|
, '--port', options.storagePort
|
||||||
, '--connect-push', options.bindDevPull
|
|
||||||
])
|
])
|
||||||
]
|
]
|
||||||
|
|
||||||
|
|
|
@ -637,6 +637,16 @@ module.exports = function(options) {
|
||||||
)
|
)
|
||||||
])
|
])
|
||||||
})
|
})
|
||||||
|
.on('screen.capture', function(channel, responseChannel) {
|
||||||
|
joinChannel(responseChannel)
|
||||||
|
push.send([
|
||||||
|
channel
|
||||||
|
, wireutil.transaction(
|
||||||
|
responseChannel
|
||||||
|
, new wire.ScreenCaptureMessage()
|
||||||
|
)
|
||||||
|
])
|
||||||
|
})
|
||||||
})
|
})
|
||||||
.finally(function() {
|
.finally(function() {
|
||||||
// Clean up all listeners and subscriptions
|
// Clean up all listeners and subscriptions
|
||||||
|
|
216
lib/roles/cache/apk.js
vendored
Normal file
216
lib/roles/cache/apk.js
vendored
Normal file
|
@ -0,0 +1,216 @@
|
||||||
|
var http = require('http')
|
||||||
|
var util = require('util')
|
||||||
|
var fs = require('fs')
|
||||||
|
|
||||||
|
var express = require('express')
|
||||||
|
var validator = require('express-validator')
|
||||||
|
var Promise = require('bluebird')
|
||||||
|
var ApkReader = require('adbkit-apkreader')
|
||||||
|
var request = require('request')
|
||||||
|
var progress = require('request-progress')
|
||||||
|
var temp = require('temp')
|
||||||
|
var zmq = require('zmq')
|
||||||
|
|
||||||
|
var logger = require('../../util/logger')
|
||||||
|
var requtil = require('../../util/requtil')
|
||||||
|
var Storage = require('../../util/storage')
|
||||||
|
var wireutil = require('../../wire/util')
|
||||||
|
|
||||||
|
module.exports = function(options) {
|
||||||
|
var log = logger.createLogger('cache-apk')
|
||||||
|
, app = express()
|
||||||
|
, server = http.createServer(app)
|
||||||
|
, storage = new Storage()
|
||||||
|
|
||||||
|
// Output
|
||||||
|
var push = zmq.socket('push')
|
||||||
|
options.endpoints.push.forEach(function(endpoint) {
|
||||||
|
log.info('Sending output to %s', endpoint)
|
||||||
|
push.connect(endpoint)
|
||||||
|
})
|
||||||
|
|
||||||
|
app.set('strict routing', true)
|
||||||
|
app.set('case sensitive routing', true)
|
||||||
|
app.set('trust proxy', true)
|
||||||
|
|
||||||
|
app.use(express.json())
|
||||||
|
app.use(validator())
|
||||||
|
|
||||||
|
storage.on('timeout', function(id) {
|
||||||
|
log.info('Cleaning up inactive resource "%s"', id)
|
||||||
|
})
|
||||||
|
|
||||||
|
function processFile(file) {
|
||||||
|
var resolver = Promise.defer()
|
||||||
|
|
||||||
|
log.info('Processing file "%s"', file.path)
|
||||||
|
|
||||||
|
resolver.progress({
|
||||||
|
percent: 0
|
||||||
|
})
|
||||||
|
|
||||||
|
process.nextTick(function() {
|
||||||
|
try {
|
||||||
|
var reader = ApkReader.readFile(file.path)
|
||||||
|
var manifest = reader.readManifestSync()
|
||||||
|
resolver.resolve(manifest)
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
err.reportCode = 'fail_invalid_app_file'
|
||||||
|
resolver.reject(err)
|
||||||
|
}
|
||||||
|
})
|
||||||
|
|
||||||
|
return resolver.promise
|
||||||
|
}
|
||||||
|
|
||||||
|
function storeFile(file) {
|
||||||
|
var id = storage.store(file)
|
||||||
|
return Promise.resolve({
|
||||||
|
id: id
|
||||||
|
, url: util.format(
|
||||||
|
'http://%s:%s/api/v1/resources/%s'
|
||||||
|
, options.publicIp
|
||||||
|
, options.port
|
||||||
|
, id
|
||||||
|
)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
function download(url) {
|
||||||
|
var resolver = Promise.defer()
|
||||||
|
var path = temp.path({
|
||||||
|
dir: options.saveDir
|
||||||
|
})
|
||||||
|
|
||||||
|
log.info('Downloading "%s" to "%s"', url, path)
|
||||||
|
|
||||||
|
function errorListener(err) {
|
||||||
|
err.reportCode = 'fail_download'
|
||||||
|
resolver.reject(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
function progressListener(state) {
|
||||||
|
resolver.progress(state)
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeListener() {
|
||||||
|
resolver.resolve({
|
||||||
|
path: path
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
resolver.progress({
|
||||||
|
percent: 0
|
||||||
|
})
|
||||||
|
|
||||||
|
try {
|
||||||
|
var req = progress(request(url), {
|
||||||
|
throttle: 100 // Throttle events, not upload speed
|
||||||
|
})
|
||||||
|
.on('progress', progressListener)
|
||||||
|
|
||||||
|
var save = req.pipe(fs.createWriteStream(path))
|
||||||
|
.on('error', errorListener)
|
||||||
|
.on('close', closeListener)
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
err.reportCode = 'fail_invalid_url'
|
||||||
|
resolver.reject(err)
|
||||||
|
}
|
||||||
|
|
||||||
|
return resolver.promise.finally(function() {
|
||||||
|
req.removeListener('progress', progressListener)
|
||||||
|
save.removeListener('error', errorListener)
|
||||||
|
save.removeListener('close', closeListener)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
app.post('/api/v1/cache', function(req, res) {
|
||||||
|
var reply = wireutil.reply(options.id)
|
||||||
|
|
||||||
|
function sendProgress(data, progress) {
|
||||||
|
if (req.query.channel) {
|
||||||
|
push.send([
|
||||||
|
req.query.channel
|
||||||
|
, reply.progress(data, progress)
|
||||||
|
])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function sendDone(success, data, body) {
|
||||||
|
if (req.query.channel) {
|
||||||
|
push.send([
|
||||||
|
req.query.channel
|
||||||
|
, reply.okay(data, body)
|
||||||
|
])
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
requtil.validate(req, function() {
|
||||||
|
req.checkQuery('channel').notEmpty()
|
||||||
|
})
|
||||||
|
.then(function() {
|
||||||
|
return requtil.validate(req, function() {
|
||||||
|
req.checkBody('url').notEmpty()
|
||||||
|
})
|
||||||
|
.then(function() {
|
||||||
|
return download(req.body.url)
|
||||||
|
.progressed(function(progress) {
|
||||||
|
sendProgress('uploading', 0.7 * progress.percent)
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.then(function(file) {
|
||||||
|
return processFile(file)
|
||||||
|
.progressed(function(progress) {
|
||||||
|
sendProgress('processing', 70 + 0.2 * progress.percent)
|
||||||
|
})
|
||||||
|
.then(function(manifest) {
|
||||||
|
sendProgress('storing', 90)
|
||||||
|
return storeFile(file)
|
||||||
|
.then(function(data) {
|
||||||
|
data.manifest = manifest
|
||||||
|
return data
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.then(function(data) {
|
||||||
|
sendDone(true, 'success', data)
|
||||||
|
data.success = true
|
||||||
|
res.json(201, data)
|
||||||
|
})
|
||||||
|
.catch(requtil.ValidationError, function(err) {
|
||||||
|
sendDone(false, err.reportCode || 'fail_validation')
|
||||||
|
res.status(400)
|
||||||
|
.json({
|
||||||
|
success: false
|
||||||
|
, error: 'ValidationError'
|
||||||
|
, validationErrors: err.errors
|
||||||
|
})
|
||||||
|
})
|
||||||
|
.catch(function(err) {
|
||||||
|
log.error('Unexpected error', err.stack)
|
||||||
|
sendDone(false, err.reportCode || 'fail')
|
||||||
|
res.status(500)
|
||||||
|
.json({
|
||||||
|
success: false
|
||||||
|
, error: 'ServerError'
|
||||||
|
})
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
app.get('/api/v1/cache/: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)
|
||||||
|
}
|
|
@ -19,6 +19,7 @@ module.exports = function(options) {
|
||||||
.dependency(require('./device/plugins/solo'))
|
.dependency(require('./device/plugins/solo'))
|
||||||
.dependency(require('./device/plugins/heartbeat'))
|
.dependency(require('./device/plugins/heartbeat'))
|
||||||
.dependency(require('./device/plugins/display'))
|
.dependency(require('./device/plugins/display'))
|
||||||
|
.dependency(require('./device/plugins/screenshot'))
|
||||||
.dependency(require('./device/plugins/http'))
|
.dependency(require('./device/plugins/http'))
|
||||||
.dependency(require('./device/plugins/service'))
|
.dependency(require('./device/plugins/service'))
|
||||||
.dependency(require('./device/plugins/browser'))
|
.dependency(require('./device/plugins/browser'))
|
||||||
|
|
66
lib/roles/device/plugins/screenshot.js
Normal file
66
lib/roles/device/plugins/screenshot.js
Normal file
|
@ -0,0 +1,66 @@
|
||||||
|
var http = require('http')
|
||||||
|
var util = require('util')
|
||||||
|
|
||||||
|
var syrup = require('syrup')
|
||||||
|
var Promise = require('bluebird')
|
||||||
|
|
||||||
|
var logger = require('../../../util/logger')
|
||||||
|
var wire = require('../../../wire')
|
||||||
|
var wireutil = require('../../../wire/util')
|
||||||
|
|
||||||
|
module.exports = syrup.serial()
|
||||||
|
.dependency(require('../support/router'))
|
||||||
|
.dependency(require('../support/push'))
|
||||||
|
.dependency(require('../support/storage'))
|
||||||
|
.dependency(require('./display'))
|
||||||
|
.define(function(options, router, push, storage, display) {
|
||||||
|
var log = logger.createLogger('device:plugins:screenshot')
|
||||||
|
var plugin = Object.create(null)
|
||||||
|
|
||||||
|
plugin.capture = function() {
|
||||||
|
log.info('Capturing screenshot from %s', display.url)
|
||||||
|
|
||||||
|
return new Promise(function(resolve, reject) {
|
||||||
|
var req = http.get(display.url)
|
||||||
|
|
||||||
|
function responseListener(res) {
|
||||||
|
if (res.statusCode !== 200) {
|
||||||
|
reject(new Error(util.format(
|
||||||
|
'Screenshot capture failed: HTTP %d'
|
||||||
|
, res.statusCode
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
resolve(storage.store(res, {
|
||||||
|
filename: util.format('%s.png', options.serial)
|
||||||
|
, contentType: 'image/png'
|
||||||
|
, knownLength: +res.headers['content-length']
|
||||||
|
}))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
req.on('response', responseListener)
|
||||||
|
req.on('error', reject)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
router.on(wire.ScreenCaptureMessage, function(channel) {
|
||||||
|
var reply = wireutil.reply(options.serial)
|
||||||
|
plugin.capture()
|
||||||
|
.then(function(url) {
|
||||||
|
push.send([
|
||||||
|
channel
|
||||||
|
, reply.okay(url)
|
||||||
|
])
|
||||||
|
})
|
||||||
|
.catch(function(err) {
|
||||||
|
log.error('Screen capture failed', err.stack)
|
||||||
|
push.send([
|
||||||
|
channel
|
||||||
|
, reply.fail(err.message)
|
||||||
|
])
|
||||||
|
})
|
||||||
|
})
|
||||||
|
|
||||||
|
return plugin
|
||||||
|
})
|
53
lib/roles/device/support/storage.js
Normal file
53
lib/roles/device/support/storage.js
Normal file
|
@ -0,0 +1,53 @@
|
||||||
|
var util = require('util')
|
||||||
|
|
||||||
|
var syrup = require('syrup')
|
||||||
|
var Promise = require('bluebird')
|
||||||
|
var request = require('request')
|
||||||
|
|
||||||
|
var logger = require('../../../util/logger')
|
||||||
|
|
||||||
|
module.exports = syrup.serial()
|
||||||
|
.define(function(options) {
|
||||||
|
var log = logger.createLogger('device:support:storage')
|
||||||
|
var plugin = Object.create(null)
|
||||||
|
|
||||||
|
plugin.store = function(stream, meta) {
|
||||||
|
var resolver = Promise.defer()
|
||||||
|
|
||||||
|
var req = request.post({
|
||||||
|
url: util.format('%sapi/v1/resources', options.storageUrl)
|
||||||
|
}
|
||||||
|
, function(err, res, body) {
|
||||||
|
if (err) {
|
||||||
|
log.error('Upload failed', err.stack)
|
||||||
|
resolver.reject(err)
|
||||||
|
}
|
||||||
|
else if (res.statusCode !== 201) {
|
||||||
|
log.error('Upload failed: HTTP %d', res.statusCode)
|
||||||
|
resolver.reject(new Error(util.format(
|
||||||
|
'Upload failed: HTTP %d'
|
||||||
|
, res.statusCode
|
||||||
|
)))
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
try {
|
||||||
|
var result = JSON.parse(body)
|
||||||
|
log.info('Uploaded to %s', result.resources.file)
|
||||||
|
resolver.resolve(result.resources.file)
|
||||||
|
}
|
||||||
|
catch (err) {
|
||||||
|
log.error('Invalid JSON in response', err.stack, body)
|
||||||
|
resolver.reject(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
req.form()
|
||||||
|
.append('file', stream, meta)
|
||||||
|
|
||||||
|
return resolver.promise
|
||||||
|
}
|
||||||
|
|
||||||
|
return plugin
|
||||||
|
})
|
|
@ -1,21 +1,14 @@
|
||||||
var http = require('http')
|
var http = require('http')
|
||||||
var util = require('util')
|
var util = require('util')
|
||||||
var fs = require('fs')
|
var path = require('path')
|
||||||
|
|
||||||
var express = require('express')
|
var express = require('express')
|
||||||
var validator = require('express-validator')
|
|
||||||
var formidable = require('formidable')
|
var formidable = require('formidable')
|
||||||
var Promise = require('bluebird')
|
var Promise = require('bluebird')
|
||||||
var ApkReader = require('adbkit-apkreader')
|
|
||||||
var request = require('request')
|
|
||||||
var progress = require('request-progress')
|
|
||||||
var temp = require('temp')
|
|
||||||
var zmq = require('zmq')
|
|
||||||
|
|
||||||
var logger = require('../../util/logger')
|
var logger = require('../../util/logger')
|
||||||
var requtil = require('../../util/requtil')
|
var requtil = require('../../util/requtil')
|
||||||
var Storage = require('../../util/storage')
|
var Storage = require('../../util/storage')
|
||||||
var wireutil = require('../../wire/util')
|
|
||||||
|
|
||||||
module.exports = function(options) {
|
module.exports = function(options) {
|
||||||
var log = logger.createLogger('storage-temp')
|
var log = logger.createLogger('storage-temp')
|
||||||
|
@ -23,193 +16,48 @@ module.exports = function(options) {
|
||||||
, server = http.createServer(app)
|
, server = http.createServer(app)
|
||||||
, storage = new Storage()
|
, storage = new Storage()
|
||||||
|
|
||||||
// Output
|
|
||||||
var push = zmq.socket('push')
|
|
||||||
options.endpoints.push.forEach(function(endpoint) {
|
|
||||||
log.info('Sending output to %s', endpoint)
|
|
||||||
push.connect(endpoint)
|
|
||||||
})
|
|
||||||
|
|
||||||
app.set('strict routing', true)
|
app.set('strict routing', true)
|
||||||
app.set('case sensitive routing', true)
|
app.set('case sensitive routing', true)
|
||||||
app.set('trust proxy', true)
|
app.set('trust proxy', true)
|
||||||
|
|
||||||
app.use(express.json())
|
|
||||||
app.use(validator())
|
|
||||||
|
|
||||||
storage.on('timeout', function(id) {
|
storage.on('timeout', function(id) {
|
||||||
log.info('Cleaning up inactive resource "%s"', id)
|
log.info('Cleaning up inactive resource "%s"', id)
|
||||||
})
|
})
|
||||||
|
|
||||||
function processFile(file) {
|
|
||||||
var resolver = Promise.defer()
|
|
||||||
|
|
||||||
log.info('Processing file "%s"', file.path)
|
|
||||||
|
|
||||||
resolver.progress({
|
|
||||||
percent: 0
|
|
||||||
})
|
|
||||||
|
|
||||||
process.nextTick(function() {
|
|
||||||
try {
|
|
||||||
var reader = ApkReader.readFile(file.path)
|
|
||||||
var manifest = reader.readManifestSync()
|
|
||||||
resolver.resolve(manifest)
|
|
||||||
}
|
|
||||||
catch (err) {
|
|
||||||
err.reportCode = 'fail_invalid_app_file'
|
|
||||||
resolver.reject(err)
|
|
||||||
}
|
|
||||||
})
|
|
||||||
|
|
||||||
return resolver.promise
|
|
||||||
}
|
|
||||||
|
|
||||||
function storeFile(file) {
|
|
||||||
var id = storage.store(file)
|
|
||||||
return Promise.resolve({
|
|
||||||
id: id
|
|
||||||
, url: util.format(
|
|
||||||
'http://%s:%s/api/v1/resources/%s'
|
|
||||||
, options.publicIp
|
|
||||||
, options.port
|
|
||||||
, id
|
|
||||||
)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
function download(url) {
|
|
||||||
var resolver = Promise.defer()
|
|
||||||
var path = temp.path({
|
|
||||||
dir: options.saveDir
|
|
||||||
})
|
|
||||||
|
|
||||||
log.info('Downloading "%s" to "%s"', url, path)
|
|
||||||
|
|
||||||
function errorListener(err) {
|
|
||||||
err.reportCode = 'fail_download'
|
|
||||||
resolver.reject(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
function progressListener(state) {
|
|
||||||
resolver.progress(state)
|
|
||||||
}
|
|
||||||
|
|
||||||
function closeListener() {
|
|
||||||
resolver.resolve({
|
|
||||||
path: path
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
resolver.progress({
|
|
||||||
percent: 0
|
|
||||||
})
|
|
||||||
|
|
||||||
try {
|
|
||||||
var req = progress(request(url), {
|
|
||||||
throttle: 100 // Throttle events, not upload speed
|
|
||||||
})
|
|
||||||
.on('progress', progressListener)
|
|
||||||
|
|
||||||
var save = req.pipe(fs.createWriteStream(path))
|
|
||||||
.on('error', errorListener)
|
|
||||||
.on('close', closeListener)
|
|
||||||
}
|
|
||||||
catch (err) {
|
|
||||||
err.reportCode = 'fail_invalid_url'
|
|
||||||
resolver.reject(err)
|
|
||||||
}
|
|
||||||
|
|
||||||
return resolver.promise.finally(function() {
|
|
||||||
req.removeListener('progress', progressListener)
|
|
||||||
save.removeListener('error', errorListener)
|
|
||||||
save.removeListener('close', closeListener)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
app.post('/api/v1/resources', function(req, res) {
|
app.post('/api/v1/resources', function(req, res) {
|
||||||
var reply = wireutil.reply(options.id)
|
var form = new formidable.IncomingForm()
|
||||||
|
Promise.promisify(form.parse, form)(req)
|
||||||
function sendProgress(data, progress) {
|
.spread(function(fields, files) {
|
||||||
if (req.query.channel) {
|
return Object.keys(files).map(function(field) {
|
||||||
push.send([
|
return {
|
||||||
req.query.channel
|
field: field
|
||||||
, reply.progress(data, progress)
|
, id: storage.store(files[field])
|
||||||
])
|
, name: files[field].name
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function sendDone(success, data, body) {
|
|
||||||
if (req.query.channel) {
|
|
||||||
push.send([
|
|
||||||
req.query.channel
|
|
||||||
, reply.okay(data, body)
|
|
||||||
])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
requtil.validate(req, function() {
|
|
||||||
req.checkQuery('channel').notEmpty()
|
|
||||||
})
|
|
||||||
.then(function() {
|
|
||||||
if (req.is('application/json')) {
|
|
||||||
return requtil.validate(req, function() {
|
|
||||||
req.checkBody('url').notEmpty()
|
|
||||||
})
|
|
||||||
.then(function() {
|
|
||||||
return download(req.body.url)
|
|
||||||
.progressed(function(progress) {
|
|
||||||
sendProgress('uploading', 0.7 * progress.percent)
|
|
||||||
})
|
|
||||||
})
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
var form = Promise.promisifyAll(new formidable.IncomingForm())
|
|
||||||
var progressListener = function(received, expected) {
|
|
||||||
if (expected) {
|
|
||||||
sendProgress('uploading', 70 * (received / expected))
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
sendProgress('uploading', 0)
|
})
|
||||||
form.on('progress', progressListener)
|
|
||||||
return form.parseAsync(req)
|
|
||||||
.finally(function() {
|
|
||||||
form.removeListener('progress', progressListener)
|
|
||||||
})
|
|
||||||
.spread(function(fields, files) {
|
|
||||||
if (!files.file) {
|
|
||||||
throw new requtil.ValidationError('validation error', [
|
|
||||||
{
|
|
||||||
"param": "file"
|
|
||||||
, "msg": "Required value"
|
|
||||||
}
|
|
||||||
])
|
|
||||||
}
|
|
||||||
return files.file
|
|
||||||
})
|
|
||||||
}
|
|
||||||
})
|
})
|
||||||
.then(function(file) {
|
.then(function(storedFiles) {
|
||||||
return processFile(file)
|
res.status(201)
|
||||||
.progressed(function(progress) {
|
.json({
|
||||||
sendProgress('processing', 70 + 0.2 * progress.percent)
|
success: true
|
||||||
})
|
, resources: (function() {
|
||||||
.then(function(manifest) {
|
var mapped = Object.create(null)
|
||||||
sendProgress('storing', 90)
|
storedFiles.forEach(function(file) {
|
||||||
return storeFile(file)
|
mapped[file.field] = util.format(
|
||||||
.then(function(data) {
|
'http://%s:%s/api/v1/resources/%s%s'
|
||||||
data.manifest = manifest
|
, options.publicIp
|
||||||
return data
|
, options.port
|
||||||
|
, file.id
|
||||||
|
, file.name
|
||||||
|
? util.format('/%s', path.basename(file.name))
|
||||||
|
: ''
|
||||||
|
)
|
||||||
})
|
})
|
||||||
|
return mapped
|
||||||
|
})()
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
.then(function(data) {
|
|
||||||
sendDone(true, 'success', data)
|
|
||||||
data.success = true
|
|
||||||
res.json(201, data)
|
|
||||||
})
|
|
||||||
.catch(requtil.ValidationError, function(err) {
|
.catch(requtil.ValidationError, function(err) {
|
||||||
sendDone(false, err.reportCode || 'fail_validation')
|
|
||||||
res.status(400)
|
res.status(400)
|
||||||
.json({
|
.json({
|
||||||
success: false
|
success: false
|
||||||
|
@ -218,8 +66,7 @@ module.exports = function(options) {
|
||||||
})
|
})
|
||||||
})
|
})
|
||||||
.catch(function(err) {
|
.catch(function(err) {
|
||||||
log.error('Unexpected error', err.stack)
|
log.error('Error storing resource', err.stack)
|
||||||
sendDone(false, err.reportCode || 'fail')
|
|
||||||
res.status(500)
|
res.status(500)
|
||||||
.json({
|
.json({
|
||||||
success: false
|
success: false
|
||||||
|
@ -239,6 +86,17 @@ module.exports = function(options) {
|
||||||
}
|
}
|
||||||
})
|
})
|
||||||
|
|
||||||
|
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)
|
server.listen(options.port)
|
||||||
log.info('Listening on port %d', options.port)
|
log.info('Listening on port %d', options.port)
|
||||||
}
|
}
|
||||||
|
|
|
@ -50,6 +50,7 @@ enum MessageType {
|
||||||
PhoneStateEvent = 47;
|
PhoneStateEvent = 47;
|
||||||
RotationEvent = 48;
|
RotationEvent = 48;
|
||||||
StoreOpenMessage = 49;
|
StoreOpenMessage = 49;
|
||||||
|
ScreenCaptureMessage = 50;
|
||||||
}
|
}
|
||||||
|
|
||||||
message Envelope {
|
message Envelope {
|
||||||
|
@ -400,6 +401,9 @@ message BrowserClearMessage {
|
||||||
message StoreOpenMessage {
|
message StoreOpenMessage {
|
||||||
}
|
}
|
||||||
|
|
||||||
|
message ScreenCaptureMessage {
|
||||||
|
}
|
||||||
|
|
||||||
// Events, these must be kept in sync with STFService/wire.proto
|
// Events, these must be kept in sync with STFService/wire.proto
|
||||||
|
|
||||||
message AirplaneModeEvent {
|
message AirplaneModeEvent {
|
||||||
|
|
|
@ -216,6 +216,10 @@ module.exports = function ControlServiceFactory(
|
||||||
return sendTwoWay('store.open')
|
return sendTwoWay('store.open')
|
||||||
}
|
}
|
||||||
|
|
||||||
|
this.screenshot = function() {
|
||||||
|
return sendTwoWay('screen.capture')
|
||||||
|
}
|
||||||
|
|
||||||
window.cc = this
|
window.cc = this
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue