Merge pull request #197 from feross/autoupdater

OS X: Add auto-updater, sign app, build .DMG and .ZIP assets
This commit is contained in:
Feross Aboukhadijeh
2016-03-21 20:33:45 -07:00
13 changed files with 184 additions and 52 deletions

View File

@@ -10,18 +10,25 @@ var electronPackager = require('electron-packager')
var fs = require('fs') var fs = require('fs')
var path = require('path') var path = require('path')
var pkg = require('../package.json') var pkg = require('../package.json')
var plist = require('plist')
var BUILD_NAME = config.APP_NAME + '-v' + config.APP_VERSION
function build () { function build () {
var platform = process.argv[2] var platform = process.argv[2]
if (platform === '--darwin') { if (platform === '--darwin') {
buildDarwin() buildDarwin(printDone)
} else if (platform === '--win32') { } else if (platform === '--win32') {
buildWin32() buildWin32(printDone)
} else if (platform === '--linux') { } else if (platform === '--linux') {
buildLinux() buildLinux(printDone)
} else { } else {
buildDarwin(() => buildWin32(() => buildLinux())) // Build all buildDarwin(function (err, buildPath) {
printDone(err, buildPath)
buildWin32(function (err, buildPath) {
printDone(err, buildPath)
buildLinux(printDone)
})
})
} }
} }
@@ -30,7 +37,7 @@ var all = {
arch: 'x64', arch: 'x64',
// The application source directory. // The application source directory.
dir: path.join(__dirname, '..'), dir: config.ROOT_PATH,
// The release version of the application. Maps to the `ProductVersion` metadata // The release version of the application. Maps to the `ProductVersion` metadata
// property on Windows, and `CFBundleShortVersionString` on OS X. // property on Windows, and `CFBundleShortVersionString` on OS X.
@@ -58,7 +65,7 @@ var all = {
name: config.APP_NAME, name: config.APP_NAME,
// The base directory where the finished package(s) are created. // The base directory where the finished package(s) are created.
out: path.join(__dirname, '..', 'dist'), out: path.join(config.ROOT_PATH, 'dist'),
// Replace an already existing output directory. // Replace an already existing output directory.
overwrite: true, overwrite: true,
@@ -130,18 +137,15 @@ var linux = {
build() build()
function buildDarwin (cb) { function buildDarwin (cb) {
electronPackager(Object.assign({}, all, darwin), function (err, appPath) { var appDmg = require('appdmg')
printDone(err, appPath) var plist = require('plist')
var sign = require('electron-osx-sign')
electronPackager(Object.assign({}, all, darwin), function (err, buildPath) {
if (err) return cb(err) if (err) return cb(err)
var contentsPath = path.join( var appPath = path.join(buildPath[0], config.APP_NAME + '.app')
__dirname, var contentsPath = path.join(appPath, 'Contents')
'..',
'dist',
`${config.APP_NAME}-darwin-x64`,
`${config.APP_NAME}.app`,
'Contents'
)
var resourcesPath = path.join(contentsPath, 'Resources') var resourcesPath = path.join(contentsPath, 'Resources')
var infoPlistPath = path.join(contentsPath, 'Info.plist') var infoPlistPath = path.join(contentsPath, 'Info.plist')
var infoPlist = plist.parse(fs.readFileSync(infoPlistPath, 'utf8')) var infoPlist = plist.parse(fs.readFileSync(infoPlistPath, 'utf8'))
@@ -178,25 +182,70 @@ function buildDarwin (cb) {
fs.writeFileSync(infoPlistPath, plist.build(infoPlist)) fs.writeFileSync(infoPlistPath, plist.build(infoPlist))
cp.execSync(`cp ${config.APP_FILE_ICON + '.icns'} ${resourcesPath}`) cp.execSync(`cp ${config.APP_FILE_ICON + '.icns'} ${resourcesPath}`)
if (cb) cb(null) var zipPath = path.join(buildPath[0], BUILD_NAME + '.zip')
cp.execSync(`zip -r -y ${zipPath} ${appPath}`)
/*
* Signing OS X apps for distribution outside the App Store requires:
*
* - Xcode
* - Xcode Command Line Tools (xcode-select --install)
* - Membership in the Apple Developer Program
*/
if (process.platform === 'darwin') {
var signOpts = {
app: appPath,
platform: 'darwin',
verbose: true
}
var dmgPath = path.join(buildPath[0], BUILD_NAME + '.dmg')
var dmgOpts = {
basepath: config.ROOT_PATH,
target: dmgPath,
specification: {
title: config.APP_NAME,
icon: config.APP_ICON + '.icns',
background: path.join(config.STATIC_PATH, 'appdmg.png'),
'icon-size': 128,
contents: [
{ x: 122, y: 240, type: 'file', path: appPath },
{ x: 380, y: 240, type: 'link', path: '/Applications' },
// Hide hidden icons out of view, for users who have hidden files shown.
// https://github.com/LinusU/node-appdmg/issues/45#issuecomment-153924954
{ x: 50, y: 500, type: 'position', path: '.background' },
{ x: 100, y: 500, type: 'position', path: '.DS_Store' },
{ x: 150, y: 500, type: 'position', path: '.Trashes' },
{ x: 200, y: 500, type: 'position', path: '.VolumeIcon.icns' }
]
}
}
sign(signOpts, function (err) {
if (err) return cb(err)
var dmg = appDmg(dmgOpts)
dmg.on('error', cb)
dmg.on('progress', function (info) {
if (info.type === 'step-begin') console.log(info.title + '...')
})
dmg.on('finish', function (info) {
cb(null, buildPath)
})
})
}
}) })
} }
function buildWin32 (cb) { function buildWin32 (cb) {
electronPackager(Object.assign({}, all, win32), function (err, appPath) { electronPackager(Object.assign({}, all, win32), cb)
printDone(err, appPath)
if (cb) cb(err)
})
} }
function buildLinux (cb) { function buildLinux (cb) {
electronPackager(Object.assign({}, all, linux), function (err, appPath) { electronPackager(Object.assign({}, all, linux), cb)
printDone(err, appPath)
if (cb) cb(err)
})
} }
function printDone (err, appPath) { function printDone (err, buildPath) {
if (err) console.error(err.message || err) if (err) console.error(err.message || err)
else console.log('Built ' + appPath) else console.log('Built ' + buildPath[0])
} }

View File

@@ -1,20 +1,28 @@
var applicationConfigPath = require('application-config-path') var applicationConfigPath = require('application-config-path')
var path = require('path') var path = require('path')
var APP_NAME = 'WebTorrent'
var APP_VERSION = require('./package.json').version
module.exports = { module.exports = {
APP_COPYRIGHT: 'Copyright © 2014-2016 The WebTorrent Project', APP_COPYRIGHT: 'Copyright © 2014-2016 The WebTorrent Project',
APP_FILE_ICON: path.join(__dirname, 'static', 'WebTorrentFile'), APP_FILE_ICON: path.join(__dirname, 'static', 'WebTorrentFile'),
APP_ICON: path.join(__dirname, 'static', 'WebTorrent'), APP_ICON: path.join(__dirname, 'static', 'WebTorrent'),
APP_NAME: 'WebTorrent', APP_NAME: APP_NAME,
APP_VERSION: APP_VERSION,
CONFIG_PATH: applicationConfigPath('WebTorrent'), AUTO_UPDATE_URL: 'https://webtorrent.io/app/update?version=' + APP_VERSION,
CONFIG_POSTER_PATH: path.join(applicationConfigPath('WebTorrent'), 'Posters'), AUTO_UPDATE_CHECK_STARTUP_DELAY: 60 * 1000 /* 1 minute */,
CONFIG_TORRENT_PATH: path.join(applicationConfigPath('WebTorrent'), 'Torrents'),
CONFIG_PATH: applicationConfigPath(APP_NAME),
CONFIG_POSTER_PATH: path.join(applicationConfigPath(APP_NAME), 'Posters'),
CONFIG_TORRENT_PATH: path.join(applicationConfigPath(APP_NAME), 'Torrents'),
INDEX: 'file://' + path.join(__dirname, 'renderer', 'index.html'), INDEX: 'file://' + path.join(__dirname, 'renderer', 'index.html'),
IS_PRODUCTION: isProduction(), IS_PRODUCTION: isProduction(),
ROOT_PATH: __dirname,
STATIC_PATH: path.join(__dirname, 'static'), STATIC_PATH: path.join(__dirname, 'static'),
SOUND_ADD: 'file://' + path.join(__dirname, 'static', 'sound', 'add.wav'), SOUND_ADD: 'file://' + path.join(__dirname, 'static', 'sound', 'add.wav'),

View File

@@ -1,10 +1,2 @@
console.time('init') console.time('init')
require('./main') require('./main')
// report crashes
// require('crash-reporter').start({
// productName: 'WebTorrent',
// companyName: 'WebTorrent',
// submitURL: 'https://webtorrent.io/crash-report',
// autoSubmit: true
// })

31
main/auto-updater.js Normal file
View File

@@ -0,0 +1,31 @@
module.exports = {
init
}
var electron = require('electron')
var config = require('../config')
var log = require('./log')
var autoUpdater = electron.autoUpdater
function init () {
autoUpdater.on('error', function (err) {
log.error('App update error: ' + err.message || err)
})
autoUpdater.setFeedURL(config.AUTO_UPDATE_URL)
/*
* We always check for updates on app startup. To keep app startup fast, we delay this
* first check so it happens when there is less going on.
*/
setTimeout(() => autoUpdater.checkForUpdates(), config.AUTO_UPDATE_CHECK_STARTUP_DELAY)
autoUpdater.on('checking-for-update', () => log('Checking for app update'))
autoUpdater.on('update-available', () => log('App update available'))
autoUpdater.on('update-not-available', () => log('App update not available'))
autoUpdater.on('update-downloaded', function (e, releaseNotes, releaseName, releaseDate, updateURL) {
log('App update downloaded: ', releaseName, updateURL)
})
}

View File

@@ -2,8 +2,10 @@ var electron = require('electron')
var app = electron.app var app = electron.app
var autoUpdater = require('./auto-updater')
var config = require('../config') var config = require('../config')
var ipc = require('./ipc') var ipc = require('./ipc')
var log = require('./log')
var menu = require('./menu') var menu = require('./menu')
var registerProtocolHandler = require('./register-handlers') var registerProtocolHandler = require('./register-handlers')
var shortcuts = require('./shortcuts') var shortcuts = require('./shortcuts')
@@ -15,7 +17,7 @@ var shouldQuit = app.makeSingleInstance(function (newArgv) {
newArgv = sliceArgv(newArgv) newArgv = sliceArgv(newArgv)
if (app.ipcReady) { if (app.ipcReady) {
windows.main.send('log', 'Second app instance attempted to open but was prevented') log('Second app instance attempted to open but was prevented')
newArgv.forEach(function (torrentId) { newArgv.forEach(function (torrentId) {
windows.main.send('dispatch', 'onOpen', torrentId) windows.main.send('dispatch', 'onOpen', torrentId)
@@ -38,6 +40,10 @@ var argv = sliceArgv(process.argv)
app.on('open-file', onOpen) app.on('open-file', onOpen)
app.on('open-url', onOpen) app.on('open-url', onOpen)
app.on('will-finish-launching', function () {
autoUpdater.init()
setupCrashReporter()
})
app.ipcReady = false // main window has finished loading and IPC is ready app.ipcReady = false // main window has finished loading and IPC is ready
app.isQuitting = false app.isQuitting = false
@@ -50,9 +56,9 @@ app.on('ready', function () {
}) })
app.on('ipcReady', function () { app.on('ipcReady', function () {
windows.main.send('log', 'IS_PRODUCTION:', config.IS_PRODUCTION) log('IS_PRODUCTION:', config.IS_PRODUCTION)
if (argv.length) { if (argv.length) {
windows.main.send('log', 'command line args:', process.argv) log('command line args:', process.argv)
} }
argv.forEach(function (torrentId) { argv.forEach(function (torrentId) {
windows.main.send('dispatch', 'onOpen', torrentId) windows.main.send('dispatch', 'onOpen', torrentId)
@@ -91,3 +97,12 @@ function onOpen (e, torrentId) {
function sliceArgv (argv) { function sliceArgv (argv) {
return argv.slice(config.IS_PRODUCTION ? 1 : 2) return argv.slice(config.IS_PRODUCTION ? 1 : 2)
} }
function setupCrashReporter () {
// require('crash-reporter').start({
// productName: 'WebTorrent',
// companyName: 'WebTorrent',
// submitURL: 'https://webtorrent.io/crash-report',
// autoSubmit: true
// })
}

View File

@@ -9,6 +9,7 @@ var app = electron.app
var ipcMain = electron.ipcMain var ipcMain = electron.ipcMain
var powerSaveBlocker = electron.powerSaveBlocker var powerSaveBlocker = electron.powerSaveBlocker
var log = require('./log')
var menu = require('./menu') var menu = require('./menu')
var windows = require('./windows') var windows = require('./windows')
@@ -51,7 +52,7 @@ function init () {
}) })
ipcMain.on('openItem', function (e, path) { ipcMain.on('openItem', function (e, path) {
console.log('opening file or folder: ' + path) log('opening file or folder: ' + path)
electron.shell.openItem(path) electron.shell.openItem(path)
}) })

31
main/log.js Normal file
View File

@@ -0,0 +1,31 @@
module.exports = log
module.exports.error = error
/**
* In the main electron process, we do not use console.log() statements because they do
* not show up in a convenient location when running the packaged (i.e. production)
* version of the app. Instead use this module, which sends the logs to the main window
* where they can be viewed in Developer Tools.
*/
var electron = require('electron')
var windows = require('./windows')
var app = electron.app
function log (...args) {
if (app.ipcReady) {
windows.main.send('log', ...args)
} else {
app.on('ipcReady', () => windows.main.send('log', ...args))
}
}
function error (...args) {
if (app.ipcReady) {
windows.main.send('error', ...args)
} else {
app.on('ipcReady', () => windows.main.send('error', ...args))
}
}

View File

@@ -1,3 +1,5 @@
var log = require('./log')
module.exports = function () { module.exports = function () {
if (process.platform === 'win32') { if (process.platform === 'win32') {
var path = require('path') var path = require('path')
@@ -86,7 +88,7 @@ function registerProtocolHandlerWin32 (protocol, name, icon, command) {
commandKey.set('', Registry.REG_SZ, '"' + command + '" "%1"', callback) commandKey.set('', Registry.REG_SZ, '"' + command + '" "%1"', callback)
function callback (err) { function callback (err) {
if (err) console.error(err.message || err) if (err) log.error(err.message || err)
} }
} }
@@ -118,6 +120,6 @@ function registerFileHandlerWin32 (ext, id, name, icon, command) {
commandKey.set('', Registry.REG_SZ, '"' + command + '" "%1"', callback) commandKey.set('', Registry.REG_SZ, '"' + command + '" "%1"', callback)
function callback (err) { function callback (err) {
if (err) console.error(err.message || err) if (err) log.error(err.message || err)
} }
} }

View File

@@ -30,6 +30,8 @@
"winreg": "^1.0.1" "winreg": "^1.0.1"
}, },
"devDependencies": { "devDependencies": {
"appdmg": "^0.3.6",
"electron-osx-sign": "^0.3.0",
"electron-packager": "^5.0.0", "electron-packager": "^5.0.0",
"electron-prebuilt": "0.37.2", "electron-prebuilt": "0.37.2",
"path-exists": "^2.1.0", "path-exists": "^2.1.0",

View File

@@ -283,6 +283,7 @@ function setupIpc () {
ipcRenderer.send('ipcReady') ipcRenderer.send('ipcReady')
ipcRenderer.on('log', (e, ...args) => console.log(...args)) ipcRenderer.on('log', (e, ...args) => console.log(...args))
ipcRenderer.on('error', (e, ...args) => console.error(...args))
ipcRenderer.on('dispatch', (e, ...args) => dispatch(...args)) ipcRenderer.on('dispatch', (e, ...args) => dispatch(...args))

View File

@@ -65,8 +65,8 @@ module.exports = {
status: 'paused', status: 'paused',
infoHash: '88594aaacbde40ef3e2510c47374ec0aa396c08e', infoHash: '88594aaacbde40ef3e2510c47374ec0aa396c08e',
displayName: 'Big Buck Bunny', displayName: 'Big Buck Bunny',
posterURL: path.join(__dirname, '..', 'static', 'bigBuckBunny.jpg'), posterURL: path.join(config.ROOT_PATH, 'static', 'bigBuckBunny.jpg'),
torrentPath: path.join(__dirname, '..', 'static', 'bigBuckBunny.torrent'), torrentPath: path.join(config.ROOT_PATH, 'static', 'bigBuckBunny.torrent'),
files: [ files: [
{ {
'name': 'bbb_sunflower_1080p_30fps_normal.mp4', 'name': 'bbb_sunflower_1080p_30fps_normal.mp4',
@@ -80,8 +80,8 @@ module.exports = {
status: 'paused', status: 'paused',
infoHash: '6a9759bffd5c0af65319979fb7832189f4f3c35d', infoHash: '6a9759bffd5c0af65319979fb7832189f4f3c35d',
displayName: 'Sintel', displayName: 'Sintel',
posterURL: path.join(__dirname, '..', 'static', 'sintel.jpg'), posterURL: path.join(config.ROOT_PATH, 'static', 'sintel.jpg'),
torrentPath: path.join(__dirname, '..', 'static', 'sintel.torrent'), torrentPath: path.join(config.ROOT_PATH, 'static', 'sintel.torrent'),
files: [ files: [
{ {
'name': 'sintel.mp4', 'name': 'sintel.mp4',
@@ -95,8 +95,8 @@ module.exports = {
status: 'paused', status: 'paused',
infoHash: '02767050e0be2fd4db9a2ad6c12416ac806ed6ed', infoHash: '02767050e0be2fd4db9a2ad6c12416ac806ed6ed',
displayName: 'Tears of Steel', displayName: 'Tears of Steel',
posterURL: path.join(__dirname, '..', 'static', 'tearsOfSteel.jpg'), posterURL: path.join(config.ROOT_PATH, 'static', 'tearsOfSteel.jpg'),
torrentPath: path.join(__dirname, '..', 'static', 'tearsOfSteel.torrent'), torrentPath: path.join(config.ROOT_PATH, 'static', 'tearsOfSteel.torrent'),
files: [ files: [
{ {
'name': 'tears_of_steel_1080p.webm', 'name': 'tears_of_steel_1080p.webm',

BIN
static/appdmg.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 10 KiB

BIN
static/appdmg@2x.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 22 KiB