- Integrate @sentry/electron and @sentry/react for crash reporting - Add Sentry Vite plugin for source maps - Add error tracking to main process, renderer, and IPC layer - Exclude source maps from packaged builds - Update README with new screenshots - Add Sentry opt-out toggle in settings - Update release workflow with Sentry config
143 lines
4.7 KiB
TypeScript
143 lines
4.7 KiB
TypeScript
import { defineConfig, externalizeDepsPlugin } from 'electron-vite'
|
|
import { sentryVitePlugin } from '@sentry/vite-plugin'
|
|
import react from '@vitejs/plugin-react'
|
|
import { readFileSync } from 'fs'
|
|
import { resolve } from 'path'
|
|
import type { Plugin } from 'vite'
|
|
|
|
// Read all production dependencies from package.json
|
|
// so they get bundled into the main process output.
|
|
// This avoids pnpm symlink issues with electron-builder's asar packaging.
|
|
const pkg = JSON.parse(readFileSync(resolve(__dirname, 'package.json'), 'utf-8'))
|
|
const prodDeps = Object.keys(pkg.dependencies || {})
|
|
|
|
// node-pty is a native addon that cannot be bundled by Rollup.
|
|
// It must remain external and be loaded at runtime via require().
|
|
const bundledDeps = prodDeps.filter(d => d !== 'node-pty' && d !== 'agent-teams-controller')
|
|
|
|
// Rollup plugin: stub out native .node addon imports with empty modules.
|
|
// ssh2 and cpu-features use optional native bindings that can't be bundled,
|
|
// but they have pure JS fallbacks when the native module isn't available.
|
|
function nativeModuleStub(): Plugin {
|
|
const STUB_ID = '\0native-stub'
|
|
return {
|
|
name: 'native-module-stub',
|
|
resolveId(source) {
|
|
if (source.endsWith('.node')) return STUB_ID
|
|
return null
|
|
},
|
|
load(id) {
|
|
if (id === STUB_ID) return 'export default {}'
|
|
return null
|
|
}
|
|
}
|
|
}
|
|
|
|
// Sentry source map upload — only active in CI when SENTRY_AUTH_TOKEN is set.
|
|
const sentryPlugins = process.env.SENTRY_AUTH_TOKEN
|
|
? [
|
|
sentryVitePlugin({
|
|
org: process.env.SENTRY_ORG ?? 'quant-jump-pro',
|
|
project: process.env.SENTRY_PROJECT ?? 'electron',
|
|
authToken: process.env.SENTRY_AUTH_TOKEN,
|
|
release: { name: `claude-agent-teams-ui@${pkg.version}` },
|
|
sourcemaps: {
|
|
filesToDeleteAfterUpload: ['./out/renderer/**/*.map', './dist-electron/**/*.map'],
|
|
},
|
|
}),
|
|
]
|
|
: []
|
|
|
|
export default defineConfig({
|
|
main: {
|
|
plugins: [
|
|
externalizeDepsPlugin({
|
|
exclude: bundledDeps
|
|
}),
|
|
nativeModuleStub(),
|
|
...sentryPlugins,
|
|
],
|
|
define: {
|
|
__APP_VERSION__: JSON.stringify(pkg.version),
|
|
// Inject DSN at compile time — process.env.SENTRY_DSN is NOT available
|
|
// at runtime in packaged Electron apps (only during CI build).
|
|
'process.env.SENTRY_DSN': JSON.stringify(process.env.SENTRY_DSN ?? ''),
|
|
},
|
|
resolve: {
|
|
alias: {
|
|
'@main': resolve(__dirname, 'src/main'),
|
|
'@shared': resolve(__dirname, 'src/shared'),
|
|
'@preload': resolve(__dirname, 'src/preload')
|
|
}
|
|
},
|
|
build: {
|
|
sourcemap: 'hidden',
|
|
outDir: 'dist-electron/main',
|
|
rollupOptions: {
|
|
input: {
|
|
index: resolve(__dirname, 'src/main/index.ts'),
|
|
'team-fs-worker': resolve(__dirname, 'src/main/workers/team-fs-worker.ts')
|
|
},
|
|
output: {
|
|
// CJS format so bundled deps can use __dirname/require.
|
|
// Use .cjs extension since package.json has "type": "module".
|
|
format: 'cjs',
|
|
entryFileNames: '[name].cjs',
|
|
// Set UV_THREADPOOL_SIZE before any module code runs.
|
|
// Must be in the banner because ESM→CJS hoists imports above top-level code.
|
|
// On Windows, fs.watch({recursive:true}) occupies a UV pool thread per watcher;
|
|
// with 3+ watchers + concurrent fs/DNS/spawn, the default 4 threads deadlock.
|
|
banner: `if(!process.env.UV_THREADPOOL_SIZE){process.env.UV_THREADPOOL_SIZE='24'}`
|
|
}
|
|
}
|
|
}
|
|
},
|
|
preload: {
|
|
plugins: [externalizeDepsPlugin()],
|
|
resolve: {
|
|
alias: {
|
|
'@preload': resolve(__dirname, 'src/preload'),
|
|
'@shared': resolve(__dirname, 'src/shared'),
|
|
'@main': resolve(__dirname, 'src/main')
|
|
}
|
|
},
|
|
build: {
|
|
outDir: 'dist-electron/preload',
|
|
rollupOptions: {
|
|
input: {
|
|
index: resolve(__dirname, 'src/preload/index.ts')
|
|
},
|
|
output: {
|
|
format: 'cjs',
|
|
entryFileNames: '[name].js'
|
|
}
|
|
}
|
|
}
|
|
},
|
|
renderer: {
|
|
optimizeDeps: {
|
|
include: ['@codemirror/language-data']
|
|
},
|
|
define: {
|
|
__APP_VERSION__: JSON.stringify(pkg.version),
|
|
// Pass SENTRY_DSN to renderer as VITE_SENTRY_DSN (Vite replaces at compile time)
|
|
'import.meta.env.VITE_SENTRY_DSN': JSON.stringify(process.env.SENTRY_DSN ?? ''),
|
|
},
|
|
resolve: {
|
|
alias: {
|
|
'@renderer': resolve(__dirname, 'src/renderer'),
|
|
'@shared': resolve(__dirname, 'src/shared'),
|
|
'@main': resolve(__dirname, 'src/main')
|
|
}
|
|
},
|
|
plugins: [react(), ...sentryPlugins],
|
|
build: {
|
|
sourcemap: 'hidden',
|
|
rollupOptions: {
|
|
input: {
|
|
index: resolve(__dirname, 'src/renderer/index.html')
|
|
}
|
|
}
|
|
}
|
|
}
|
|
})
|