mirror of
https://github.com/lobehub/lobehub.git
synced 2026-03-26 13:19:34 +07:00
- Use same app name (LobeHub) and icon as stable for canary builds - Add build channel tag in Settings > About for non-stable channels (Canary, Nightly, Beta) - Add getBuildChannel IPC to expose build-time channel for display Made-with: Cursor
307 lines
9.7 KiB
JavaScript
307 lines
9.7 KiB
JavaScript
import { execSync } from 'node:child_process';
|
|
import fs from 'node:fs/promises';
|
|
import os from 'node:os';
|
|
import path from 'node:path';
|
|
import { fileURLToPath } from 'node:url';
|
|
|
|
import dotenv from 'dotenv';
|
|
|
|
import {
|
|
copyNativeModules,
|
|
copyNativeModulesToSource,
|
|
getAsarUnpackPatterns,
|
|
getNativeModulesFilesConfig,
|
|
} from './native-deps.config.mjs';
|
|
|
|
dotenv.config();
|
|
|
|
const __dirname = path.dirname(fileURLToPath(import.meta.url));
|
|
|
|
const packageJSON = JSON.parse(await fs.readFile(path.join(__dirname, 'package.json'), 'utf8'));
|
|
|
|
const channel = process.env.UPDATE_CHANNEL;
|
|
const arch = os.arch();
|
|
const hasAppleCertificate = Boolean(process.env.CSC_LINK);
|
|
|
|
// 自定义更新服务器 URL (用于 stable 频道)
|
|
const updateServerUrl = process.env.UPDATE_SERVER_URL;
|
|
|
|
console.info(`🚄 Build Version ${packageJSON.version}, Channel: ${channel}`);
|
|
console.info(`🏗️ Building for architecture: ${arch}`);
|
|
|
|
// Channel identity derived solely from UPDATE_CHANNEL env var.
|
|
// Supported channels: stable, nightly, canary
|
|
const isStable = !channel || channel === 'stable';
|
|
const isNightly = channel === 'nightly';
|
|
const isCanary = channel === 'canary';
|
|
|
|
// Strip trailing channel path from URL for re-appending the correct channel
|
|
// Handles both base URL (https://cdn.example.com) and legacy URL with channel (https://cdn.example.com/stable)
|
|
const stripChannelSuffix = (url) => url.replace(/\/(stable|nightly|canary|beta)\/?$/, '');
|
|
|
|
// 根据 channel 配置 publish provider
|
|
// - 所有渠道 + UPDATE_SERVER_URL: 使用 generic (S3)
|
|
// - 无 UPDATE_SERVER_URL: 回退到 GitHub (本地开发)
|
|
const getPublishConfig = () => {
|
|
const channelPath = isStable ? 'stable' : isNightly ? 'nightly' : channel || 'stable';
|
|
|
|
if (updateServerUrl) {
|
|
const baseUrl = stripChannelSuffix(updateServerUrl);
|
|
const fullUrl = `${baseUrl}/${channelPath}`;
|
|
console.info(`📦 ${channelPath} channel: Using generic provider (${fullUrl})`);
|
|
return [
|
|
{
|
|
provider: 'generic',
|
|
url: fullUrl,
|
|
},
|
|
];
|
|
}
|
|
|
|
// 本地开发无 S3 时回退到 GitHub
|
|
console.info(`📦 ${channelPath} channel: No UPDATE_SERVER_URL, falling back to GitHub provider`);
|
|
return [
|
|
{
|
|
owner: 'lobehub',
|
|
provider: 'github',
|
|
repo: 'lobehub',
|
|
},
|
|
];
|
|
};
|
|
|
|
// Keep only these Electron Framework localization folders (*.lproj)
|
|
// (aligned with previous Electron Forge build config)
|
|
const keepLanguages = new Set(['en', 'en_GB', 'en-US', 'en_US']);
|
|
|
|
// https://www.electron.build/code-signing-mac#how-to-disable-code-signing-during-the-build-process-on-macos
|
|
if (!hasAppleCertificate) {
|
|
// Disable auto discovery to keep electron-builder from searching unavailable signing identities
|
|
process.env.CSC_IDENTITY_AUTO_DISCOVERY = 'false';
|
|
console.info('⚠️ Apple certificate link not found, macOS artifacts will be unsigned.');
|
|
}
|
|
|
|
// 根据版本类型确定协议 scheme
|
|
const getProtocolScheme = () => {
|
|
if (isCanary) return 'lobehub-canary';
|
|
if (isNightly) return 'lobehub-nightly';
|
|
return 'lobehub';
|
|
};
|
|
|
|
const protocolScheme = getProtocolScheme();
|
|
|
|
// Determine icon file based on version type
|
|
const getIconFileName = () => {
|
|
if (isStable || isCanary) return 'Icon';
|
|
// nightly uses pre-release icon
|
|
return 'Icon-nightly';
|
|
};
|
|
|
|
/**
|
|
* @type {import('electron-builder').Configuration}
|
|
* @see https://www.electron.build/configuration
|
|
*/
|
|
const config = {
|
|
/**
|
|
* BeforePack hook to resolve pnpm symlinks for native modules.
|
|
* This ensures native modules are properly included in the asar archive.
|
|
*/
|
|
beforePack: async () => {
|
|
await copyNativeModulesToSource();
|
|
|
|
console.info('📦 Downloading agent-browser binary...');
|
|
execSync('node scripts/download-agent-browser.mjs', { stdio: 'inherit', cwd: __dirname });
|
|
},
|
|
/**
|
|
* AfterPack hook for post-processing:
|
|
* 1. Copy native modules to asar.unpacked (resolving pnpm symlinks)
|
|
* 2. Copy Liquid Glass Assets.car for macOS 26+
|
|
* 3. Remove unused Electron Framework localizations
|
|
*
|
|
* @see https://github.com/electron-userland/electron-builder/issues/9254
|
|
* @see https://github.com/MultiboxLabs/flow-browser/pull/159
|
|
* @see https://github.com/electron/packager/pull/1806
|
|
*/
|
|
afterPack: async (context) => {
|
|
const isMac = ['darwin', 'mas'].includes(context.electronPlatformName);
|
|
|
|
// Determine resources path based on platform
|
|
let resourcesPath;
|
|
if (isMac) {
|
|
resourcesPath = path.join(
|
|
context.appOutDir,
|
|
`${context.packager.appInfo.productFilename}.app`,
|
|
'Contents',
|
|
'Resources',
|
|
);
|
|
} else {
|
|
// Windows and Linux: resources is directly in appOutDir
|
|
resourcesPath = path.join(context.appOutDir, 'resources');
|
|
}
|
|
|
|
// Copy native modules to asar.unpacked, resolving pnpm symlinks
|
|
const unpackedNodeModules = path.join(resourcesPath, 'app.asar.unpacked', 'node_modules');
|
|
await copyNativeModules(unpackedNodeModules);
|
|
|
|
// macOS-specific post-processing
|
|
if (!isMac) {
|
|
return;
|
|
}
|
|
|
|
const iconFileName = getIconFileName();
|
|
const assetsCarSource = path.join(__dirname, 'build', `${iconFileName}.Assets.car`);
|
|
const assetsCarDest = path.join(resourcesPath, 'Assets.car');
|
|
|
|
// Remove unused Electron Framework localizations to reduce app size
|
|
const frameworkResourcePath = path.join(
|
|
context.appOutDir,
|
|
`${context.packager.appInfo.productFilename}.app`,
|
|
'Contents',
|
|
'Frameworks',
|
|
'Electron Framework.framework',
|
|
'Versions',
|
|
'A',
|
|
'Resources',
|
|
);
|
|
|
|
try {
|
|
const entries = await fs.readdir(frameworkResourcePath);
|
|
await Promise.all(
|
|
entries.map(async (file) => {
|
|
if (!file.endsWith('.lproj')) return;
|
|
|
|
const lang = file.split('.')[0];
|
|
if (keepLanguages.has(lang)) return;
|
|
|
|
await fs.rm(path.join(frameworkResourcePath, file), { force: true, recursive: true });
|
|
}),
|
|
);
|
|
} catch {
|
|
// Non-critical: folder may not exist depending on packaging details
|
|
}
|
|
|
|
try {
|
|
await fs.access(assetsCarSource);
|
|
await fs.copyFile(assetsCarSource, assetsCarDest);
|
|
console.info(`✅ Copied Liquid Glass icon: ${iconFileName}.Assets.car`);
|
|
} catch {
|
|
// Non-critical: Assets.car not found or copy failed
|
|
// App will use fallback .icns icon on all macOS versions
|
|
console.info(`⏭️ Skipping Assets.car (not found or copy failed)`);
|
|
}
|
|
},
|
|
appId: 'com.lobehub.lobehub-desktop',
|
|
appImage: {
|
|
artifactName: '${productName}-${version}.${ext}',
|
|
},
|
|
|
|
// Native modules must be unpacked from asar to work correctly
|
|
asarUnpack: getAsarUnpackPatterns(),
|
|
|
|
detectUpdateChannel: true,
|
|
|
|
directories: {
|
|
buildResources: 'build',
|
|
output: 'release',
|
|
},
|
|
|
|
dmg: {
|
|
artifactName: '${productName}-${version}-${arch}.${ext}',
|
|
background: 'resources/dmg.png',
|
|
contents: [
|
|
{ type: 'file', x: 150, y: 240 },
|
|
{ type: 'link', path: '/Applications', x: 450, y: 240 },
|
|
],
|
|
iconSize: 80,
|
|
window: {
|
|
height: 400,
|
|
width: 600,
|
|
},
|
|
},
|
|
|
|
electronDownload: {
|
|
mirror: 'https://npmmirror.com/mirrors/electron/',
|
|
},
|
|
|
|
files: [
|
|
'dist',
|
|
'resources',
|
|
'dist/renderer/**/*',
|
|
'!resources/locales',
|
|
'!resources/dmg.png',
|
|
// Exclude all node_modules first
|
|
'!node_modules',
|
|
// Then explicitly include native modules using object form (handles pnpm symlinks)
|
|
...getNativeModulesFilesConfig(),
|
|
],
|
|
generateUpdatesFilesForAllChannels: true,
|
|
linux: {
|
|
category: 'Utility',
|
|
maintainer: 'electronjs.org',
|
|
target: ['AppImage', 'snap', 'deb', 'rpm', 'tar.gz'],
|
|
},
|
|
mac: {
|
|
compression: 'maximum',
|
|
entitlementsInherit: 'build/entitlements.mac.plist',
|
|
extendInfo: {
|
|
CFBundleIconName: 'AppIcon',
|
|
CFBundleURLTypes: [
|
|
{
|
|
CFBundleURLName: 'LobeHub Protocol',
|
|
CFBundleURLSchemes: [protocolScheme],
|
|
},
|
|
],
|
|
NSAppleEventsUsageDescription:
|
|
'Application needs to control System Settings to help you grant Full Disk Access automatically.',
|
|
NSCameraUsageDescription: "Application requests access to the device's camera.",
|
|
NSDocumentsFolderUsageDescription:
|
|
"Application requests access to the user's Documents folder.",
|
|
NSDownloadsFolderUsageDescription:
|
|
"Application requests access to the user's Downloads folder.",
|
|
NSMicrophoneUsageDescription: "Application requests access to the device's microphone.",
|
|
NSScreenCaptureUsageDescription:
|
|
'Application requests access to record and analyze screen content for AI assistance.',
|
|
},
|
|
gatekeeperAssess: false,
|
|
hardenedRuntime: hasAppleCertificate,
|
|
notarize: hasAppleCertificate,
|
|
...(hasAppleCertificate ? {} : { identity: null }),
|
|
target: [
|
|
{ arch: [arch === 'arm64' ? 'arm64' : 'x64'], target: 'dmg' },
|
|
{ arch: [arch === 'arm64' ? 'arm64' : 'x64'], target: 'zip' },
|
|
],
|
|
},
|
|
npmRebuild: true,
|
|
nsis: {
|
|
allowToChangeInstallationDirectory: true,
|
|
artifactName: '${productName}-${version}-setup.${ext}',
|
|
createDesktopShortcut: 'always',
|
|
installerHeader: './build/nsis-header.bmp',
|
|
installerSidebar: './build/nsis-sidebar.bmp',
|
|
oneClick: false,
|
|
shortcutName: '${productName}',
|
|
uninstallDisplayName: '${productName}',
|
|
uninstallerSidebar: './build/nsis-sidebar.bmp',
|
|
},
|
|
protocols: [
|
|
{
|
|
name: 'LobeHub Protocol',
|
|
schemes: [protocolScheme],
|
|
},
|
|
],
|
|
publish: getPublishConfig(),
|
|
|
|
// Release notes 配置
|
|
// 可以通过环境变量 RELEASE_NOTES 传入,或从文件读取
|
|
// 这会被写入 latest-mac.yml / latest.yml 中,供 generic provider 使用
|
|
releaseInfo: {
|
|
releaseNotes: process.env.RELEASE_NOTES || undefined,
|
|
},
|
|
|
|
extraResources: [{ from: 'resources/bin', to: 'bin' }],
|
|
|
|
win: {
|
|
executableName: 'LobeHub',
|
|
},
|
|
};
|
|
|
|
export default config;
|