2023-07-06 16:02:13 +00:00
|
|
|
import { spawn, exec } from 'child_process'
|
2023-07-07 13:54:59 +00:00
|
|
|
import { app, autoUpdater, dialog, Tray, Menu } from 'electron'
|
2023-06-23 22:38:22 +00:00
|
|
|
import * as path from 'path'
|
2023-07-06 16:23:08 +00:00
|
|
|
import * as fs from 'fs'
|
2023-06-27 16:35:51 +00:00
|
|
|
|
2023-07-06 21:32:48 +00:00
|
|
|
import { analytics, id } from './telemetry'
|
|
|
|
|
2023-06-27 16:35:51 +00:00
|
|
|
require('@electron/remote/main').initialize()
|
|
|
|
|
2023-07-05 20:10:30 +00:00
|
|
|
let tray: Tray | null = null
|
2023-07-06 22:05:31 +00:00
|
|
|
|
2023-07-06 20:59:36 +00:00
|
|
|
const SingleInstanceLock = app.requestSingleInstanceLock()
|
2023-07-06 22:05:31 +00:00
|
|
|
if (!SingleInstanceLock) {
|
|
|
|
app.quit()
|
|
|
|
}
|
2023-07-06 18:41:25 +00:00
|
|
|
|
2023-07-05 20:10:30 +00:00
|
|
|
const createSystemtray = () => {
|
2023-07-07 13:54:59 +00:00
|
|
|
let iconPath = path.join(__dirname, '..', '..', 'assets', 'ollama_icon_16x16Template.png')
|
2023-07-06 16:45:58 +00:00
|
|
|
|
|
|
|
if (app.isPackaged) {
|
2023-07-07 13:54:59 +00:00
|
|
|
iconPath = path.join(process.resourcesPath, 'ollama_icon_16x16Template.png')
|
2023-07-06 16:45:58 +00:00
|
|
|
}
|
2023-06-22 16:45:31 +00:00
|
|
|
|
2023-07-07 13:54:59 +00:00
|
|
|
tray = new Tray(iconPath)
|
2023-06-22 16:45:31 +00:00
|
|
|
|
2023-07-06 22:02:37 +00:00
|
|
|
const contextMenu = Menu.buildFromTemplate([{ role: 'quit', label: 'Quit Ollama', accelerator: 'Command+Q' }])
|
2023-06-27 16:35:51 +00:00
|
|
|
|
2023-07-05 20:10:30 +00:00
|
|
|
tray.setContextMenu(contextMenu)
|
|
|
|
tray.setToolTip('Ollama')
|
|
|
|
}
|
|
|
|
|
|
|
|
// Handle creating/removing shortcuts on Windows when installing/uninstalling.
|
|
|
|
if (require('electron-squirrel-startup')) {
|
|
|
|
app.quit()
|
2023-06-25 04:30:02 +00:00
|
|
|
}
|
|
|
|
|
2023-07-06 16:17:42 +00:00
|
|
|
const ollama = path.join(process.resourcesPath, 'ollama')
|
|
|
|
|
2023-06-25 04:30:02 +00:00
|
|
|
// if the app is packaged then run the server
|
|
|
|
if (app.isPackaged) {
|
2023-06-23 22:38:22 +00:00
|
|
|
// Start the executable
|
2023-07-06 16:02:13 +00:00
|
|
|
console.log(`Starting server`)
|
|
|
|
const proc = spawn(ollama, ['serve'])
|
2023-06-25 04:30:02 +00:00
|
|
|
proc.stdout.on('data', data => {
|
2023-06-23 22:38:22 +00:00
|
|
|
console.log(`server: ${data}`)
|
|
|
|
})
|
2023-06-25 04:30:02 +00:00
|
|
|
proc.stderr.on('data', data => {
|
2023-06-23 22:38:22 +00:00
|
|
|
console.error(`server: ${data}`)
|
|
|
|
})
|
|
|
|
|
2023-06-25 04:30:02 +00:00
|
|
|
process.on('exit', () => {
|
|
|
|
proc.kill()
|
|
|
|
})
|
2023-06-22 16:45:31 +00:00
|
|
|
}
|
|
|
|
|
2023-07-06 18:32:48 +00:00
|
|
|
function server() {
|
|
|
|
const binary = app.isPackaged
|
|
|
|
? path.join(process.resourcesPath, 'ollama')
|
|
|
|
: path.resolve(__dirname, '..', '..', 'ollama')
|
|
|
|
|
|
|
|
console.log(`Starting server`)
|
|
|
|
const proc = spawn(binary, ['serve'])
|
|
|
|
proc.stdout.on('data', data => {
|
|
|
|
console.log(`server: ${data}`)
|
|
|
|
})
|
|
|
|
proc.stderr.on('data', data => {
|
|
|
|
console.error(`server: ${data}`)
|
|
|
|
})
|
|
|
|
|
|
|
|
process.on('exit', () => {
|
|
|
|
proc.kill()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-07-06 16:02:13 +00:00
|
|
|
function installCLI() {
|
2023-07-06 16:23:08 +00:00
|
|
|
const symlinkPath = '/usr/local/bin/ollama'
|
|
|
|
|
|
|
|
if (fs.existsSync(symlinkPath) && fs.readlinkSync(symlinkPath) === ollama) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2023-07-06 16:02:13 +00:00
|
|
|
dialog
|
|
|
|
.showMessageBox({
|
|
|
|
type: 'info',
|
|
|
|
title: 'Ollama CLI installation',
|
2023-07-07 17:44:36 +00:00
|
|
|
message: 'To make the Ollama command work in your terminal, it needs administrator privileges.',
|
2023-07-06 16:02:13 +00:00
|
|
|
buttons: ['OK'],
|
|
|
|
})
|
|
|
|
.then(result => {
|
|
|
|
if (result.response === 0) {
|
2023-07-06 20:08:53 +00:00
|
|
|
const command = `
|
2023-07-06 16:23:08 +00:00
|
|
|
do shell script "ln -F -s ${ollama} /usr/local/bin/ollama" with administrator privileges
|
|
|
|
`
|
2023-07-06 16:02:13 +00:00
|
|
|
exec(`osascript -e '${command}'`, (error: Error | null, stdout: string, stderr: string) => {
|
|
|
|
if (error) {
|
|
|
|
console.error(`exec error: ${error}`)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
console.log(`stdout: ${stdout}`)
|
|
|
|
console.error(`stderr: ${stderr}`)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
2023-07-06 22:05:31 +00:00
|
|
|
app.on('ready', () => {
|
|
|
|
if (process.platform === 'darwin') {
|
|
|
|
app.dock.hide()
|
|
|
|
|
|
|
|
if (!app.isInApplicationsFolder()) {
|
|
|
|
const chosen = dialog.showMessageBoxSync({
|
|
|
|
type: 'question',
|
|
|
|
buttons: ['Move to Applications', 'Do Not Move'],
|
|
|
|
message: 'Ollama works best when run from the Applications directory.',
|
|
|
|
defaultId: 0,
|
|
|
|
cancelId: 1,
|
|
|
|
})
|
|
|
|
|
|
|
|
if (chosen === 0) {
|
|
|
|
try {
|
|
|
|
app.moveToApplicationsFolder({
|
|
|
|
conflictHandler: conflictType => {
|
|
|
|
if (conflictType === 'existsAndRunning') {
|
|
|
|
dialog.showMessageBoxSync({
|
|
|
|
type: 'info',
|
|
|
|
message: 'Cannot move to Applications directory',
|
|
|
|
detail:
|
|
|
|
'Another version of Ollama is currently running from your Applications directory. Close it first and try again.',
|
|
|
|
})
|
|
|
|
}
|
|
|
|
return true
|
|
|
|
},
|
|
|
|
})
|
|
|
|
return
|
|
|
|
} catch (e) {
|
|
|
|
console.error('Failed to move to applications folder')
|
|
|
|
console.error(e)
|
2023-07-06 20:08:53 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2023-07-06 22:05:31 +00:00
|
|
|
}
|
2023-07-06 22:02:37 +00:00
|
|
|
|
2023-07-06 22:05:31 +00:00
|
|
|
createSystemtray()
|
2023-07-06 22:02:37 +00:00
|
|
|
|
2023-07-06 22:05:31 +00:00
|
|
|
if (app.isPackaged) {
|
|
|
|
installCLI()
|
|
|
|
}
|
|
|
|
})
|
2023-07-06 18:32:48 +00:00
|
|
|
|
2023-06-22 16:45:31 +00:00
|
|
|
// Quit when all windows are closed, except on macOS. There, it's common
|
|
|
|
// for applications and their menu bar to stay active until the user quits
|
|
|
|
// explicitly with Cmd + Q.
|
|
|
|
app.on('window-all-closed', () => {
|
|
|
|
if (process.platform !== 'darwin') {
|
|
|
|
app.quit()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// In this file you can include the rest of your app's specific main process
|
|
|
|
// code. You can also put them in separate files and import them here.
|
2023-07-06 04:04:06 +00:00
|
|
|
autoUpdater.setFeedURL({
|
|
|
|
url: `https://ollama.ai/api/update?os=${process.platform}&arch=${process.arch}&version=${app.getVersion()}`,
|
|
|
|
})
|
2023-06-27 21:50:50 +00:00
|
|
|
|
2023-07-06 21:32:48 +00:00
|
|
|
async function heartbeat() {
|
|
|
|
analytics.track({
|
|
|
|
anonymousId: id(),
|
|
|
|
event: 'heartbeat',
|
2023-07-07 17:44:36 +00:00
|
|
|
properties: {
|
|
|
|
version: app.getVersion(),
|
|
|
|
},
|
2023-07-06 21:32:48 +00:00
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
heartbeat()
|
|
|
|
|
2023-07-06 18:50:46 +00:00
|
|
|
if (app.isPackaged) {
|
2023-07-06 21:32:48 +00:00
|
|
|
heartbeat()
|
2023-06-27 21:50:50 +00:00
|
|
|
autoUpdater.checkForUpdates()
|
2023-07-06 18:50:46 +00:00
|
|
|
setInterval(() => {
|
2023-07-06 21:32:48 +00:00
|
|
|
heartbeat()
|
2023-07-06 18:50:46 +00:00
|
|
|
autoUpdater.checkForUpdates()
|
|
|
|
}, 60000)
|
|
|
|
}
|
2023-06-28 00:31:02 +00:00
|
|
|
|
2023-07-06 16:02:13 +00:00
|
|
|
autoUpdater.on('error', e => {
|
|
|
|
console.error('update check failed', e)
|
|
|
|
})
|
|
|
|
|
2023-06-28 00:31:02 +00:00
|
|
|
autoUpdater.on('update-downloaded', (event, releaseNotes, releaseName) => {
|
|
|
|
dialog
|
|
|
|
.showMessageBox({
|
|
|
|
type: 'info',
|
|
|
|
buttons: ['Restart Now', 'Later'],
|
|
|
|
title: 'New update available',
|
|
|
|
message: process.platform === 'win32' ? releaseNotes : releaseName,
|
|
|
|
detail: 'A new version of Ollama is available. Restart to apply the update.',
|
|
|
|
})
|
|
|
|
.then(returnValue => {
|
|
|
|
if (returnValue.response === 0) autoUpdater.quitAndInstall()
|
|
|
|
})
|
|
|
|
})
|