From 801174585b1cf90df83094f001c47fcf9de71030 Mon Sep 17 00:00:00 2001 From: cody conder Date: Sat, 8 Nov 2025 00:18:25 -0600 Subject: [PATCH] Initial commit with kopia app working --- .github/workflows/renovate.yml | 45 +++ .github/workflows/test.yml | 23 ++ .gitignore | 1 + LICENSE | 13 + README.md | 31 ++ __tests__/apps.test.ts | 77 ++++ apps/kopia/config.json | 37 ++ apps/kopia/docker-compose.json | 112 ++++++ apps/minio_Potentially Edit/config.json | 43 ++ .../docker-compose.json | 50 +++ .../minio_Potentially Edit/docker-compose.yml | 69 ++++ .../metadata/description.md | 28 ++ apps/minio_Potentially Edit/metadata/logo.jpg | Bin 0 -> 40144 bytes apps/silverbullet_Reference/config.json | 38 ++ .../docker-compose.json | 25 ++ .../silverbullet_Reference/docker-compose.yml | 36 ++ .../metadata/description.md | 22 ++ apps/silverbullet_Reference/metadata/logo.jpg | Bin 0 -> 18592 bytes apps/siyuan_Need to edit/config.json | 33 ++ apps/siyuan_Need to edit/docker-compose.json | 23 ++ apps/siyuan_Need to edit/docker-compose.yml | 40 ++ .../metadata/description.md | 374 ++++++++++++++++++ apps/siyuan_Need to edit/metadata/logo.jpg | Bin 0 -> 29131 bytes apps/whoami/config.json | 24 ++ apps/whoami/docker-compose.json | 10 + apps/whoami/docker-compose.yml | 34 ++ apps/whoami/metadata/description.md | 43 ++ apps/whoami/metadata/logo.jpg | Bin 0 -> 18692 bytes bun.lockb | Bin 0 -> 9101 bytes config.js | 3 + package.json | 25 ++ renovate.json | 62 +++ scripts/update-config.ts | 35 ++ tsconfig.json | 24 ++ 34 files changed, 1380 insertions(+) create mode 100644 .github/workflows/renovate.yml create mode 100644 .github/workflows/test.yml create mode 100644 .gitignore create mode 100644 LICENSE create mode 100644 README.md create mode 100644 __tests__/apps.test.ts create mode 100644 apps/kopia/config.json create mode 100644 apps/kopia/docker-compose.json create mode 100644 apps/minio_Potentially Edit/config.json create mode 100644 apps/minio_Potentially Edit/docker-compose.json create mode 100644 apps/minio_Potentially Edit/docker-compose.yml create mode 100644 apps/minio_Potentially Edit/metadata/description.md create mode 100644 apps/minio_Potentially Edit/metadata/logo.jpg create mode 100644 apps/silverbullet_Reference/config.json create mode 100644 apps/silverbullet_Reference/docker-compose.json create mode 100644 apps/silverbullet_Reference/docker-compose.yml create mode 100644 apps/silverbullet_Reference/metadata/description.md create mode 100644 apps/silverbullet_Reference/metadata/logo.jpg create mode 100644 apps/siyuan_Need to edit/config.json create mode 100644 apps/siyuan_Need to edit/docker-compose.json create mode 100644 apps/siyuan_Need to edit/docker-compose.yml create mode 100644 apps/siyuan_Need to edit/metadata/description.md create mode 100644 apps/siyuan_Need to edit/metadata/logo.jpg create mode 100644 apps/whoami/config.json create mode 100644 apps/whoami/docker-compose.json create mode 100644 apps/whoami/docker-compose.yml create mode 100644 apps/whoami/metadata/description.md create mode 100644 apps/whoami/metadata/logo.jpg create mode 100755 bun.lockb create mode 100644 config.js create mode 100644 package.json create mode 100644 renovate.json create mode 100644 scripts/update-config.ts create mode 100644 tsconfig.json diff --git a/.github/workflows/renovate.yml b/.github/workflows/renovate.yml new file mode 100644 index 0000000..23a7deb --- /dev/null +++ b/.github/workflows/renovate.yml @@ -0,0 +1,45 @@ +name: Renovate +on: + workflow_dispatch: + inputs: + log_level: + type: choice + description: Log level + default: INFO + options: + - DEBUG + - INFO + - WARN + - ERROR + - FATAL + schedule: + - cron: 0 2 * * * + +jobs: + renovate: + runs-on: ubuntu-latest + steps: + - name: Checkout + uses: actions/checkout@v4 + + - name: Install node + uses: actions/setup-node@v4 + with: + node-version: "22" + + - name: Install bun + uses: oven-sh/setup-bun@v2 + + - name: Cache Bun global packages + uses: actions/cache@v4 + with: + path: ~/.bun/install/global + key: ${{ runner.os }}-bun-global-renovate-40 + restore-keys: | + ${{ runner.os }}-bun-global- + + - name: Install Renovate + run: bun install -g renovate@40 + + - name: Run renovate + run: LOG_LEVEL=${{ github.event.inputs.log_level || 'INFO' }} renovate --token ${{ secrets.GITHUB_TOKEN }} ${{ github.repository }} diff --git a/.github/workflows/test.yml b/.github/workflows/test.yml new file mode 100644 index 0000000..d890982 --- /dev/null +++ b/.github/workflows/test.yml @@ -0,0 +1,23 @@ +name: Test + +on: + push: + branches: [main] + pull_request: + branches: [main] + +jobs: + test: + runs-on: ubuntu-latest + + steps: + - uses: actions/checkout@v4 + + - name: Setup Bun + uses: oven-sh/setup-bun@v2 + + - name: Install dependencies + run: bun install + + - name: Run tests + run: bun test diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..c2658d7 --- /dev/null +++ b/.gitignore @@ -0,0 +1 @@ +node_modules/ diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..456c488 --- /dev/null +++ b/LICENSE @@ -0,0 +1,13 @@ + DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE + Version 2, December 2004 + + Copyright (C) 2004 Sam Hocevar + + Everyone is permitted to copy and distribute verbatim or modified + copies of this license document, and changing it is allowed as long + as the name is changed. + + DO WHAT THE FUCK YOU WANT TO PUBLIC LICENSE + TERMS AND CONDITIONS FOR COPYING, DISTRIBUTION AND MODIFICATION + + 0. You just DO WHAT THE FUCK YOU WANT TO. diff --git a/README.md b/README.md new file mode 100644 index 0000000..41d64e0 --- /dev/null +++ b/README.md @@ -0,0 +1,31 @@ +# Example App Store Template + +This repository serves as a template for creating your own custom app store for the Runtipi platform. Use this as a starting point to create and share your own collection of applications. + +## Repository Structure + +- **apps/**: Contains individual app directories + + - Each app has its own folder (e.g., `whoami/`) with the following structure: + - `config.json`: App configuration file + - `docker-compose.json`: Docker setup for the app + - `metadata/`: Contains app visuals and descriptions + - `description.md`: Markdown description of the app + - `logo.jpg`: App logo image + +- **tests/**: Contains test files for the app store + + - `apps.test.ts`: Test suite for validating apps + +## Getting Started + +This repository is intended to serve as a template for creating your own app store. Follow these steps to get started: + +1. Click the "Use this template" button to create a new repository based on this template +2. Customize the apps or add your own app folders in the `apps/` directory +3. Test your app store by using it with Runtipi + +## Documentation + +For detailed instructions on creating your own app store, please refer to the official guide: +[Create Your Own App Store Guide](https://runtipi.io/docs/guides/create-your-own-app-store) diff --git a/__tests__/apps.test.ts b/__tests__/apps.test.ts new file mode 100644 index 0000000..4fd30c9 --- /dev/null +++ b/__tests__/apps.test.ts @@ -0,0 +1,77 @@ +import { expect, test, describe } from "bun:test"; +import { appInfoSchema, dynamicComposeSchema } from '@runtipi/common/schemas' +import { fromError } from 'zod-validation-error'; +import fs from 'node:fs' +import path from 'node:path' + +const getApps = async () => { + const appsDir = await fs.promises.readdir(path.join(process.cwd(), 'apps')) + + const appDirs = appsDir.filter((app) => { + const stat = fs.statSync(path.join(process.cwd(), 'apps', app)) + return stat.isDirectory() + }) + + return appDirs +}; + +const getFile = async (app: string, file: string) => { + const filePath = path.join(process.cwd(), 'apps', app, file) + try { + const file = await fs.promises.readFile(filePath, 'utf-8') + return file + } catch (err) { + return null + } +} + +describe("each app should have the required files", async () => { + const apps = await getApps() + + for (const app of apps) { + const files = ['config.json', 'docker-compose.json', 'metadata/logo.jpg', 'metadata/description.md'] + + for (const file of files) { + test(`app ${app} should have ${file}`, async () => { + const fileContent = await getFile(app, file) + expect(fileContent).not.toBeNull() + }) + } + } +}) + +describe("each app should have a valid config.json", async () => { + const apps = await getApps() + + for (const app of apps) { + test(`app ${app} should have a valid config.json`, async () => { + const fileContent = await getFile(app, 'config.json') + const parsed = appInfoSchema.omit({ urn: true }).safeParse(JSON.parse(fileContent || '{}')) + + if (!parsed.success) { + const validationError = fromError(parsed.error); + console.error(`Error parsing config.json for app ${app}:`, validationError.toString()); + } + + expect(parsed.success).toBe(true) + }) + } +}) + +describe("each app should have a valid docker-compose.json", async () => { + const apps = await getApps() + + for (const app of apps) { + test(`app ${app} should have a valid docker-compose.json`, async () => { + const fileContent = await getFile(app, 'docker-compose.json') + const parsed = dynamicComposeSchema.safeParse(JSON.parse(fileContent || '{}')) + + if (!parsed.success) { + const validationError = fromError(parsed.error); + console.error(`Error parsing docker-compose.json for app ${app}:`, validationError.toString()); + } + + expect(parsed.success).toBe(true) + }) + } +}); diff --git a/apps/kopia/config.json b/apps/kopia/config.json new file mode 100644 index 0000000..1691e0a --- /dev/null +++ b/apps/kopia/config.json @@ -0,0 +1,37 @@ +{ + "name": "kopia", + "min_tipi_version": "4.5.0", + "available": true, + "port": 51515, + "exposable": true, + "dynamic_config": true, + "id": "kopia", + "description": "", + "tipi_version": 2, + "version": "20251023.0.52914", + "categories": ["utilities"], + "short_desc": "", + "author": "kopia", + "dynamic":true, + "source": "https://github.com/kopia/kopia", + "website": "https://kopia.io", + "form_fields": [ + {"type":"text", + "label":"username", + "required":true, + "env_variable":"USER"}, + {"type":"text", + "label":"TZ", + "required":false, + "env_variable":"TZ"}, + {"type":"password", + "label":"Password", + "required":true, + "env_variable":"PASSWORD"}, + {"type":"text", + "label":"FingerPrint", + "required":false, + "env_variable":"KOPIA_SERVER_FINGERPRINT"} + ], + "supported_architectures": ["amd64"] +} diff --git a/apps/kopia/docker-compose.json b/apps/kopia/docker-compose.json new file mode 100644 index 0000000..9f7b0eb --- /dev/null +++ b/apps/kopia/docker-compose.json @@ -0,0 +1,112 @@ +{ + "schemaVersion": 2, + "services": [ + { + "name": "kopia", + "image": "kopia/kopia:latest", + "command": [ + "server", + "start", + "--disable-csrf-token-checks", + "--tls-cert-file=/app/certs/my.cert", + "--tls-key-file=/app/certs/my.key", + "--address=0.0.0.0:51515", + "--server-username=${USER}", + "--server-password=${PASSWORD}" + ], + "environment": [ + { + "key": "KOPIA_PASSWORD", + "value": "${PASSWORD}" + }, + { + "key": "KOPIA_SERVER_FINGERPRINT", + "value": "${KOPIA_SERVER_FINGERPRINT}" + }, + { + "key": "TZ", + "value": "${TZ}" + }, + { + "key": "USER", + "value": "${USER}" + } + ], + "addPorts": [ + {"containerPort":51515, + "hostPort":"$APP_PORT", + "tcp":true, + "udp":false} + ], + "volumes": [ + { + "hostPath": "/home/${USER}/kopia/certs", + "containerPath": "/app/certs", + "readOnly": false, + "shared": false, + "private": false + }, + { + "hostPath": "/home/${USER}/kopia/config", + "containerPath": "/app/config", + "readOnly": false, + "shared": false, + "private": false + }, + { + "hostPath": "/home/${USER}/kopia/cache", + "containerPath": "/app/cache", + "readOnly": false, + "shared": false, + "private": false + }, + { + "hostPath": "/home/${USER}/kopia/logs", + "containerPath": "/app/logs", + "readOnly": false, + "shared": false, + "private": false + }, + { + "hostPath": "/mnt/Backups/Kopia", + "containerPath": "/repository", + "readOnly": false, + "shared": false, + "private": false + }, + { + "hostPath": "/home/${USER}", + "containerPath": "/data/${USER}", + "readOnly": true, + "shared": false, + "private": false + }, + { + "hostPath": "/mnt/ext", + "containerPath": "/data/ext", + "readOnly": true, + "shared": false, + "private": false + }, + { + "hostPath": "/media/Cinema", + "containerPath": "/data/Cinema", + "readOnly": true, + "shared": false, + "private": false + } + ], + "hostname": "kopia-server", + "devices": [ + "/dev/fuse:/dev/fuse:rwm" + ], + "privileged": true, + "capAdd": [ + "SYS_ADMIN" + ], + "securityOpt": [ + "apparmor:unconfined" + ] + } + ] +} diff --git a/apps/minio_Potentially Edit/config.json b/apps/minio_Potentially Edit/config.json new file mode 100644 index 0000000..a23f21a --- /dev/null +++ b/apps/minio_Potentially Edit/config.json @@ -0,0 +1,43 @@ +{ + "name": "Minio", + "available": true, + "exposable": true, + "dynamic_config": true, + "port": 8001, + "id": "minio", + "description": "MinIO is a high-performance, S3 compatible object store. It is built for large scale AI/ML, data lake and database workloads.", + "tipi_version": 5, + "version": "RELEASE.2025-02-07T23-21-09Z", + "categories": ["development"], + "short_desc": "High Performance Object Storage.", + "author": "MinIO, Inc", + "source": "https://github.com/minio/minio", + "website": "https://min.io", + "form_fields": [ + { + "type": "text", + "label": "Minio root username", + "required": true, + "env_variable": "MINIO_ROOT_USER" + }, + { + "type": "password", + "label": "Minio root password", + "required": true, + "min": 8, + "env_variable": "MINIO_ROOT_PASSWORD" + }, + { + "type": "text", + "label": "Minio S3 API URL (when exposed)", + "placeholder": "s3.example.com", + "required": false, + "env_variable": "MINIO_API_URL" + } + ], + "supported_architectures": ["arm64", "amd64"], + "created_at": 1691943801422, + "updated_at": 1742038537982, + "$schema": "../app-info-schema.json", + "force_pull": false +} diff --git a/apps/minio_Potentially Edit/docker-compose.json b/apps/minio_Potentially Edit/docker-compose.json new file mode 100644 index 0000000..486bf7c --- /dev/null +++ b/apps/minio_Potentially Edit/docker-compose.json @@ -0,0 +1,50 @@ +{ + "$schema": "https://schemas.runtipi.io/dynamic-compose.json", + "services": [ + { + "name": "minio", + "image": "minio/minio:RELEASE.2025-02-07T23-21-09Z", + "isMain": true, + "internalPort": 9001, + "addPorts": [ + { + "hostPort": 8000, + "containerPort": 9000 + } + ], + "environment": { + "MINIO_ROOT_USER": "${MINIO_ROOT_USER}", + "MINIO_ROOT_PASSWORD": "${MINIO_ROOT_PASSWORD}", + "MINIO_BROWSER_REDIRECT_URL": "${APP_PROTOCOL:-http}://${APP_DOMAIN}", + "MINIO_DOMAIN": "${APP_DOMAIN}" + }, + "volumes": [ + { + "hostPath": "${APP_DATA_DIR}/data/minio/data", + "containerPath": "/data" + } + ], + "command": "server --console-address :9001 /data", + "extraLabels": { + "traefik.http.middlewares.{{RUNTIPI_APP_ID}}-api-web-redirect.redirectscheme.scheme": "https", + "traefik.http.services.{{RUNTIPI_APP_ID}}-api.loadbalancer.server.port": "9000", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-insecure.rule": "Host(`${MINIO_API_URL}`)", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-insecure.entrypoints": "web", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-insecure.service": "{{RUNTIPI_APP_ID}}-api", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-insecure.middlewares": "{{RUNTIPI_APP_ID}}-api-web-redirect", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api.rule": "Host(`${MINIO_API_URL}`)", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api.entrypoints": "websecure", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api.service": "{{RUNTIPI_APP_ID}}-api", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api.tls.certresolver": "myresolver", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-local-insecure.rule": "Host(`{{RUNTIPI_APP_ID}}-api.${LOCAL_DOMAIN}`)", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-local-insecure.entrypoints": "web", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-local-insecure.service": "{{RUNTIPI_APP_ID}}-api", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-local-insecure.middlewares": "{{RUNTIPI_APP_ID}}-api-web-redirect", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-local.rule": "Host(`{{RUNTIPI_APP_ID}}-api.${LOCAL_DOMAIN}`)", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-local.entrypoints": "websecure", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-local.service": "{{RUNTIPI_APP_ID}}-api", + "traefik.http.routers.{{RUNTIPI_APP_ID}}-api-local.tls": "true" + } + } + ] +} diff --git a/apps/minio_Potentially Edit/docker-compose.yml b/apps/minio_Potentially Edit/docker-compose.yml new file mode 100644 index 0000000..0bb41f8 --- /dev/null +++ b/apps/minio_Potentially Edit/docker-compose.yml @@ -0,0 +1,69 @@ +services: + minio: + container_name: minio + image: minio/minio:RELEASE.2025-02-07T23-21-09Z + restart: unless-stopped + environment: + - MINIO_ROOT_USER=${MINIO_ROOT_USER} + - MINIO_ROOT_PASSWORD=${MINIO_ROOT_PASSWORD} + - MINIO_BROWSER_REDIRECT_URL=${APP_PROTOCOL:-http}://${APP_DOMAIN} + - MINIO_DOMAIN=${APP_DOMAIN} + ports: + - 8000:9000 + - ${APP_PORT}:9001 + volumes: + - ${APP_DATA_DIR}/data/minio/data:/data + networks: + - tipi_main_network + command: "server --console-address :9001 /data" + labels: + # Main + traefik.enable: true + # Console + traefik.http.middlewares.minio-console-web-redirect.redirectscheme.scheme: https + traefik.http.services.minio-console.loadbalancer.server.port: 9001 + # API + traefik.http.middlewares.minio-api-web-redirect.redirectscheme.scheme: https + traefik.http.services.minio-api.loadbalancer.server.port: 9000 + # Web - Console + traefik.http.routers.minio-console-insecure.rule: Host(`${APP_DOMAIN}`) + traefik.http.routers.minio-console-insecure.entrypoints: web + traefik.http.routers.minio-console-insecure.service: minio-console + traefik.http.routers.minio-console-insecure.middlewares: minio-console-web-redirect + # Web - API + traefik.http.routers.minio-api-insecure.rule: Host(`${MINIO_API_URL}`) + traefik.http.routers.minio-api-insecure.entrypoints: web + traefik.http.routers.minio-api-insecure.service: minio-api + traefik.http.routers.minio-api-insecure.middlewares: minio-api-web-redirect + # Websecure - Console + traefik.http.routers.minio-console.rule: Host(`${APP_DOMAIN}`) + traefik.http.routers.minio-console.entrypoints: websecure + traefik.http.routers.minio-console.service: minio-console + traefik.http.routers.minio-console.tls.certresolver: myresolver + # Websecure - API + traefik.http.routers.minio-api.rule: Host(`${MINIO_API_URL}`) + traefik.http.routers.minio-api.entrypoints: websecure + traefik.http.routers.minio-api.service: minio-api + traefik.http.routers.minio-api.tls.certresolver: myresolver + # Local domain - Console + traefik.http.routers.minio-console-local-insecure.rule: Host(`minio.${LOCAL_DOMAIN}`) + traefik.http.routers.minio-console-local-insecure.entrypoints: web + traefik.http.routers.minio-console-local-insecure.service: minio-console + traefik.http.routers.minio-console-local-insecure.middlewares: minio-console-web-redirect + # Local domain - API + traefik.http.routers.minio-api-local-insecure.rule: Host(`minio-api.${LOCAL_DOMAIN}`) + traefik.http.routers.minio-api-local-insecure.entrypoints: web + traefik.http.routers.minio-api-local-insecure.service: minio-api + traefik.http.routers.minio-api-local-insecure.middlewares: minio-api-web-redirect + # Local domain secure - Console + traefik.http.routers.minio-console-local.rule: Host(`minio.${LOCAL_DOMAIN}`) + traefik.http.routers.minio-console-local.entrypoints: websecure + traefik.http.routers.minio-console-local.service: minio-console + traefik.http.routers.minio-console-local.tls: true + # Local domain secure - API + traefik.http.routers.minio-api-local.rule: Host(`minio-api.${LOCAL_DOMAIN}`) + traefik.http.routers.minio-api-local.entrypoints: websecure + traefik.http.routers.minio-api-local.service: minio-api + traefik.http.routers.minio-api-local.tls: true + # Runtipi managed + runtipi.managed: true diff --git a/apps/minio_Potentially Edit/metadata/description.md b/apps/minio_Potentially Edit/metadata/description.md new file mode 100644 index 0000000..65d45f7 --- /dev/null +++ b/apps/minio_Potentially Edit/metadata/description.md @@ -0,0 +1,28 @@ +# Minio - Open Source Object Storage + +Host your own Amazon S3 compatible Object Storage! + +## Installation Info + +This MinIO setup requires two exposed URLS + +- The console URL +- The S3 API URL + +While the console URL is defined by the runtipi's default expose URL you need to define the S3 API URL in the settings when installing. + +**Example:** + +- API: Minio S3 API URL: s3.mydomain.com +- Console: Exposed URL: minio.mydomain.com + +| Exposed Service | Local Port | Exposed Domain | +| ------------------- | ---------- | ------------------ | +| Minio API | IP:8000 | s3.mydomain.com | +| Minio Admin Console | IP:8001 | minio.mydomain.com | + +## About + +MinIO is a High Performance Object Storage released under GNU Affero General Public License v3.0. It is API compatible with Amazon S3 cloud storage service. Use MinIO to build high performance infrastructure for machine learning, analytics and application data workloads. + +See https://github.com/minio/minio for more details diff --git a/apps/minio_Potentially Edit/metadata/logo.jpg b/apps/minio_Potentially Edit/metadata/logo.jpg new file mode 100644 index 0000000000000000000000000000000000000000..93303cb3ed15bbebbb5e3f1785e4c8721900faf0 GIT binary patch literal 40144 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 [!IMPORTANT] +> If you got `permission denied` error in `Logs` after installation +> +> You need to +> 1. Execute the following command on your server: +> `sudo chown -R 1000:1000 ~/runtipi/app-data/siyuan/data/workspace` +> +> 2. Restart SiYuan +> +> For the reason, please refer to [User Permissions](#user-permissions) section + +

+SiYuan +
+Refactor your thinking +

+ + + +
+ + + +
+ + + +
+ + +

+Twitter Follow +Chat on Discord +

+SiYuan - A privacy-first personal knowledge management software | Product Hunt +

+ +

+中文 +

+ +--- + +## Table of Contents + +- [Table of Contents](#table-of-contents) +- [💡 Introduction](#-introduction) +- [🔮 Features](#-features) +- [🏗️ Architecture and Ecosystem](#️-architecture-and-ecosystem) +- [🌟 Star History](#-star-history) +- [🗺️ Roadmap](#️-roadmap) +- [🚀 Download Setup](#-download-setup) + - [App Market](#app-market) + - [Installation Package](#installation-package) + - [Docker Hosting](#docker-hosting) + - [Overview](#overview) + - [File structure](#file-structure) + - [Entrypoint](#entrypoint) + - [User permissions](#user-permissions) + - [Hidden port](#hidden-port) + - [Note](#note) + - [Limitations](#limitations) + - [Unraid Hosting](#unraid-hosting) + - [Insider Preview](#insider-preview) +- [🏘️ Community](#️-community) +- [🛠️ Development Guide](#️-development-guide) +- [❓ FAQ](#-faq) + - [How does SiYuan store data?](#how-does-siyuan-store-data) + - [Does it support data synchronization through a third-party sync disk?](#does-it-support-data-synchronization-through-a-third-party-sync-disk) + - [Is SiYuan open source?](#is-siyuan-open-source) + - [How to upgrade to a new version?](#how-to-upgrade-to-a-new-version) + - [What if some blocks (such as paragraph blocks in list items) cannot find the block icon?](#what-if-some-blocks-such-as-paragraph-blocks-in-list-items-cannot-find-the-block-icon) + - [What should I do if the data repo key is lost?](#what-should-i-do-if-the-data-repo-key-is-lost) + - [Do I need to pay for it?](#do-i-need-to-pay-for-it) +- [🙏 Acknowledgement](#-acknowledgement) + - [Contributors](#contributors) + +--- + +## 💡 Introduction + +SiYuan is a privacy-first personal knowledge management system, support fine-grained block-level reference and Markdown +WYSIWYG. + +Welcome to [SiYuan English Discussion Forum](https://liuyun.io) to learn more. + +![feature0.png](https://b3logfile.com/file/2024/01/feature0-1orBRlI.png) + +![feature51.png](https://b3logfile.com/file/2024/02/feature5-1-uYYjAqy.png) + +## 🔮 Features + +Most features are free, even for commercial use. + +* Content block + * Block-level reference and two-way links + * Custom attributes + * SQL query embed + * Protocol `siyuan://` +* Editor + * Block-style + * Markdown WYSIWYG + * List outline + * Block zoom-in + * Million-word large document editing + * Mathematical formulas, charts, flowcharts, Gantt charts, timing charts, staffs, etc. + * Web clipping + * PDF Annotation link +* Export + * Block ref and embed + * Standard Markdown with assets + * PDF, Word and HTML + * Copy to WeChat MP, Zhihu and Yuque +* Database + * Table view +* Flashcard spaced repetition +* AI writing and Q/A chat via OpenAI API +* Tesseract OCR +* Multi-tab, drag and drop to split screen +* Template snippet +* JavaScript/CSS snippet +* Android/iOS App +* Docker deployment +* [API](https://github.com/siyuan-note/siyuan/blob/master/API.md) +* Community marketplace + +Some features are only available to paid members, for more details please refer to [Pricing](https://b3log.org/siyuan/en/pricing.html). + +## 🏗️ Architecture and Ecosystem + +![SiYuan Arch](https://b3logfile.com/file/2023/05/SiYuan_Arch-Sgu8vXT.png "SiYuan Arch") + +| Project | Description | Forks | Stars | +|----------------------------------------------------------|-----------------------|---------------------------------------------------------------------------------|--------------------------------------------------------------------------------------| +| [lute](https://github.com/88250/lute) | Editor engine | ![GitHub forks](https://img.shields.io/github/forks/88250/lute) | ![GitHub Repo stars](https://img.shields.io/github/stars/88250/lute) | +| [chrome](https://github.com/siyuan-note/siyuan-chrome) | Chrome/Edge extension | ![GitHub forks](https://img.shields.io/github/forks/siyuan-note/siyuan-chrome) | ![GitHub Repo stars](https://img.shields.io/github/stars/siyuan-note/siyuan-chrome) | +| [bazaar](https://github.com/siyuan-note/bazaar) | Community marketplace | ![GitHub forks](https://img.shields.io/github/forks/siyuan-note/bazaar) | ![GitHub Repo stars](https://img.shields.io/github/stars/siyuan-note/bazaar) | +| [dejavu](https://github.com/siyuan-note/dejavu) | Data repo | ![GitHub forks](https://img.shields.io/github/forks/siyuan-note/dejavu) | ![GitHub Repo stars](https://img.shields.io/github/stars/siyuan-note/dejavu) | +| [petal](https://github.com/siyuan-note/petal) | Plugin API | ![GitHub forks](https://img.shields.io/github/forks/siyuan-note/petal) | ![GitHub Repo stars](https://img.shields.io/github/stars/siyuan-note/petal) | +| [android](https://github.com/siyuan-note/siyuan-android) | Android App | ![GitHub forks](https://img.shields.io/github/forks/siyuan-note/siyuan-android) | ![GitHub Repo stars](https://img.shields.io/github/stars/siyuan-note/siyuan-android) | +| [ios](https://github.com/siyuan-note/siyuan-ios) | iOS App | ![GitHub forks](https://img.shields.io/github/forks/siyuan-note/siyuan-ios) | ![GitHub Repo stars](https://img.shields.io/github/stars/siyuan-note/siyuan-ios) | +| [riff](https://github.com/siyuan-note/riff) | Spaced repetition | ![GitHub forks](https://img.shields.io/github/forks/siyuan-note/riff) | ![GitHub Repo stars](https://img.shields.io/github/stars/siyuan-note/riff) | + +## 🌟 Star History + +[![Star History Chart](https://api.star-history.com/svg?repos=siyuan-note/siyuan&type=Date)](https://star-history.com/#siyuan-note/siyuan&Date) + +## 🗺️ Roadmap + +* [SiYuan development plan and progress](https://github.com/orgs/siyuan-note/projects/1) +* [SiYuan changelog](CHANGELOG.md) + +## 🚀 Download Setup + +It is recommended to give priority to installing through the application market on the desktop and mobile, so that you can upgrade the version with one click in the future. + +### App Market + +Mobile: + +* [App Store](https://apps.apple.com/us/app/siyuan/id1583226508) +* [Google Play](https://play.google.com/store/apps/details?id=org.b3log.siyuan) +* [F-Droid](https://f-droid.org/packages/org.b3log.siyuan) + +Desktop: + +* [Microsoft Store](https://apps.microsoft.com/detail/9p7hpmxp73k4) + +### Installation Package + +* [B3log](https://b3log.org/siyuan/en/download.html) +* [GitHub](https://github.com/siyuan-note/siyuan/releases) + +### Docker Hosting + +
+Docker Deployment + +#### Overview + +The easiest way to serve SiYuan on a server is to deploy it through Docker. + +* Image name `b3log/siyuan` +* [Image URL](https://hub.docker.com/r/b3log/siyuan) + +#### File structure + +The overall program is located under `/opt/siyuan/`, which is basically the structure under the resources folder of the Electron installation package: + +* appearance: icon, theme, languages +* guide: user guide document +* stage: interface and static resources +* kernel: kernel program + +#### Entrypoint + +The entry point is set when building the Docker image: `ENTRYPOINT ["/opt/siyuan/kernel" ]`, use `docker run b3log/siyuan` with parameters to start: + +* `--workspace`: Specifies the workspace folder path, mounted to the container via `-v` on the host +* `--accessAuthCode`: Specifies the access authorization code + +More parameters can refer to `--help`. The following is an example of a startup command: + +``` +docker run -d -v workspace_dir_host:workspace_dir_container -p 6806:6806 b3log/siyuan --workspace=workspace_dir_container --accessAuthCode=xxx +``` + +* `workspace_dir_host`: The workspace folder path on the host +* `workspace_dir_container`: The path of the workspace folder in the container, which is the same as specified in `--workspace` +* `accessAuthCode`: Access authorization code, please **be sure to modify**, otherwise anyone can read and write your data + +To simplify, it is recommended to configure the workspace folder path to be consistent on the host and container, such as: `workspace_dir_host` and `workspace_dir_container` are configured as `/siyuan/workspace`, the corresponding startup commands is: + +``` +docker run -d -v /siyuan/workspace:/siyuan/workspace -p 6806:6806 -u 1000:1000 b3log/siyuan --workspace=/siyuan/workspace/ --accessAuthCode=xxx +``` + +Alternatively, see below for an example Docker Compose file: + +``` +version: "3.9" +services: + main: + image: b3log/siyuan + command: ['--workspace=/siyuan/workspace/', '--accessAuthCode=${AuthCode}'] + user: '1000:1000' + ports: + - 6806:6806 + volumes: + - /siyuan/workspace:/siyuan/workspace + restart: unless-stopped + environment: + # A list of time zone identifiers can be found at https://en.wikipedia.org/wiki/List_of_tz_database_time_zones + - TZ=${TimeZone} +``` + +#### User permissions + +In the image, the normal user `siyuan` (uid 1000/gid 1000) created by default is used to start the kernel process. Therefore, when the host creates a workspace folder, please pay attention to setting the user group of the folder: `chown -R 1000:1000 /siyuan/workspace`. The parameter `-u 1000:1000` is required when starting the container. + +#### Hidden port + +Use NGINX reverse proxy to hide port 6806, please note: + +* Configure WebSocket reverse proxy `/ws` + +#### Note + +* Be sure to confirm the correctness of the mounted volume, otherwise the data will be lost after the container is deleted +* Do not use URL rewriting for redirection, otherwise there may be problems with authentication, it is recommended to configure a reverse proxy + +#### Limitations + +* Does not support desktop and mobile application connections, only supports use on browsers +* Export to PDF, HTML and Word formats is not supported +* Import Markdown file is not supported + +
+ +### Unraid Hosting + +
+Unraid Deployment + +Note: First run `chown -R 1000:1000 /mnt/user/appdata/siyuan` in the terminal + +Template reference: + +``` +Web UI: 6806 +Container Port: 6806 +Container Path: /home/siyuan +Host path: /mnt/user/appdata/siyuan +PUID: 1000 +PGID: 1000 +Publish parameters: --accessAuthCode=******(Access authorization code) +``` + +
+ +### Insider Preview + +We release insider preview before major updates, please visit [https://github.com/siyuan-note/insider](https://github.com/siyuan-note/insider). + +## 🏘️ Community + +* [English Discussion Forum](https://liuyun.io) +* [User community summary](https://liuyun.io/article/1687779743723) +* [Awesome SiYuan](https://github.com/siyuan-note/awesome) + +## 🛠️ Development Guide + +See [Development Guide](https://github.com/siyuan-note/siyuan/blob/master/.github/CONTRIBUTING.md). + +## ❓ FAQ + +### How does SiYuan store data? + +The data is saved in the workspace folder, in the workspace data folder: + +* `assets` is used to save all inserted assets +* `emojis` is used to save emoji images +* `snippets` is used to save code snippets +* `storage` is used to save query conditions, layouts and flashcards, etc. +* `templates` is used to save template snippets +* `widgets` is used to save widgets +* `plugins` is used to save plugins +* `public` is used to save public data +* The rest of the folders are the notebook folders created by the user, files with the suffix of `.sy` in the notebook folder are used to save the document data, and the data format is JSON + +### Does it support data synchronization through a third-party sync disk? + +Data synchronization through third-party synchronization disks is not supported, otherwise data may be corrupted. + +Although it does not support third-party sync disks, it supports connect with third-party cloud storage (Member's privileges). + +In addition, you can also consider manually exporting and importing data to achieve data synchronization: + +* Desktop: Settings - Export - Export Data / Import Data +* Mobile: Right column - About - Export Data / Import Data + +### Is SiYuan open source? + +SiYuan is completely open source, and contributions are welcome: + +* [User Interface and Kernel](https://github.com/siyuan-note/siyuan) +* [Android](https://github.com/siyuan-note/siyuan-android) +* [iOS](https://github.com/siyuan-note/siyuan-ios) +* [Chrome Clipping Extension](https://github.com/siyuan-note/siyuan-chrome) + +For more details, please refer to [Development Guide](https://github.com/siyuan-note/siyuan/blob/master/.github/CONTRIBUTING.md). + +### How to upgrade to a new version? + +* If installed via app store, please update via app store +* If it is installed through the installation package on the desktop, you can open the option of Settings - About - Automatically download update installation package, so that SiYuan will automatically download The latest version of the installation package and prompts to install +* If it is installed by manual installation package, please download the installation package again to install + +You can Check update in Settings - About - Current Version, or pay attention to [Official Download](https://b3log.org/siyuan/en/download.html) or [GitHub Releases](https://github.com/siyuan-note/siyuan/releases) to get the new version. + +### What if some blocks (such as paragraph blocks in list items) cannot find the block icon? + +The first sub-block under the list item is the block icon omitted. You can move the cursor into this block and trigger its block menu with Ctrl+/ . + +### What should I do if the data repo key is lost? + +* If the data repo key is correctly initialized on multiple devices before, the key is the same on all devices and can be set in Settings - About - Data repo key - Copy key string retrieve +* If it has not been configured correctly before (for example, the keys on multiple devices are inconsistent) or all devices are unavailable and the key string cannot be obtained, you can reset the key by following the steps below: + + 1. Manually back up the data, you can use Export Data or directly copy the workspace/data/ folder on the file system + 2. Settings - About - Data rep key - Reset data repo + 3. Reinitialize the data repo key. After initializing the key on one device, other devices import the key + 4. The cloud uses the new synchronization directory, the old synchronization directory is no longer available and can be deleted + 5. The existing cloud snapshots are no longer available and can be deleted + +### Do I need to pay for it? + +Most features are free, even for commercial use. + +Member's privileges can only be used after payment, please refer to [Pricing](https://b3log.org/siyuan/en/pricing.html). + +## 🙏 Acknowledgement + +The birth of SiYuan is inseparable from many open source projects and contributors, please refer to the project source code kernel/go.mod, app/package.json and project homepage. + +The growth of SiYuan is inseparable from user feedback and promotion, thank you for everyone's help to SiYuan ❤️ + +### Contributors + +Welcome to join us and contribute code to SiYuan together. + + + + \ No newline at end of file diff --git a/apps/siyuan_Need to edit/metadata/logo.jpg b/apps/siyuan_Need to edit/metadata/logo.jpg new file mode 100644 index 0000000000000000000000000000000000000000..734ebba0748f4b7b312a937729141491c40cc3ae GIT binary patch literal 29131 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 A0ssI2 literal 0 HcmV?d00001 diff --git a/apps/whoami/config.json b/apps/whoami/config.json new file mode 100644 index 0000000..c0bee99 --- /dev/null +++ b/apps/whoami/config.json @@ -0,0 +1,24 @@ +{ + "name": "Whoami", + "id": "whoami", + "available": true, + "short_desc": "Tiny Go server that prints os information and HTTP request to output.", + "author": "traefik", + "port": 8382, + "categories": [ + "utilities" + ], + "description": "Tiny Go webserver that prints OS information and HTTP request to output.", + "tipi_version": 2, + "version": "v1.11.0", + "source": "https://github.com/traefik/whoami", + "exposable": true, + "supported_architectures": [ + "arm64", + "amd64" + ], + "created_at": 1745082405284, + "updated_at": 1745674974072, + "dynamic_config": true, + "form_fields": [] +} \ No newline at end of file diff --git a/apps/whoami/docker-compose.json b/apps/whoami/docker-compose.json new file mode 100644 index 0000000..2f6c4a1 --- /dev/null +++ b/apps/whoami/docker-compose.json @@ -0,0 +1,10 @@ +{ + "services": [ + { + "name": "whoami", + "image": "traefik/whoami:v1.11.0", + "isMain": true, + "internalPort": "80" + } + ] +} diff --git a/apps/whoami/docker-compose.yml b/apps/whoami/docker-compose.yml new file mode 100644 index 0000000..f975633 --- /dev/null +++ b/apps/whoami/docker-compose.yml @@ -0,0 +1,34 @@ +services: + whoami: + image: traefik/whoami:v1.11.0 + container_name: whoami + ports: + - ${APP_PORT}:80 + networks: + - tipi_main_network + labels: + # Main + traefik.enable: true + traefik.http.middlewares.whoami-web-redirect.redirectscheme.scheme: https + traefik.http.services.whoami.loadbalancer.server.port: 80 + # Web + traefik.http.routers.whoami-insecure.rule: Host(`${APP_DOMAIN}`) + traefik.http.routers.whoami-insecure.entrypoints: web + traefik.http.routers.whoami-insecure.service: whoami + traefik.http.routers.whoami-insecure.middlewares: whoami-web-redirect + # Websecure + traefik.http.routers.whoami.rule: Host(`${APP_DOMAIN}`) + traefik.http.routers.whoami.entrypoints: websecure + traefik.http.routers.whoami.service: whoami + traefik.http.routers.whoami.tls.certresolver: myresolver + # Local domain + traefik.http.routers.whoami-local-insecure.rule: Host(`whoami.${LOCAL_DOMAIN}`) + traefik.http.routers.whoami-local-insecure.entrypoints: web + traefik.http.routers.whoami-local-insecure.service: whoami + traefik.http.routers.whoami-local-insecure.middlewares: whoami-web-redirect + # Local domain secure + traefik.http.routers.whoami-local.rule: Host(`whoami.${LOCAL_DOMAIN}`) + traefik.http.routers.whoami-local.entrypoints: websecure + traefik.http.routers.whoami-local.service: whoami + traefik.http.routers.whoami-local.tls: true + runtipi.managed: true diff --git a/apps/whoami/metadata/description.md b/apps/whoami/metadata/description.md new file mode 100644 index 0000000..27cd734 --- /dev/null +++ b/apps/whoami/metadata/description.md @@ -0,0 +1,43 @@ +# Whoami + +Tiny Go webserver that prints OS information and HTTP request to output. + +## Usage + +### Paths + +#### `/[?wait=d]` + +Returns the whoami information (request and network information). + +The optional `wait` query parameter can be provided to tell the server to wait before sending the response. +The duration is expected in Go's [`time.Duration`](https://golang.org/pkg/time/#ParseDuration) format (e.g. `/?wait=100ms` to wait 100 milliseconds). + +The optional `env` query parameter can be set to `true` to add the environment variables to the response. + +#### `/api` + +Returns the whoami information (and some extra information) as JSON. + +The optional `env` query parameter can be set to `true` to add the environment variables to the response. + +#### `/bench` + +Always return the same response (`1`). + +#### `/data?size=n[&unit=u]` + +Creates a response with a size `n`. + +The unit of measure, if specified, accepts the following values: `KB`, `MB`, `GB`, `TB` (optional, default: bytes). + +#### `/echo` + +WebSocket echo. + +#### `/health` + +Heath check. + +- `GET`, `HEAD`, ...: returns a response with the status code defined by the `POST` +- `POST`: changes the status code of the `GET` (`HEAD`, ...) response. diff --git a/apps/whoami/metadata/logo.jpg b/apps/whoami/metadata/logo.jpg new file mode 100644 index 0000000000000000000000000000000000000000..24ed99d3f04b8a10fdaf3708a8b0bdf3ee296b58 GIT binary patch literal 18692 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 literal 0 HcmV?d00001 diff --git a/bun.lockb b/bun.lockb new file mode 100755 index 0000000000000000000000000000000000000000..a28598447958537019db5053a477e7276aeca197 GIT binary patch literal 9101 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 literal 0 HcmV?d00001 diff --git a/config.js b/config.js new file mode 100644 index 0000000..9a95d69 --- /dev/null +++ b/config.js @@ -0,0 +1,3 @@ +export default { + allowedCommands: ["bun ./scripts/update-config.ts", "bun install && bun run test"], +}; diff --git a/package.json b/package.json new file mode 100644 index 0000000..925fe34 --- /dev/null +++ b/package.json @@ -0,0 +1,25 @@ +{ + "name": "example-appstore", + "version": "1.0.0", + "description": "", + "main": "index.js", + "type": "module", + "scripts": { + "test": "bun test" + }, + "keywords": [], + "author": "", + "license": "ISC", + "devDependencies": { + "@types/bun": "latest", + "@types/node": "^22.14.1" + }, + "dependencies": { + "@runtipi/common": "^0.8.0", + "bun": "^1.2.10", + "zod-validation-error": "^3.4.0" + }, + "peerDependencies": { + "typescript": "^5.0.0" + } +} diff --git a/renovate.json b/renovate.json new file mode 100644 index 0000000..936fd91 --- /dev/null +++ b/renovate.json @@ -0,0 +1,62 @@ +{ + "$schema": "https://docs.renovatebot.com/renovate-schema.json", + "automerge": false, + "extends": [ + "config:recommended" + ], + "addLabels": [ + "renovate" + ], + "enabledManagers": ["regex"], + "automergeStrategy": "rebase", + "customManagers": [ + { + "customType": "regex", + "fileMatch": [ + "^.*docker-compose\\.json$" + ], + "matchStrings": [ + "\"image\": \"(?.*?):(?.*?)\"," + ], + "datasourceTemplate": "docker" + } + ], + "packageRules": [ + { + "matchUpdateTypes": [ + "minor", + "major", + "patch", + "pin", + "digest" + ], + "automerge": false + }, + { + "matchDepTypes": [ + "devDependencies" + ], + "automerge": false + }, + { + "matchPackageNames": [ + "mariadb", + "mysql", + "monogdb", + "postgres", + "redis" + ], + "enabled": false + } + ], + "postUpgradeTasks": { + "commands": [ + "bun ./scripts/update-config.ts {{packageFile}} {{newVersion}}", + "bun install && bun run test" + ], + "fileFilters": [ + "**/*" + ], + "executionMode": "update" + } +} diff --git a/scripts/update-config.ts b/scripts/update-config.ts new file mode 100644 index 0000000..42891c7 --- /dev/null +++ b/scripts/update-config.ts @@ -0,0 +1,35 @@ +import path from "node:path"; +import fs from "fs/promises"; + +const packageFile = process.argv[2]; +const newVersion = process.argv[3]; + +type AppConfig = { + tipi_version: string; + version: string; + updated_at: number; +}; + +const updateAppConfig = async (packageFile: string, newVersion: string) => { + try { + const packageRoot = path.dirname(packageFile); + const configPath = path.join(packageRoot, "config.json"); + + const config = await fs.readFile(configPath, "utf-8"); + const configParsed = JSON.parse(config) as AppConfig; + + configParsed.tipi_version = configParsed.tipi_version + 1; + configParsed.version = newVersion; + configParsed.updated_at = new Date().getTime(); + + await fs.writeFile(configPath, JSON.stringify(configParsed, null, 2)); + } catch (e) { + console.error(`Failed to update app config, error: ${e}`); + } +}; + +if (!packageFile || !newVersion) { + console.error("Usage: node update-config.js "); + process.exit(1); +} +updateAppConfig(packageFile, newVersion); diff --git a/tsconfig.json b/tsconfig.json new file mode 100644 index 0000000..01ef57a --- /dev/null +++ b/tsconfig.json @@ -0,0 +1,24 @@ +{ + "compilerOptions": { + "esModuleInterop": true, + "skipLibCheck": true, + "target": "es2022", + "allowJs": true, + "resolveJsonModule": true, + "moduleDetection": "force", + "isolatedModules": true, + "verbatimModuleSyntax": true, + "strict": true, + "noUncheckedIndexedAccess": true, + "noImplicitOverride": true, + "module": "NodeNext", + "outDir": "dist", + "sourceMap": true, + "lib": [ + "es2022" + ] + }, + "include": [ + "**/*.ts", + ], +}