mirror of
https://github.com/NousResearch/hermes-agent.git
synced 2026-05-21 03:39:54 +00:00
5ec0667fb3
Add GitHub Actions release channels for signed desktop installers and document the stable/nightly download paths.
101 lines
3.1 KiB
JavaScript
101 lines
3.1 KiB
JavaScript
const fs = require('node:fs')
|
|
const os = require('node:os')
|
|
const path = require('node:path')
|
|
const { execFile } = require('node:child_process')
|
|
|
|
function run(command, args) {
|
|
return new Promise((resolve, reject) => {
|
|
execFile(command, args, (error, stdout, stderr) => {
|
|
if (error) {
|
|
reject(
|
|
new Error(
|
|
`${command} ${args.join(' ')} failed: ${stderr?.trim() || stdout?.trim() || error.message}`
|
|
)
|
|
)
|
|
return
|
|
}
|
|
resolve({ stdout, stderr })
|
|
})
|
|
})
|
|
}
|
|
|
|
function inlineKeyLooksValid(value) {
|
|
return value.includes('BEGIN PRIVATE KEY') && value.includes('END PRIVATE KEY')
|
|
}
|
|
|
|
function resolveApiKeyPath(rawValue) {
|
|
const value = String(rawValue || '').trim()
|
|
if (!value) return { keyPath: '', cleanup: () => {} }
|
|
|
|
if (fs.existsSync(value)) {
|
|
return { keyPath: value, cleanup: () => {} }
|
|
}
|
|
|
|
if (!inlineKeyLooksValid(value)) {
|
|
throw new Error('APPLE_API_KEY must be a file path or inline .p8 key content')
|
|
}
|
|
|
|
const tempPath = path.join(os.tmpdir(), `hermes-notary-${Date.now()}-${process.pid}.p8`)
|
|
fs.writeFileSync(tempPath, value, 'utf8')
|
|
return {
|
|
keyPath: tempPath,
|
|
cleanup: () => {
|
|
try {
|
|
fs.rmSync(tempPath, { force: true })
|
|
} catch {
|
|
// Best-effort cleanup.
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
exports.default = async function notarize(context) {
|
|
const { electronPlatformName, appOutDir, packager } = context
|
|
if (electronPlatformName !== 'darwin') return
|
|
|
|
const appName = packager.appInfo.productFilename
|
|
const appPath = path.join(appOutDir, `${appName}.app`)
|
|
if (!fs.existsSync(appPath)) {
|
|
throw new Error(`Cannot notarize missing app bundle: ${appPath}`)
|
|
}
|
|
|
|
const profile = String(process.env.APPLE_NOTARY_PROFILE || '').trim()
|
|
if (profile) {
|
|
const zipPath = path.join(appOutDir, `${appName}.zip`)
|
|
await run('ditto', ['-c', '-k', '--sequesterRsrc', '--keepParent', appPath, zipPath])
|
|
await run('xcrun', ['notarytool', 'submit', zipPath, '--keychain-profile', profile, '--wait'])
|
|
await run('xcrun', ['stapler', 'staple', '-v', appPath])
|
|
try {
|
|
fs.rmSync(zipPath, { force: true })
|
|
} catch {
|
|
// Best-effort cleanup.
|
|
}
|
|
return
|
|
}
|
|
|
|
const keyId = String(process.env.APPLE_API_KEY_ID || '').trim()
|
|
const issuer = String(process.env.APPLE_API_ISSUER || '').trim()
|
|
const rawApiKey = process.env.APPLE_API_KEY
|
|
if (!rawApiKey || !keyId || !issuer) {
|
|
console.log(
|
|
'Skipping notarization: APPLE_API_KEY, APPLE_API_KEY_ID, and APPLE_API_ISSUER are not fully configured.'
|
|
)
|
|
return
|
|
}
|
|
|
|
const { keyPath, cleanup } = resolveApiKeyPath(rawApiKey)
|
|
const zipPath = path.join(appOutDir, `${appName}.zip`)
|
|
try {
|
|
await run('ditto', ['-c', '-k', '--sequesterRsrc', '--keepParent', appPath, zipPath])
|
|
await run('xcrun', ['notarytool', 'submit', zipPath, '--key', keyPath, '--key-id', keyId, '--issuer', issuer, '--wait'])
|
|
await run('xcrun', ['stapler', 'staple', '-v', appPath])
|
|
} finally {
|
|
try {
|
|
fs.rmSync(zipPath, { force: true })
|
|
} catch {
|
|
// Best-effort cleanup.
|
|
}
|
|
cleanup()
|
|
}
|
|
}
|