Skip to content

Commit 244d425

Browse files
GregoMac1fmaclen
andauthored
feat: add Desktop release for macOS, Windows & Linux (#130)
Closes #64 ![image](https://github.com/user-attachments/assets/6f971403-5ac0-4355-9066-eaf482efbe87) ![image](https://github.com/user-attachments/assets/ca4f62c4-c019-4acd-b229-d88133a68ac2) --------- Co-authored-by: Fernando Maclen <[email protected]>
1 parent 63670f7 commit 244d425

12 files changed

+3457
-41
lines changed

.github/workflows/publish.yml .github/workflows/release.yml

+37-11
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,13 @@
1-
name: Publish Docker image to container registry
1+
name: Release
22

33
on:
44
push:
55
branches:
66
- main
77

88
jobs:
9-
release:
10-
name: Semantic release
9+
versioning:
10+
name: Versioning
1111
runs-on: ubuntu-latest
1212
outputs:
1313
new_release_published: ${{ steps.semantic.outputs.new_release_published }}
@@ -30,16 +30,16 @@ jobs:
3030
GITHUB_TOKEN: ${{ secrets.GH_PERSONAL_ACCESS_TOKEN }}
3131

3232
cloudflare:
33-
name: CloudFlare Pages
34-
needs: release
35-
if: ${{ needs.release.outputs.new_release_published == 'true' }}
33+
name: Live demo
34+
needs: versioning
35+
if: ${{ needs.versioning.outputs.new_release_published == 'true' }}
3636
runs-on: ubuntu-latest
3737
steps:
3838
- name: Checkout
3939
uses: actions/checkout@v4
4040

4141
- name: Update app version in package.json
42-
run: node .github/workflows/publish/update-package-json.cjs ${{ needs.release.outputs.new_release_version }}
42+
run: node .github/workflows/publish/update-package-json.cjs ${{ needs.versioning.outputs.new_release_version }}
4343

4444
- name: Set up Node.js
4545
uses: actions/setup-node@v4
@@ -55,9 +55,35 @@ jobs:
5555
accountId: ${{ secrets.CF_ACCOUNT_ID }}
5656
command: pages publish .svelte-kit/cloudflare --project-name hollama
5757

58+
electron:
59+
needs: versioning
60+
if: ${{ needs.versioning.outputs.new_release_published == 'true' }}
61+
62+
name: Desktop app (${{ matrix.os }})
63+
runs-on: ${{ matrix.os }}
64+
65+
strategy:
66+
matrix:
67+
os: [macos-13, ubuntu-latest, windows-latest] # macos-13 is the lastest one on x86
68+
69+
steps:
70+
- uses: actions/checkout@v4
71+
- uses: actions/setup-node@v4
72+
73+
- name: Update app version in package.json
74+
run: node .github/workflows/publish/update-package-json.cjs ${{ needs.versioning.outputs.new_release_version }}
75+
76+
- name: Install Electron & SvelteKit's dependencies
77+
run: npm ci
78+
79+
- name: Build & release Electron app
80+
run: npm run electron:build
81+
env:
82+
GITHUB_TOKEN: ${{ secrets.GH_PERSONAL_ACCESS_TOKEN }}
83+
5884
docker:
59-
needs: release
60-
if: ${{ needs.release.outputs.new_release_published == 'true' }}
85+
needs: versioning
86+
if: ${{ needs.versioning.outputs.new_release_published == 'true' }}
6187

6288
name: Docker image
6389
runs-on: ubuntu-latest
@@ -66,7 +92,7 @@ jobs:
6692
uses: actions/checkout@v4
6793

6894
- name: Update app version in package.json
69-
run: node .github/workflows/publish/update-package-json.cjs ${{ needs.release.outputs.new_release_version }}
95+
run: node .github/workflows/publish/update-package-json.cjs ${{ needs.versioning.outputs.new_release_version }}
7096

7197
- name: Build & publish
7298
uses: mr-smithers-excellent/docker-build-push@v6
@@ -75,6 +101,6 @@ jobs:
75101
registry: ghcr.io
76102
multiPlatform: true
77103
platform: linux/amd64,linux/arm64
78-
tags: latest, ${{ needs.release.outputs.new_release_version }}
104+
tags: latest, ${{ needs.versioning.outputs.new_release_version }}
79105
username: ${{ secrets.GH_REGISTRY_USERNAME }}
80106
password: ${{ secrets.GH_PERSONAL_ACCESS_TOKEN }}

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,7 @@
11
.DS_Store
22
node_modules
33
/build
4+
/dist
45
/.svelte-kit
56
/package
67
.env

CONTRIBUTING.md

+8-1
Original file line numberDiff line numberDiff line change
@@ -15,8 +15,9 @@ Hollama is a static site built with:
1515
- TypeScript
1616
- Svelte & SvelteKit
1717
- Vite
18-
- Playwright
1918
- Tailwind CSS
19+
- Playwright
20+
- Electron
2021

2122
Install dependencies with `npm install`, start a development server:
2223

@@ -25,6 +26,9 @@ npm run dev
2526

2627
# or start the server and open the app in a new browser tab
2728
npm run dev -- --open
29+
30+
# run the app with Electron
31+
npm run electron
2832
```
2933

3034
## Building
@@ -33,6 +37,9 @@ To create a production version of Hollama:
3337

3438
```bash
3539
npm run build
40+
41+
# or package with Electron
42+
npm run electron:build
3643
```
3744

3845
You can preview the production build with `npm run preview`.

README.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -19,7 +19,7 @@ A minimal web-UI for talking to [Ollama](https://github.com/jmorganca/ollama/) s
1919
- ⚡️ [Live demo](https://hollama.fernando.is)
2020
- _No sign-up required_
2121
- 🐳 [Self-hosting](#self-host-docker) with Docker
22-
- 🖥️ Download for [macOS, Windows & Linux](https://github.com/fmaclen/hollama/releases) _(Coming soon)_
22+
- 🖥️ Download for [macOS, Windows & Linux](https://github.com/fmaclen/hollama/releases)
2323
- 🐞 [Contribute](CONTRIBUTING.md)
2424

2525
> ![session](tests/docs.test.ts-snapshots/session.png)

electron-builder.yml

+30
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,30 @@
1+
appId: com.hollama.desktop
2+
productName: Hollama
3+
artifactName: ${productName}_${version}-${os}-${arch}.${ext}
4+
5+
files:
6+
- build/**/*
7+
- electron/**/*
8+
9+
mac:
10+
category: public.app-category.developer-tools
11+
icon: electron/resources/icons/mac/icon.icns
12+
hardenedRuntime: true
13+
target: dmg
14+
15+
win:
16+
target: nsis
17+
icon: electron/resources/icons/win/icon.ico
18+
19+
nsis:
20+
oneClick: false
21+
allowToChangeInstallationDirectory: true
22+
23+
linux:
24+
target: tar.gz
25+
icon: electron/resources/icons/png/1024x1024.png
26+
27+
publish:
28+
provider: github
29+
releaseType: release
30+
vPrefixedTagName: false

electron/main.js

+84
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,84 @@
1+
import net from 'net';
2+
import { join } from 'path';
3+
import { app, BrowserWindow, utilityProcess, dialog } from 'electron';
4+
5+
// Vite default dev & production ports
6+
const hollamaPort = app.isPackaged ? '4173' : '5173';
7+
const HOLLAMA_HOST = '127.0.0.1';
8+
9+
function createWindow() {
10+
const mainWindow = new BrowserWindow({
11+
width: 1280,
12+
height: 800,
13+
minWidth: 400,
14+
minHeight: 640
15+
});
16+
17+
mainWindow.menuBarVisible = false; // Windows: hides the menu bar
18+
mainWindow.loadURL(`http://${HOLLAMA_HOST}:${hollamaPort}`);
19+
}
20+
21+
function checkServerAvailability(port) {
22+
const MAX_RETRIES = 10;
23+
const RETRY_INTERVAL_IN_MS = 1000;
24+
25+
return new Promise((resolve, reject) => {
26+
let retries = 0;
27+
28+
function tryConnection() {
29+
const socket = new net.Socket();
30+
31+
const onError = () => {
32+
socket.destroy();
33+
34+
if (retries >= MAX_RETRIES) {
35+
reject(new Error(`Couldn't connect to Hollama server after ${MAX_RETRIES} attempts`));
36+
} else {
37+
retries++;
38+
setTimeout(tryConnection, RETRY_INTERVAL_IN_MS);
39+
}
40+
};
41+
42+
socket.setTimeout(1000);
43+
socket.once('error', onError);
44+
socket.once('timeout', onError);
45+
46+
socket.connect(port, HOLLAMA_HOST, () => {
47+
socket.destroy();
48+
resolve();
49+
});
50+
}
51+
52+
tryConnection();
53+
});
54+
}
55+
56+
app
57+
.whenReady()
58+
.then(async () => {
59+
if (app.isPackaged) {
60+
utilityProcess.fork(join(app.getAppPath(), 'build', 'index.js'), {
61+
env: { ...process.env, PORT: hollamaPort }
62+
});
63+
} else {
64+
console.warn('##### Running Electron in development mode');
65+
console.log('##### Run `npm run dev` to start the Hollama server separately');
66+
}
67+
68+
await checkServerAvailability(parseInt(hollamaPort));
69+
createWindow();
70+
})
71+
.catch((error) => {
72+
dialog.showErrorBox('Error', error.message);
73+
app.quit();
74+
});
75+
76+
// Quit when all windows are closed, except on macOS.
77+
app.on('window-all-closed', function () {
78+
if (process.platform !== 'darwin') app.quit();
79+
});
80+
81+
// macOS: Open a window if none are open
82+
app.on('activate', function () {
83+
if (BrowserWindow.getAllWindows().length === 0) createWindow();
84+
});
1.44 MB
Binary file not shown.
803 KB
Loading

electron/resources/icons/win/icon.ico

353 KB
Binary file not shown.

0 commit comments

Comments
 (0)