fix(Kiwix): initial download and setup

This commit is contained in:
Jake Turner 2025-12-07 16:04:41 -08:00
parent ce8dbd91ab
commit 2ff7b055b5
No known key found for this signature in database
GPG Key ID: 694BC38EF2ED4844
5 changed files with 14 additions and 83 deletions

View File

@ -16,7 +16,7 @@ export class DockerService {
public static CYBERCHEF_SERVICE_NAME = 'nomad_cyberchef'
public static FLATNOTES_SERVICE_NAME = 'nomad_flatnotes'
public static KOLIBRI_SERVICE_NAME = 'nomad_kolibri'
public static NOMAD_STORAGE_ABS_PATH = '/opt/project-nomad/storage'
public static NOMAD_STORAGE_PATH = '/storage'
constructor() {
this.docker = new Docker({ socketPath: '/var/run/docker.sock' })
@ -167,7 +167,7 @@ export class DockerService {
// }
const containerConfig = this._parseContainerConfig(service.container_config)
this._createContainer(service, containerConfig) // Don't await this method - we will use server-sent events to notify the client of progress
await this._createContainer(service, containerConfig)
return {
success: true,
@ -178,8 +178,7 @@ export class DockerService {
/**
* Handles the long-running process of creating a Docker container for a service.
* NOTE: This method should not be called directly. Instead, use `createContainerPreflight` to check prerequisites first
* and return an HTTP response to the client, if needed. This method will then transmit server-sent events to the client
* to notify them of the progress.
* This method will also transmit server-sent events to the client to notify of progress.
* @param serviceName
* @returns
*/
@ -332,7 +331,7 @@ export class DockerService {
const WIKIPEDIA_ZIM_URL =
'https://github.com/Crosstalk-Solutions/project-nomad/raw/refs/heads/master/install/wikipedia_en_100_mini_2025-06.zim'
const zimPath = '/zim/wikipedia_en_100_mini_2025-06.zim'
const filepath = path.join(DockerService.NOMAD_STORAGE_ABS_PATH, zimPath)
const filepath = path.join(process.cwd(), DockerService.NOMAD_STORAGE_PATH, zimPath)
logger.info(`[DockerService] Kiwix Serve pre-install: Downloading ZIM file to ${filepath}`)
this._broadcast(

View File

@ -1,74 +1,13 @@
import {
DoResumableDownloadParams,
DoResumableDownloadWithRetryParams,
DoSimpleDownloadParams,
} from '../../types/downloads.js'
import axios, { AxiosResponse } from 'axios'
import axios from 'axios'
import { Transform } from 'stream'
import { deleteFileIfExists, ensureDirectoryExists, getFileStatsIfExists } from './fs.js'
import { createWriteStream } from 'fs'
import logger from '@adonisjs/core/services/logger'
import path from 'path'
export async function doSimpleDownload({
url,
filepath,
timeout = 30000,
signal,
}: DoSimpleDownloadParams): Promise<string> {
return new Promise(async (resolve, reject) => {
let response: AxiosResponse<any> | undefined
let writer: ReturnType<typeof createWriteStream> | undefined
const cleanup = (err?: Error) => {
try {
response?.data?.destroy?.()
} catch {}
try {
writer?.destroy?.()
} catch {}
if (err) {
try {
logger.error(`Download failed for ${url}: ${err.message}`)
} catch {}
reject(err)
}
}
try {
const dirname = path.dirname(filepath)
await ensureDirectoryExists(dirname)
response = await axios.get(url, {
responseType: 'stream',
signal,
timeout,
})
writer = createWriteStream(filepath)
response?.data.pipe(writer)
response?.data.on('error', cleanup)
writer?.on('error', cleanup)
writer?.on('finish', () => {
cleanup()
resolve(filepath)
})
signal?.addEventListener(
'abort',
() => {
cleanup(new Error('Download aborted'))
},
{ once: true }
)
} catch (error) {
cleanup(error as Error)
}
})
}
/**
* Perform a resumable download with progress tracking
* @param param0 - Download parameters. Leave allowedMimeTypes empty to skip mime type checking.

View File

@ -4,17 +4,18 @@ import { BaseSeeder } from '@adonisjs/lucid/seeders'
import { ModelAttributes } from '@adonisjs/lucid/types/model'
export default class ServiceSeeder extends BaseSeeder {
private static NOMAD_STORAGE_ABS_PATH = '/opt/project-nomad/storage'
private static DEFAULT_SERVICES: Omit<ModelAttributes<Service>, 'created_at' | 'updated_at' | 'metadata' | 'id'>[] = [
{
service_name: DockerService.KIWIX_SERVICE_NAME,
friendly_name: 'Kiwix',
description: 'Offline Wikipedia, eBooks, and more',
container_image: 'ghcr.io/kiwix/kiwix-serve',
container_command: '*.zim --address=0.0.0.0',
container_image: 'ghcr.io/kiwix/kiwix-serve:3.8.1',
container_command: '*.zim --address=all',
container_config: JSON.stringify({
HostConfig: {
RestartPolicy: { Name: 'unless-stopped' },
Binds: [`${DockerService.NOMAD_STORAGE_ABS_PATH}/zim:/data`],
Binds: [`${ServiceSeeder.NOMAD_STORAGE_ABS_PATH}/zim:/data`],
PortBindings: { '8080/tcp': [{ HostPort: '8090' }] }
},
ExposedPorts: { '8080/tcp': {} }
@ -33,7 +34,7 @@ export default class ServiceSeeder extends BaseSeeder {
container_config: JSON.stringify({
HostConfig: {
RestartPolicy: { Name: 'unless-stopped' },
Binds: [`${DockerService.NOMAD_STORAGE_ABS_PATH}/ollama:/root/.ollama`],
Binds: [`${ServiceSeeder.NOMAD_STORAGE_ABS_PATH}/ollama:/root/.ollama`],
PortBindings: { '11434/tcp': [{ HostPort: '11434' }] }
},
ExposedPorts: { '11434/tcp': {} }
@ -53,7 +54,7 @@ export default class ServiceSeeder extends BaseSeeder {
HostConfig: {
RestartPolicy: { Name: 'unless-stopped' },
NetworkMode: 'host',
Binds: [`${DockerService.NOMAD_STORAGE_ABS_PATH}/open-webui:/app/backend/data`]
Binds: [`${ServiceSeeder.NOMAD_STORAGE_ABS_PATH}/open-webui:/app/backend/data`]
},
Env: ['WEBUI_AUTH=False', 'PORT=3000', 'OLLAMA_BASE_URL=http://127.0.0.1:11434']
}),
@ -90,7 +91,7 @@ export default class ServiceSeeder extends BaseSeeder {
HostConfig: {
RestartPolicy: { Name: 'unless-stopped' },
PortBindings: { '8080/tcp': [{ HostPort: '8200' }] },
Binds: [`${DockerService.NOMAD_STORAGE_ABS_PATH}/flatnotes:/data`]
Binds: [`${ServiceSeeder.NOMAD_STORAGE_ABS_PATH}/flatnotes:/data`]
},
ExposedPorts: { '8080/tcp': {} },
Env: ['FLATNOTES_AUTH_TYPE=none']
@ -110,7 +111,7 @@ export default class ServiceSeeder extends BaseSeeder {
HostConfig: {
RestartPolicy: { Name: 'unless-stopped' },
PortBindings: { '8080/tcp': [{ HostPort: '8300' }] },
Binds: [`${DockerService.NOMAD_STORAGE_ABS_PATH}/kolibri:/root/.kolibri`]
Binds: [`${ServiceSeeder.NOMAD_STORAGE_ABS_PATH}/kolibri:/root/.kolibri`]
},
ExposedPorts: { '8080/tcp': {} },
}),

View File

@ -1,10 +1,3 @@
export type DoSimpleDownloadParams = {
url: string
filepath: string
timeout: number
signal?: AbortSignal
}
export type DoResumableDownloadParams = {
url: string
filepath: string

View File

@ -1,7 +1,6 @@
services:
admin:
image: ghcr.io/crosstalk-solutions/project-nomad:latest
pull_policy: always
container_name: nomad_admin
restart: unless-stopped
ports:
@ -25,7 +24,7 @@ services:
- DB_PASSWORD=replaceme
- DB_NAME=nomad
- DB_SSL=false
- REDIS_HOST=localhost
- REDIS_HOST=redis
- REDIS_PORT=6379
depends_on:
mysql: