"vscode:/vscode.git/clone" did not exist on "5dc55a5f02de2aff87a12f5dfcf5a8a781f1220a"
index.ts 6.1 KB
Newer Older
1
import { spawn, exec } from 'child_process'
2
import { app, autoUpdater, dialog, Tray, Menu } 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
import * as fs from 'fs'
Jeffrey Morgan's avatar
Jeffrey Morgan committed
8

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

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

Eva Ho's avatar
Eva Ho committed
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
const logFile = new winston.transports.DailyRotateFile({
  filename: path.join(app.getPath('home'), '.ollama', 'logs', 'server-%DATE%.log'),
  datePattern: 'YYYY-MM-DD-HH',
  zippedArchive: true,
  maxSize: '20m',
  maxFiles: '7d',
});

const logger = winston.createLogger({ 
  transports: [logFile],
  format: winston.format.combine(
    winston.format.colorize(),
    winston.format.timestamp(),
    winston.format.printf(info => `${info.timestamp} ${info.level}: ${info.message}`)
  )
})

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

38
const createSystemtray = () => {
39
  let iconPath = path.join(__dirname, '..', '..', 'assets', 'ollama_icon_16x16Template.png')
Eva Ho's avatar
Eva Ho committed
40
41

  if (app.isPackaged) {
42
    iconPath = path.join(process.resourcesPath, 'ollama_icon_16x16Template.png')
Eva Ho's avatar
Eva Ho committed
43
  }
Jeffrey Morgan's avatar
Jeffrey Morgan committed
44

45
  tray = new Tray(iconPath)
Jeffrey Morgan's avatar
Jeffrey Morgan committed
46

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

49
50
51
52
53
54
55
  tray.setContextMenu(contextMenu)
  tray.setToolTip('Ollama')
}

// Handle creating/removing shortcuts on Windows when installing/uninstalling.
if (require('electron-squirrel-startup')) {
  app.quit()
Jeffrey Morgan's avatar
Jeffrey Morgan committed
56
57
}

Jeffrey Morgan's avatar
Jeffrey Morgan committed
58
59
const ollama = path.join(process.resourcesPath, 'ollama')

Jeffrey Morgan's avatar
Jeffrey Morgan committed
60
61
function server() {
  const binary = app.isPackaged
62
63
  ? path.join(process.resourcesPath, 'ollama')
  : path.resolve(process.cwd(), '..', 'ollama')
Jeffrey Morgan's avatar
Jeffrey Morgan committed
64
65
66

  const proc = spawn(binary, ['serve'])
  proc.stdout.on('data', data => {
Eva Ho's avatar
Eva Ho committed
67
    logger.info(`[server] ${data.toString()}`)
Jeffrey Morgan's avatar
Jeffrey Morgan committed
68
69
  })
  proc.stderr.on('data', data => {
Eva Ho's avatar
Eva Ho committed
70
    logger.error(`[server] ${data.toString()}`)
Jeffrey Morgan's avatar
Jeffrey Morgan committed
71
72
  })

73
  proc.on('exit', () => {
Eva Ho's avatar
Eva Ho committed
74
    logger.info('Restarting the server...');
75
76
77
78
    server();
  })

  proc.on('disconnect', () => {
Eva Ho's avatar
Eva Ho committed
79
    logger.info('Server disconnected. Reconnecting...');
80
81
82
    server();
  })

Jeffrey Morgan's avatar
Jeffrey Morgan committed
83
84
85
86
87
  process.on('exit', () => {
    proc.kill()
  })
}

88

89
function installCLI() {
Jeffrey Morgan's avatar
Jeffrey Morgan committed
90
91
92
93
94
95
  const symlinkPath = '/usr/local/bin/ollama'

  if (fs.existsSync(symlinkPath) && fs.readlinkSync(symlinkPath) === ollama) {
    return
  }

96
97
98
99
  dialog
    .showMessageBox({
      type: 'info',
      title: 'Ollama CLI installation',
Jeffrey Morgan's avatar
Jeffrey Morgan committed
100
      message: 'To make the Ollama command work in your terminal, it needs administrator privileges.',
101
102
103
104
      buttons: ['OK'],
    })
    .then(result => {
      if (result.response === 0) {
105
        const command = `
Jeffrey Morgan's avatar
Jeffrey Morgan committed
106
107
    do shell script "ln -F -s ${ollama} /usr/local/bin/ollama" with administrator privileges
    `
108
109
        exec(`osascript -e '${command}'`, (error: Error | null, stdout: string, stderr: string) => {
          if (error) {
Eva Ho's avatar
Eva Ho committed
110
            logger.error(`[CLI] Failed to install CLI - ${error.message}`)
111
112
            return
          }
Eva Ho's avatar
Eva Ho committed
113
114
115

          logger.info(`[CLI] ${stdout}}`)
          logger.error(`[CLI] ${stderr}`)
116
117
118
119
120
        })
      }
    })
}

121
122
123
124
app.on('ready', () => {
  if (process.platform === 'darwin') {
    app.dock.hide()

Eva Ho's avatar
Eva Ho committed
125
126
127
    if (!store.has('first-time-run')) {
      // This is the first run
      app.setLoginItemSettings({ openAtLogin: true })
Jeffrey Morgan's avatar
Jeffrey Morgan committed
128
      store.set('first-time-run', false)
Eva Ho's avatar
Eva Ho committed
129
130
131
132
133
    } else {
      // The app has been run before
      app.setLoginItemSettings({ openAtLogin: app.getLoginItemSettings().openAtLogin })
    }

134
135
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
    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
161
            logger.error(`[Move to Applications] Failed to move to applications folder - ${e.message}}`)
162
          }
163
164
        }
      }
165
166

      installCLI()
167
    }
168
  }
Jeffrey Morgan's avatar
Jeffrey Morgan committed
169

170
  createSystemtray()
171
  server()
172
})
Jeffrey Morgan's avatar
Jeffrey Morgan committed
173

Jeffrey Morgan's avatar
Jeffrey Morgan committed
174
175
176
177
178
179
180
181
182
183
184
// 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
185
186
187
autoUpdater.setFeedURL({
  url: `https://ollama.ai/api/update?os=${process.platform}&arch=${process.arch}&version=${app.getVersion()}`,
})
188

Jeffrey Morgan's avatar
Jeffrey Morgan committed
189
190
191
192
async function heartbeat() {
  analytics.track({
    anonymousId: id(),
    event: 'heartbeat',
Jeffrey Morgan's avatar
Jeffrey Morgan committed
193
194
195
    properties: {
      version: app.getVersion(),
    },
Jeffrey Morgan's avatar
Jeffrey Morgan committed
196
197
198
  })
}

199
if (app.isPackaged) {
Jeffrey Morgan's avatar
Jeffrey Morgan committed
200
  heartbeat()
201
  autoUpdater.checkForUpdates()
202
  setInterval(() => {
Jeffrey Morgan's avatar
Jeffrey Morgan committed
203
    heartbeat()
204
    autoUpdater.checkForUpdates()
Jeffrey Morgan's avatar
Jeffrey Morgan committed
205
  }, 60 * 60 * 1000)
206
}
Jeffrey Morgan's avatar
Jeffrey Morgan committed
207

208
autoUpdater.on('error', e => {
Eva Ho's avatar
Eva Ho committed
209
  logger.error(`[auto updater] update check failed - ${e.message}`)
210
211
})

Jeffrey Morgan's avatar
Jeffrey Morgan committed
212
213
214
215
216
217
218
219
220
221
222
223
224
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()
    })
})