index.ts 6.46 KB
Newer Older
1
import { spawn } from 'child_process'
2
import { app, autoUpdater, dialog, Tray, Menu, BrowserWindow, nativeTheme } from 'electron'
Eva Ho's avatar
Eva Ho committed
3
import Store from 'electron-store'
Eva Ho's avatar
Eva Ho committed
4
5
import winston from 'winston'
import 'winston-daily-rotate-file'
Bruce MacDonald's avatar
Bruce MacDonald committed
6
import * as path from 'path'
Jeffrey Morgan's avatar
Jeffrey Morgan committed
7

Jeffrey Morgan's avatar
Jeffrey Morgan committed
8
import { analytics, id } from './telemetry'
Jeffrey Morgan's avatar
Jeffrey Morgan committed
9
import { installed } from './install'
Jeffrey Morgan's avatar
Jeffrey Morgan committed
10

Jeffrey Morgan's avatar
Jeffrey Morgan committed
11
12
require('@electron/remote/main').initialize()

Jeffrey Morgan's avatar
Jeffrey Morgan committed
13
const store = new Store()
14
let tray: Tray | null = null
15
16
17
let welcomeWindow: BrowserWindow | null = null

declare const MAIN_WINDOW_WEBPACK_ENTRY: string
18

19
20
21
22
23
24
25
26
27
const logger = winston.createLogger({
  transports: [
    new winston.transports.Console(),
    new winston.transports.File({
      filename: path.join(app.getPath('home'), '.ollama', 'logs', 'server.log'),
      maxsize: 1024 * 1024 * 20,
      maxFiles: 5,
    }),
  ],
Jeffrey Morgan's avatar
Jeffrey Morgan committed
28
  format: winston.format.printf(info => info.message),
Eva Ho's avatar
Eva Ho committed
29
30
})

Eva Ho's avatar
Eva Ho committed
31
const SingleInstanceLock = app.requestSingleInstanceLock()
32
33
34
if (!SingleInstanceLock) {
  app.quit()
}
Jeffrey Morgan's avatar
Jeffrey Morgan committed
35

36
37
38
39
40
41
42
43
function firstRunWindow() {
  // Create the browser window.
  welcomeWindow = new BrowserWindow({
    width: 400,
    height: 500,
    frame: false,
    fullscreenable: false,
    resizable: false,
44
45
    movable: true,
    show: false,
46
47
48
49
    webPreferences: {
      nodeIntegration: true,
      contextIsolation: false,
    },
50
    alwaysOnTop: true,
51
52
53
54
55
56
57
  })

  require('@electron/remote/main').enable(welcomeWindow.webContents)

  // and load the index.html of the app.
  welcomeWindow.loadURL(MAIN_WINDOW_WEBPACK_ENTRY)

58
59
  welcomeWindow.on('ready-to-show', () => welcomeWindow.show())

60
61
62
63
64
  // for debugging
  // welcomeWindow.webContents.openDevTools()

  if (process.platform === 'darwin') {
    app.dock.hide()
65
  }
66
67
68
}

function createSystemtray() {
69
70
71
  let iconPath = nativeTheme.shouldUseDarkColors
    ? path.join(__dirname, '..', '..', 'assets', 'ollama_icon_16x16Template.png') 
    : path.join(__dirname, '..', '..', 'assets', 'ollama_outline_icon_16x16Template.png')
Eva Ho's avatar
Eva Ho committed
72
73

  if (app.isPackaged) {
74
75
76
    iconPath = nativeTheme.shouldUseDarkColors
    ? path.join(process.resourcesPath, 'ollama_icon_16x16Template.png') 
    : path.join(process.resourcesPath, 'ollama_outline_icon_16x16Template.png')
Eva Ho's avatar
Eva Ho committed
77
  }
Jeffrey Morgan's avatar
Jeffrey Morgan committed
78

79
  tray = new Tray(iconPath)
Jeffrey Morgan's avatar
Jeffrey Morgan committed
80

81
82
83
84
85
86
87
88
89
90
91
92
  nativeTheme.on('updated', function theThemeHasChanged () {
    if (nativeTheme.shouldUseDarkColors) {
      app.isPackaged 
        ? tray.setImage(path.join(process.resourcesPath, 'ollama_icon_16x16Template.png')) 
        : tray.setImage(path.join(__dirname, '..', '..', 'assets', 'ollama_icon_16x16Template.png'))
    } else {
      app.isPackaged 
        ? tray.setImage(path.join(process.resourcesPath, 'ollama_outline_icon_16x16Template.png')) 
        : tray.setImage(path.join(__dirname, '..', '..', 'assets', 'ollama_outline_icon_16x16Template.png'))
    }
  })

Jeffrey Morgan's avatar
Jeffrey Morgan committed
93
  const contextMenu = Menu.buildFromTemplate([{ role: 'quit', label: 'Quit Ollama', accelerator: 'Command+Q' }])
Jeffrey Morgan's avatar
Jeffrey Morgan committed
94

95
96
97
98
99
100
  tray.setContextMenu(contextMenu)
  tray.setToolTip('Ollama')
}

if (require('electron-squirrel-startup')) {
  app.quit()
Jeffrey Morgan's avatar
Jeffrey Morgan committed
101
102
}

Jeffrey Morgan's avatar
Jeffrey Morgan committed
103
104
function server() {
  const binary = app.isPackaged
105
  ? path.join(process.resourcesPath, 'ollama')
Eva Ho's avatar
Eva Ho committed
106
  : path.resolve(process.cwd(), '..', 'ollama')
Jeffrey Morgan's avatar
Jeffrey Morgan committed
107

Eva Ho's avatar
Eva Ho committed
108
  const proc = spawn(binary, ['serve'])
Jeffrey Morgan's avatar
Jeffrey Morgan committed
109

Jeffrey Morgan's avatar
Jeffrey Morgan committed
110
  proc.stdout.on('data', data => {
Eva Ho's avatar
Eva Ho committed
111
112
    logger.info(data.toString().trim())
  })
Jeffrey Morgan's avatar
Jeffrey Morgan committed
113

Jeffrey Morgan's avatar
Jeffrey Morgan committed
114
  proc.stderr.on('data', data => {
Eva Ho's avatar
Eva Ho committed
115
116
    logger.error(data.toString().trim())
  })
117
118
119
    

  proc.on('exit', (code, signal) => {
hoyyeva's avatar
hoyyeva committed
120
121
    logger.error(`Server exited with code: ${code}`)
    setTimeout(server, 3000)
Eva Ho's avatar
Eva Ho committed
122
  })
123

124
  app.on('before-quit', () => {
Eva Ho's avatar
Eva Ho committed
125
126
127
    proc.off('exit', server)
    proc.kill()
  })
Jeffrey Morgan's avatar
Jeffrey Morgan committed
128
129
}

130
131
if (process.platform === 'darwin') {
  app.dock.hide()
132
133
}

134
135
app.on('ready', () => {
  if (process.platform === 'darwin') {
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
    if (app.isPackaged) {
      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) {
Eva Ho's avatar
Eva Ho committed
163
            logger.error(`[Move to Applications] Failed to move to applications folder - ${e.message}}`)
164
          }
165
166
167
        }
      }
    }
168
  }
Jeffrey Morgan's avatar
Jeffrey Morgan committed
169

170
  createSystemtray()
171
  server()
172

173
  if (store.get('first-time-run') && installed()) {
174
    app.setLoginItemSettings({ openAtLogin: app.getLoginItemSettings().openAtLogin })
175
    return
176
  }
177
178
179
180

  // This is the first run or the CLI is no longer installed
  app.setLoginItemSettings({ openAtLogin: true })
  firstRunWindow()
181
})
Jeffrey Morgan's avatar
Jeffrey Morgan committed
182

Jeffrey Morgan's avatar
Jeffrey Morgan committed
183
184
185
186
187
188
189
190
191
192
193
// 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.
Jeffrey Morgan's avatar
Jeffrey Morgan committed
194
195
196
autoUpdater.setFeedURL({
  url: `https://ollama.ai/api/update?os=${process.platform}&arch=${process.arch}&version=${app.getVersion()}`,
})
197

Jeffrey Morgan's avatar
Jeffrey Morgan committed
198
199
200
201
async function heartbeat() {
  analytics.track({
    anonymousId: id(),
    event: 'heartbeat',
Jeffrey Morgan's avatar
Jeffrey Morgan committed
202
203
204
    properties: {
      version: app.getVersion(),
    },
Jeffrey Morgan's avatar
Jeffrey Morgan committed
205
206
207
  })
}

208
if (app.isPackaged) {
Jeffrey Morgan's avatar
Jeffrey Morgan committed
209
  heartbeat()
210
  autoUpdater.checkForUpdates()
211
  setInterval(() => {
Jeffrey Morgan's avatar
Jeffrey Morgan committed
212
    heartbeat()
213
    autoUpdater.checkForUpdates()
Jeffrey Morgan's avatar
Jeffrey Morgan committed
214
  }, 60 * 60 * 1000)
215
}
Jeffrey Morgan's avatar
Jeffrey Morgan committed
216

217
autoUpdater.on('error', e => {
Jeffrey Morgan's avatar
Jeffrey Morgan committed
218
  logger.error(`update check failed - ${e.message}`)
219
220
})

Jeffrey Morgan's avatar
Jeffrey Morgan committed
221
222
223
224
225
226
227
228
229
230
231
232
233
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()
    })
})