123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379 |
- /**
- * @module sign
- */
- 'use strict'
- const path = require('path')
- const Promise = require('bluebird')
- const compareVersion = require('compare-version')
- const pkg = require('./package.json')
- const util = require('./util')
- const debuglog = util.debuglog
- const debugwarn = util.debugwarn
- const getAppContentsPath = util.getAppContentsPath
- const execFileAsync = util.execFileAsync
- const validateOptsAppAsync = util.validateOptsAppAsync
- const validateOptsPlatformAsync = util.validateOptsPlatformAsync
- const walkAsync = util.walkAsync
- const Identity = require('./util-identities').Identity
- const findIdentitiesAsync = require('./util-identities').findIdentitiesAsync
- const ProvisioningProfile = require('./util-provisioning-profiles').ProvisioningProfile
- const preEmbedProvisioningProfile = require('./util-provisioning-profiles').preEmbedProvisioningProfile
- const preAutoEntitlements = require('./util-entitlements').preAutoEntitlements
- /**
- * This function returns a promise validating opts.binaries, the additional binaries to be signed along with the discovered enclosed components.
- * @function
- * @param {Object} opts - Options.
- * @returns {Promise} Promise.
- */
- function validateOptsBinariesAsync (opts) {
- return new Promise(function (resolve, reject) {
- if (opts.binaries) {
- if (!Array.isArray(opts.binaries)) {
- reject(new Error('Additional binaries should be an Array.'))
- return
- }
- // TODO: Presence check for binary files, reject if any does not exist
- }
- resolve()
- })
- }
- /**
- * This function returns a promise validating all options passed in opts.
- * @function
- * @param {Object} opts - Options.
- * @returns {Promise} Promise.
- */
- function validateSignOptsAsync (opts) {
- if (opts.ignore && !(opts.ignore instanceof Array)) {
- opts.ignore = [opts.ignore]
- }
- if (opts['provisioning-profile']) {
- if (typeof opts['provisioning-profile'] !== 'string' && !(opts['provisioning-profile'] instanceof ProvisioningProfile)) return Promise.reject(new Error('Path to provisioning profile should be a string or a ProvisioningProfile object.'))
- }
- if (opts['type']) {
- if (opts['type'] !== 'development' && opts['type'] !== 'distribution') return Promise.reject(new Error('Type must be either `development` or `distribution`.'))
- } else {
- opts['type'] = 'distribution'
- }
- return Promise.map([
- validateOptsAppAsync,
- validateOptsPlatformAsync,
- validateOptsBinariesAsync
- ], function (validate) {
- return validate(opts)
- })
- }
- /**
- * This function returns a promise verifying the code sign of application bundle.
- * @function
- * @param {Object} opts - Options.
- * @returns {Promise} Promise resolving output.
- */
- function verifySignApplicationAsync (opts) {
- // Verify with codesign
- var compareVersion = require('compare-version')
- var osRelease = require('os').release()
- debuglog('Verifying application bundle with codesign...')
- var promise = execFileAsync('codesign', [
- '--verify',
- '--deep'
- ]
- .concat(
- opts['strict-verify'] !== false &&
- compareVersion(osRelease, '15.0.0') >= 0 // Only pass strict flag in El Capitan and later
- ? ['--strict' +
- (opts['strict-verify']
- ? '=' + opts['strict-verify'] // Array should be converted to a comma separated string
- : '')]
- : [],
- ['--verbose=2', opts.app]))
- // Additionally test Gatekeeper acceptance for darwin platform
- if (opts.platform === 'darwin' && opts['gatekeeper-assess'] !== false) {
- promise = promise
- .then(function () {
- debuglog('Verifying Gatekeeper acceptance for darwin platform...')
- return execFileAsync('spctl', [
- '--assess',
- '--type', 'execute',
- '--verbose',
- '--ignore-cache',
- '--no-cache',
- opts.app
- ])
- })
- }
- return promise
- .thenReturn()
- }
- /**
- * This function returns a promise codesigning only.
- * @function
- * @param {Object} opts - Options.
- * @returns {Promise} Promise.
- */
- function signApplicationAsync (opts) {
- return walkAsync(getAppContentsPath(opts))
- .then(function (childPaths) {
- function ignoreFilePath (opts, filePath) {
- if (opts.ignore) {
- return opts.ignore.some(function (ignore) {
- if (typeof ignore === 'function') {
- return ignore(filePath)
- }
- return filePath.match(ignore)
- })
- }
- return false
- }
- if (opts.binaries) childPaths = childPaths.concat(opts.binaries)
- var args = [
- '--sign', opts.identity.hash || opts.identity.name,
- '--force'
- ]
- if (opts.keychain) {
- args.push('--keychain', opts.keychain)
- }
- if (opts.requirements) {
- args.push('--requirements', opts.requirements)
- }
- if (opts.timestamp) {
- args.push('--timestamp=' + opts.timestamp)
- }
- var promise
- if (opts.entitlements) {
- // Sign with entitlements
- promise = Promise.mapSeries(childPaths, function (filePath) {
- if (ignoreFilePath(opts, filePath)) {
- debuglog('Skipped... ' + filePath)
- return
- }
- debuglog('Signing... ' + filePath)
- return execFileAsync('codesign', args.concat('--entitlements', opts['entitlements-inherit'], filePath))
- })
- .then(function () {
- debuglog('Signing... ' + opts.app)
- return execFileAsync('codesign', args.concat('--entitlements', opts.entitlements, opts.app))
- })
- } else {
- // Otherwise normally
- promise = Promise.mapSeries(childPaths, function (filePath) {
- if (ignoreFilePath(opts, filePath)) {
- debuglog('Skipped... ' + filePath)
- return
- }
- debuglog('Signing... ' + filePath)
- return execFileAsync('codesign', args.concat(filePath))
- })
- .then(function () {
- debuglog('Signing... ' + opts.app)
- return execFileAsync('codesign', args.concat(opts.app))
- })
- }
- return promise
- .then(function () {
- // Verify code sign
- debuglog('Verifying...')
- var promise = verifySignApplicationAsync(opts)
- .then(function (result) {
- debuglog('Verified.')
- })
- // Check entitlements if applicable
- if (opts.entitlements) {
- promise = promise
- .then(function () {
- debuglog('Displaying entitlements...')
- return execFileAsync('codesign', [
- '--display',
- '--entitlements', ':-', // Write to standard output and strip off the blob header
- opts.app
- ])
- })
- .then(function (result) {
- debuglog('Entitlements:', '\n',
- result)
- })
- }
- return promise
- })
- })
- }
- /**
- * This function returns a promise signing the application.
- * @function
- * @param {mixed} opts - Options.
- * @returns {Promise} Promise.
- */
- var signAsync = module.exports.signAsync = function (opts) {
- debuglog('electron-osx-sign@%s', pkg.version)
- return validateSignOptsAsync(opts)
- .then(function () {
- // Determine identity for signing
- var promise
- if (opts.identity) {
- debuglog('`identity` passed in arguments.')
- if (opts['identity-validation'] === false) {
- if (!(opts.identity instanceof Identity)) {
- opts.identity = new Identity(opts.identity)
- }
- return Promise.resolve()
- }
- promise = findIdentitiesAsync(opts, opts.identity)
- } else {
- debugwarn('No `identity` passed in arguments...')
- if (opts.platform === 'mas') {
- if (opts.type === 'distribution') {
- debuglog('Finding `3rd Party Mac Developer Application` certificate for signing app distribution in the Mac App Store...')
- promise = findIdentitiesAsync(opts, '3rd Party Mac Developer Application:')
- } else {
- debuglog('Finding `Mac Developer` certificate for signing app in development for the Mac App Store signing...')
- promise = findIdentitiesAsync(opts, 'Mac Developer:')
- }
- } else {
- debuglog('Finding `Developer ID Application` certificate for distribution outside the Mac App Store...')
- promise = findIdentitiesAsync(opts, 'Developer ID Application:')
- }
- }
- return promise
- .then(function (identities) {
- if (identities.length > 0) {
- // Identity(/ies) found
- if (identities.length > 1) {
- debugwarn('Multiple identities found, will use the first discovered.')
- } else {
- debuglog('Found 1 identity.')
- }
- opts.identity = identities[0]
- } else {
- // No identity found
- return Promise.reject(new Error('No identity found for signing.'))
- }
- })
- })
- .then(function () {
- // Determine entitlements for code signing
- var filePath
- if (opts.platform === 'mas') {
- // To sign apps for Mac App Store, an entitlements file is required, especially for app sandboxing (as well some other services).
- // Fallback entitlements for sandboxing by default: Note this may cause troubles while running an signed app due to missing keys special to the project.
- // Further reading: https://developer.apple.com/library/mac/documentation/Miscellaneous/Reference/EntitlementKeyReference/Chapters/EnablingAppSandbox.html
- if (!opts.entitlements) {
- filePath = path.join(__dirname, 'default.entitlements.mas.plist')
- debugwarn('No `entitlements` passed in arguments:', '\n',
- '* Sandbox entitlements are required for Mac App Store distribution, your codesign entitlements file is default to:', filePath)
- opts.entitlements = filePath
- }
- if (!opts['entitlements-inherit']) {
- filePath = path.join(__dirname, 'default.entitlements.mas.inherit.plist')
- debugwarn('No `entitlements-inherit` passed in arguments:', '\n',
- '* Sandbox entitlements file for enclosing app files is default to:', filePath)
- opts['entitlements-inherit'] = filePath
- }
- } else {
- // Not necessary to have entitlements for non Mac App Store distribution
- if (!opts.entitlements) {
- debugwarn('No `entitlements` passed in arguments:', '\n',
- '* Provide `entitlements` to specify entitlements file for codesign.')
- } else {
- // If entitlements is provided as a flag, fallback to default
- if (opts.entitlements === true) {
- filePath = path.join(__dirname, 'default.entitlements.darwin.plist')
- debugwarn('`entitlements` not specified in arguments:', '\n',
- '* Provide `entitlements` to specify entitlements file for codesign.', '\n',
- '* Sandbox entitlements file for enclosing app files is default to:', filePath)
- opts.entitlements = filePath
- }
- if (!opts['entitlements-inherit']) {
- filePath = path.join(__dirname, 'default.entitlements.darwin.inherit.plist')
- debugwarn('No `entitlements-inherit` passed in arguments:', '\n',
- '* Sandbox entitlements file for enclosing app files is default to:', filePath)
- opts['entitlements-inherit'] = filePath
- }
- }
- }
- })
- .then(function () {
- // Pre-sign operations
- var preSignOperations = []
- if (opts['pre-embed-provisioning-profile'] === false) {
- debugwarn('Pre-sign operation disabled for provisioning profile embedding:', '\n',
- '* Enable by setting `pre-embed-provisioning-profile` to `true`.')
- } else {
- debuglog('Pre-sign operation enabled for provisioning profile:', '\n',
- '* Disable by setting `pre-embed-previsioning-profile` to `false`.')
- preSignOperations.push(preEmbedProvisioningProfile)
- }
- if (opts['pre-auto-entitlements'] === false) {
- debugwarn('Pre-sign operation disabled for entitlements automation.')
- } else {
- debuglog('Pre-sign operation enabled for entitlements automation with versions >= `1.1.1`:', '\n',
- '* Disable by setting `pre-auto-entitlements` to `false`.')
- if (opts.entitlements && (!opts.version || compareVersion(opts.version, '1.1.1') >= 0)) {
- // Enable Mac App Store sandboxing without using temporary-exception, introduced in Electron v1.1.1. Relates to electron#5601
- preSignOperations.push(preAutoEntitlements)
- }
- }
- return Promise.mapSeries(preSignOperations, function (preSignOperation) {
- return preSignOperation(opts)
- })
- })
- .then(function () {
- debuglog('Signing application...', '\n',
- '> Application:', opts.app, '\n',
- '> Platform:', opts.platform, '\n',
- '> Entitlements:', opts.entitlements, '\n',
- '> Child entitlements:', opts['entitlements-inherit'], '\n',
- '> Additional binaries:', opts.binaries, '\n',
- '> Identity:', opts.identity)
- return signApplicationAsync(opts)
- })
- .then(function () {
- // Post-sign operations
- debuglog('Application signed.')
- })
- }
- /**
- * This function is a normal callback implementation.
- * @function
- * @param {Object} opts - Options.
- * @param {RequestCallback} cb - Callback.
- */
- module.exports.sign = function (opts, cb) {
- signAsync(opts)
- .then(function () {
- debuglog('Application signed: ' + opts.app)
- if (cb) cb()
- })
- .catch(function (err) {
- debuglog('Sign failed:')
- if (err.message) debuglog(err.message)
- else if (err.stack) debuglog(err.stack)
- else debuglog(err)
- if (cb) cb(err)
- })
- }
|