* fix: set cortex data folder path when starting jan * fix: change port to 1338 * fix: add migration in advanced setting * update * update new cortex * feat: add import model error handler Signed-off-by: James <namnh0122@gmail.com> --------- Signed-off-by: James <namnh0122@gmail.com>
281 lines
6.8 KiB
TypeScript
281 lines
6.8 KiB
TypeScript
import { app, BrowserWindow } from 'electron'
|
|
|
|
import { join, resolve } from 'path'
|
|
import { exec, execSync, ChildProcess } from 'child_process'
|
|
import { cortexPath } from './cortex-runner'
|
|
|
|
/**
|
|
* Managers
|
|
**/
|
|
import { windowManager } from './managers/window'
|
|
|
|
/**
|
|
* IPC Handlers
|
|
**/
|
|
import { handleAppUpdates } from './handlers/update'
|
|
import { handleAppIPCs } from './handlers/native'
|
|
|
|
/**
|
|
* Utils
|
|
**/
|
|
import { setupMenu } from './utils/menu'
|
|
import { createUserSpace, getAppConfigurations } from './utils/path'
|
|
import { migrate } from './utils/migration'
|
|
import { cleanUpAndQuit } from './utils/clean'
|
|
import { setupCore } from './utils/setup'
|
|
import { setupReactDevTool } from './utils/dev'
|
|
|
|
import log from 'electron-log'
|
|
|
|
const preloadPath = join(__dirname, 'preload.js')
|
|
const rendererPath = join(__dirname, '..', 'renderer')
|
|
const mainPath = join(rendererPath, 'index.html')
|
|
|
|
const mainUrl = 'http://localhost:3000'
|
|
|
|
const gotTheLock = app.requestSingleInstanceLock()
|
|
|
|
if (process.defaultApp) {
|
|
if (process.argv.length >= 2) {
|
|
app.setAsDefaultProtocolClient('jan', process.execPath, [
|
|
resolve(process.argv[1]),
|
|
])
|
|
}
|
|
} else {
|
|
app.setAsDefaultProtocolClient('jan')
|
|
}
|
|
|
|
const createMainWindow = () => {
|
|
const startUrl = app.isPackaged ? `file://${mainPath}` : mainUrl
|
|
windowManager.createMainWindow(preloadPath, startUrl)
|
|
}
|
|
|
|
log.initialize()
|
|
log.info('Log from the main process')
|
|
|
|
// replace all console.log to log
|
|
Object.assign(console, log.functions)
|
|
|
|
let cortexService: ChildProcess | undefined = undefined
|
|
|
|
const cortexJsPort = 1338
|
|
const cortexCppPort = 3940
|
|
const host = '127.0.0.1'
|
|
|
|
app
|
|
.whenReady()
|
|
.then(() => killProcessesOnPort(cortexCppPort))
|
|
.then(() => killProcessesOnPort(cortexJsPort))
|
|
.then(() => {
|
|
const appConfiguration = getAppConfigurations()
|
|
const janDataFolder = appConfiguration.data_folder
|
|
|
|
const cortexParams: Record<string, string> = {
|
|
'-n': 'jan',
|
|
'-a': host,
|
|
'-p': cortexJsPort.toString(),
|
|
'-ep': cortexCppPort.toString(),
|
|
'--dataFolder': janDataFolder,
|
|
}
|
|
|
|
// add cortex parameters to the command
|
|
const command = Object.entries(cortexParams).reduce(
|
|
(acc, [key, value]) => `${acc} ${key} ${value}`,
|
|
`${cortexPath}`
|
|
)
|
|
|
|
log.info('Starting cortex with command:', command)
|
|
// init cortex
|
|
cortexService = exec(`${command}`, (error, stdout, stderr) => {
|
|
if (error) {
|
|
log.error(`error: ${error.message}`)
|
|
return
|
|
}
|
|
if (stderr) {
|
|
log.error(`stderr: ${stderr}`)
|
|
return
|
|
}
|
|
log.info(`stdout: ${stdout}`)
|
|
})
|
|
})
|
|
.then(() => {
|
|
if (!gotTheLock) {
|
|
app.quit()
|
|
throw new Error('Another instance of the app is already running')
|
|
} else {
|
|
app.on(
|
|
'second-instance',
|
|
(_event, commandLine, _workingDirectory): void => {
|
|
if (process.platform === 'win32' || process.platform === 'linux') {
|
|
// this is for handling deeplink on windows and linux
|
|
// since those OS will emit second-instance instead of open-url
|
|
const url = commandLine.pop()
|
|
if (url) {
|
|
windowManager.sendMainAppDeepLink(url)
|
|
}
|
|
}
|
|
windowManager.showMainWindow()
|
|
}
|
|
)
|
|
}
|
|
})
|
|
.then(setupCore)
|
|
.then(createUserSpace)
|
|
.then(migrate)
|
|
.then(setupMenu)
|
|
.then(handleIPCs)
|
|
.then(handleAppUpdates)
|
|
.then(createMainWindow)
|
|
.then(() => {
|
|
if (!app.isPackaged) {
|
|
setupReactDevTool()
|
|
windowManager.mainWindow?.webContents.openDevTools()
|
|
}
|
|
})
|
|
.then(() => {
|
|
app.on('activate', () => {
|
|
if (!BrowserWindow.getAllWindows().length) {
|
|
createMainWindow()
|
|
} else {
|
|
windowManager.showMainWindow()
|
|
}
|
|
})
|
|
})
|
|
|
|
app.on('open-url', (_event, url) => {
|
|
windowManager.sendMainAppDeepLink(url)
|
|
})
|
|
|
|
app.once('quit', async () => {
|
|
cleanUpAndQuit()
|
|
})
|
|
|
|
app.once('window-all-closed', async () => {
|
|
await stopApiServer()
|
|
await stopCortexService()
|
|
cleanUpAndQuit()
|
|
})
|
|
|
|
async function stopCortexService() {
|
|
try {
|
|
const pid = cortexService?.pid
|
|
if (!pid) {
|
|
console.log('No cortex service to stop.')
|
|
return
|
|
}
|
|
process.kill(pid)
|
|
console.log(`Service with PID ${pid} has been terminated.`)
|
|
} catch (error) {
|
|
console.error('Error killing service:', error)
|
|
}
|
|
}
|
|
|
|
async function stopApiServer() {
|
|
// this function is not meant to be success. It will throw an error.
|
|
try {
|
|
await fetch(`http://${host}:${cortexJsPort}/v1/system`, {
|
|
method: 'DELETE',
|
|
})
|
|
} catch (error) {
|
|
// do nothing
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Handles various IPC messages from the renderer process.
|
|
*/
|
|
function handleIPCs() {
|
|
// Inject core handlers for IPCs
|
|
// Handle native IPCs
|
|
handleAppIPCs()
|
|
}
|
|
|
|
function killProcessesOnPort(port: number): void {
|
|
try {
|
|
console.log(`Killing processes on port ${port}...`)
|
|
if (process.platform === 'win32') {
|
|
killProcessesOnWindowsPort(port)
|
|
} else {
|
|
killProcessesOnUnixPort(port)
|
|
}
|
|
} catch (error) {
|
|
console.error(
|
|
`Failed to kill process(es) on port ${port}: ${(error as Error).message}`
|
|
)
|
|
}
|
|
}
|
|
|
|
function killProcessesOnWindowsPort(port: number): void {
|
|
let result: string
|
|
try {
|
|
result = execSync(`netstat -ano | findstr :${port}`).toString()
|
|
} catch (error) {
|
|
console.log(`No processes found on port ${port}.`)
|
|
return
|
|
}
|
|
|
|
const lines = result.split('\n').filter(Boolean)
|
|
|
|
if (lines.length === 0) {
|
|
console.log(`No processes found on port ${port}.`)
|
|
return
|
|
}
|
|
|
|
const pids = lines
|
|
.map((line) => {
|
|
const parts = line.trim().split(/\s+/)
|
|
return parts[parts.length - 1]
|
|
})
|
|
.filter((pid): pid is string => Boolean(pid) && !isNaN(Number(pid)))
|
|
|
|
if (pids.length === 0) {
|
|
console.log(`No valid PIDs found for port ${port}.`)
|
|
return
|
|
}
|
|
const uniquePids = Array.from(new Set(pids))
|
|
console.log('uniquePids', uniquePids)
|
|
|
|
uniquePids.forEach((pid) => {
|
|
try {
|
|
execSync(`taskkill /PID ${pid} /F`)
|
|
console.log(
|
|
`Process with PID ${pid} on port ${port} has been terminated.`
|
|
)
|
|
} catch (error) {
|
|
console.error(
|
|
`Failed to kill process with PID ${pid}: ${(error as Error).message}`
|
|
)
|
|
}
|
|
})
|
|
}
|
|
|
|
function killProcessesOnUnixPort(port: number): void {
|
|
let pids: string[]
|
|
|
|
try {
|
|
pids = execSync(`lsof -ti tcp:${port}`)
|
|
.toString()
|
|
.trim()
|
|
.split('\n')
|
|
.filter(Boolean)
|
|
} catch (error) {
|
|
if ((error as { status?: number }).status === 1) {
|
|
console.log(`No processes found on port ${port}.`)
|
|
return
|
|
}
|
|
throw error // Re-throw if it's not the "no processes found" error
|
|
}
|
|
|
|
pids.forEach((pid) => {
|
|
process.kill(parseInt(pid), 'SIGTERM')
|
|
console.log(`Process with PID ${pid} on port ${port} has been terminated.`)
|
|
})
|
|
}
|
|
|
|
/**
|
|
* Suppress Node error messages
|
|
*/
|
|
process.on('uncaughtException', function (err) {
|
|
log.error(`Error: ${err}`)
|
|
})
|