Compare commits
19 commits
| Author | SHA1 | Date | |
|---|---|---|---|
|
|
7d80b21c1b | ||
|
|
7e6d8517b0 | ||
|
|
370ea5dae9 | ||
|
|
8874bc604c | ||
|
|
81fb160eb5 | ||
|
|
cd66562e9d | ||
|
|
5eb4a2908d | ||
|
|
512edd4e42 | ||
|
|
03e9c9fb13 | ||
|
|
d71fbe6c75 | ||
|
|
c4129189bb | ||
|
|
be5e2a5e7c | ||
|
|
c50bc8601d | ||
|
|
463bb4e306 | ||
|
|
4c3c4805c8 | ||
|
|
a81ac47be9 | ||
|
|
36b909d4a4 | ||
|
|
7b552f5080 | ||
|
|
0f767d14bb |
84 changed files with 1155 additions and 225 deletions
4
.github/workflows/crowdin.yml
vendored
4
.github/workflows/crowdin.yml
vendored
|
|
@ -17,9 +17,9 @@ jobs:
|
|||
|
||||
steps:
|
||||
- name: Checkout
|
||||
uses: actions/checkout@v6
|
||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||
- name: crowdin action
|
||||
uses: crowdin/github-action@v2
|
||||
uses: crowdin/github-action@7ca9c452bfe9197d3bb7fa83a4d7e2b0c9ae835d # v2
|
||||
with:
|
||||
upload_translations: false
|
||||
download_translations: true
|
||||
|
|
|
|||
52
.github/workflows/docker-publish.yml
vendored
52
.github/workflows/docker-publish.yml
vendored
|
|
@ -17,44 +17,12 @@ env:
|
|||
IMAGE_NAME: ${{ github.repository }}
|
||||
|
||||
jobs:
|
||||
pre-commit:
|
||||
name: Linting Checks
|
||||
runs-on: ubuntu-22.04
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v6
|
||||
|
||||
- name: Install python
|
||||
uses: actions/setup-python@v6
|
||||
with:
|
||||
python-version: 3.x
|
||||
|
||||
- name: Check files
|
||||
uses: pre-commit/action@v3.0.1
|
||||
|
||||
- name: Install pnpm
|
||||
uses: pnpm/action-setup@v5
|
||||
with:
|
||||
version: 10
|
||||
run_install: false
|
||||
|
||||
- name: Setup Node.js
|
||||
uses: actions/setup-node@v6
|
||||
with:
|
||||
node-version: 24
|
||||
cache: 'pnpm'
|
||||
|
||||
- name: Install dependencies
|
||||
run: pnpm install
|
||||
|
||||
- name: Lint frontend
|
||||
run: pnpm run lint
|
||||
|
||||
build:
|
||||
name: Docker Build & Push
|
||||
if: github.repository == 'gethomepage/homepage'
|
||||
runs-on: ubuntu-22.04
|
||||
needs: [ pre-commit ]
|
||||
permissions:
|
||||
contents: read
|
||||
packages: write
|
||||
|
|
@ -62,11 +30,11 @@ jobs:
|
|||
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v6
|
||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||
|
||||
- name: Extract Docker metadata
|
||||
id: meta
|
||||
uses: docker/metadata-action@v6
|
||||
uses: docker/metadata-action@030e881283bb7a6894de51c315a6bfe6a94e05cf # v6
|
||||
with:
|
||||
images: |
|
||||
${{ env.IMAGE_NAME }}
|
||||
|
|
@ -84,7 +52,7 @@ jobs:
|
|||
latest=auto
|
||||
|
||||
- name: Next.js build cache
|
||||
uses: actions/cache@v5
|
||||
uses: actions/cache@668228422ae6a00e4ad889ee87cd7109ec5666a7 # v5
|
||||
with:
|
||||
path: .next/cache
|
||||
key: nextjs-${{ runner.os }}-${{ hashFiles('pnpm-lock.yaml') }}-${{ hashFiles('**/*.js', '**/*.jsx') }}
|
||||
|
|
@ -92,13 +60,13 @@ jobs:
|
|||
nextjs-${{ runner.os }}-${{ hashFiles('pnpm-lock.yaml') }}
|
||||
|
||||
- name: Install pnpm
|
||||
uses: pnpm/action-setup@v5
|
||||
uses: pnpm/action-setup@a8198c4bff370c8506180b035930dea56dbd5288 # v5
|
||||
with:
|
||||
version: 10
|
||||
run_install: false
|
||||
|
||||
- name: Setup Node.js
|
||||
uses: actions/setup-node@v6
|
||||
uses: actions/setup-node@53b83947a5a98c8d113130e565377fae1a50d02f # v6
|
||||
with:
|
||||
node-version: 24
|
||||
cache: 'pnpm'
|
||||
|
|
@ -115,7 +83,7 @@ jobs:
|
|||
|
||||
- name: Log into registry ${{ env.REGISTRY }}
|
||||
if: github.event_name != 'pull_request'
|
||||
uses: docker/login-action@v4
|
||||
uses: docker/login-action@b45d80f862d83dbcd57f89517bcf500b2ab88fb2 # v4
|
||||
with:
|
||||
registry: ghcr.io
|
||||
username: ${{ github.actor }}
|
||||
|
|
@ -123,20 +91,20 @@ jobs:
|
|||
|
||||
- name: Login to Docker Hub
|
||||
if: github.event_name != 'pull_request'
|
||||
uses: docker/login-action@v4
|
||||
uses: docker/login-action@b45d80f862d83dbcd57f89517bcf500b2ab88fb2 # v4
|
||||
with:
|
||||
username: ${{ secrets.DOCKERHUB_USERNAME }}
|
||||
password: ${{ secrets.DOCKERHUB_TOKEN }}
|
||||
|
||||
- name: Setup QEMU
|
||||
uses: docker/setup-qemu-action@v4.0.0
|
||||
uses: docker/setup-qemu-action@ce360397dd3f832beb865e1373c09c0e9f86d70a # v4.0.0
|
||||
|
||||
- name: Setup Docker buildx
|
||||
uses: docker/setup-buildx-action@v4
|
||||
uses: docker/setup-buildx-action@4d04d5d9486b7bd6fa91e7baf45bbb4f8b9deedd # v4
|
||||
|
||||
- name: Build and push Docker image
|
||||
id: build-and-push
|
||||
uses: docker/build-push-action@v7
|
||||
uses: docker/build-push-action@d08e5c354a6adb9ed34480a06d141179aa583294 # v7
|
||||
with:
|
||||
context: .
|
||||
push: ${{ github.event_name != 'pull_request' }}
|
||||
|
|
|
|||
34
.github/workflows/docs-publish.yml
vendored
34
.github/workflows/docs-publish.yml
vendored
|
|
@ -14,32 +14,18 @@ permissions:
|
|||
id-token: write
|
||||
|
||||
jobs:
|
||||
pre-commit:
|
||||
name: Linting Checks
|
||||
runs-on: ubuntu-22.04
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@v6
|
||||
- name: Install python
|
||||
uses: actions/setup-python@v6
|
||||
with:
|
||||
python-version: 3.x
|
||||
- name: Check files
|
||||
uses: pre-commit/action@v3.0.1
|
||||
|
||||
test:
|
||||
name: Test Build Docs
|
||||
if: github.repository == 'gethomepage/homepage' && github.event_name == 'pull_request'
|
||||
runs-on: ubuntu-latest
|
||||
needs:
|
||||
- pre-commit
|
||||
steps:
|
||||
- uses: actions/checkout@v6
|
||||
- uses: actions/setup-python@v6
|
||||
- uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||
- uses: actions/setup-python@a309ff8b426b58ec0e2a45f0f869d46889d02405 # v6
|
||||
with:
|
||||
python-version-file: ".python-version"
|
||||
- name: Install uv
|
||||
uses: astral-sh/setup-uv@v7
|
||||
uses: astral-sh/setup-uv@94527f2e458b27549849d47d273a16bec83a01e9 # v7
|
||||
- run: sudo apt-get install pngquant
|
||||
- name: Test Docs Build
|
||||
run: uv run --frozen zensical build --clean
|
||||
|
|
@ -50,21 +36,19 @@ jobs:
|
|||
environment:
|
||||
name: github-pages
|
||||
url: ${{ steps.deployment.outputs.page_url }}
|
||||
needs:
|
||||
- pre-commit
|
||||
steps:
|
||||
- uses: actions/configure-pages@v5
|
||||
- uses: actions/checkout@v6
|
||||
- uses: actions/setup-python@v6
|
||||
- uses: actions/configure-pages@983d7736d9b0ae728b81ab479565c72886d7745b # v5
|
||||
- uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||
- uses: actions/setup-python@a309ff8b426b58ec0e2a45f0f869d46889d02405 # v6
|
||||
with:
|
||||
python-version-file: ".python-version"
|
||||
- name: Install uv
|
||||
uses: astral-sh/setup-uv@v7
|
||||
uses: astral-sh/setup-uv@94527f2e458b27549849d47d273a16bec83a01e9 # v7
|
||||
- run: sudo apt-get install pngquant
|
||||
- name: Build Docs
|
||||
run: uv run --frozen zensical build --clean
|
||||
- uses: actions/upload-pages-artifact@v4
|
||||
- uses: actions/upload-pages-artifact@7b1f4a764d45c48632c6b24a0339c27f5614fb0b # v4
|
||||
with:
|
||||
path: site
|
||||
- uses: actions/deploy-pages@v4
|
||||
- uses: actions/deploy-pages@d6db90164ac5ed86f2b6aed7e0febac5b3c0c03e # v4
|
||||
id: deployment
|
||||
|
|
|
|||
41
.github/workflows/lint.yml
vendored
Normal file
41
.github/workflows/lint.yml
vendored
Normal file
|
|
@ -0,0 +1,41 @@
|
|||
name: Lint
|
||||
|
||||
on:
|
||||
pull_request:
|
||||
push:
|
||||
workflow_dispatch:
|
||||
merge_group:
|
||||
|
||||
jobs:
|
||||
lint:
|
||||
name: Linting Checks
|
||||
runs-on: ubuntu-22.04
|
||||
steps:
|
||||
- name: Checkout repository
|
||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||
|
||||
- name: Install python
|
||||
uses: actions/setup-python@a309ff8b426b58ec0e2a45f0f869d46889d02405 # v6
|
||||
with:
|
||||
python-version: 3.x
|
||||
|
||||
- name: Check files
|
||||
uses: pre-commit/action@2c7b3805fd2a0fd8c1884dcaebf91fc102a13ecd # v3.0.1
|
||||
|
||||
- name: Install pnpm
|
||||
uses: pnpm/action-setup@a8198c4bff370c8506180b035930dea56dbd5288 # v5
|
||||
with:
|
||||
version: 10
|
||||
run_install: false
|
||||
|
||||
- name: Setup Node.js
|
||||
uses: actions/setup-node@53b83947a5a98c8d113130e565377fae1a50d02f # v6
|
||||
with:
|
||||
node-version: 24
|
||||
cache: 'pnpm'
|
||||
|
||||
- name: Install dependencies
|
||||
run: pnpm install
|
||||
|
||||
- name: Lint frontend
|
||||
run: pnpm run lint
|
||||
2
.github/workflows/pr-quality.yml
vendored
2
.github/workflows/pr-quality.yml
vendored
|
|
@ -13,6 +13,6 @@ jobs:
|
|||
anti-slop:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: peakoss/anti-slop@v0
|
||||
- uses: peakoss/anti-slop@a5a4b2440c9de6f65b64f0718a0136a1fdb04f6f # v0
|
||||
with:
|
||||
max-failures: 4
|
||||
|
|
|
|||
2
.github/workflows/reaction-comments.yml
vendored
2
.github/workflows/reaction-comments.yml
vendored
|
|
@ -15,4 +15,4 @@ jobs:
|
|||
action:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: dessant/reaction-comments@v4
|
||||
- uses: dessant/reaction-comments@e86d247c12bd5c043eec379a1a4453f20cadf913 # v4
|
||||
|
|
|
|||
4
.github/workflows/release-drafter.yml
vendored
4
.github/workflows/release-drafter.yml
vendored
|
|
@ -26,14 +26,14 @@ jobs:
|
|||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- if: github.event_name == 'workflow_dispatch' && github.event.inputs.version != ''
|
||||
uses: release-drafter/release-drafter@v7
|
||||
uses: release-drafter/release-drafter@a6acf82562eee06318b77ab8cb0b11ed81c677a7 # v7
|
||||
with:
|
||||
config-name: release-drafter.yml
|
||||
version: ${{ github.event.inputs.version }}
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
- if: github.event_name != 'workflow_dispatch' || github.event.inputs.version == ''
|
||||
uses: release-drafter/release-drafter@v7
|
||||
uses: release-drafter/release-drafter@a6acf82562eee06318b77ab8cb0b11ed81c677a7 # v7
|
||||
with:
|
||||
config-name: release-drafter.yml
|
||||
env:
|
||||
|
|
|
|||
10
.github/workflows/repo-maintenance.yml
vendored
10
.github/workflows/repo-maintenance.yml
vendored
|
|
@ -18,7 +18,7 @@ jobs:
|
|||
name: 'Stale'
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/stale@v10
|
||||
- uses: actions/stale@b5d41d4e1d5dceea10e7104786b73624c18a190f # v10
|
||||
with:
|
||||
days-before-stale: 7
|
||||
days-before-close: 14
|
||||
|
|
@ -32,7 +32,7 @@ jobs:
|
|||
name: 'Lock Old Threads'
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: dessant/lock-threads@v6
|
||||
- uses: dessant/lock-threads@7266a7ce5c1df01b1c6db85bf8cd86c737dadbe7 # v6
|
||||
with:
|
||||
issue-inactive-days: '30'
|
||||
pr-inactive-days: '30'
|
||||
|
|
@ -57,7 +57,7 @@ jobs:
|
|||
name: 'Close Answered Discussions'
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/github-script@v8
|
||||
- uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8
|
||||
with:
|
||||
script: |
|
||||
function sleep(ms) {
|
||||
|
|
@ -113,7 +113,7 @@ jobs:
|
|||
name: 'Close Outdated Discussions'
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/github-script@v8
|
||||
- uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8
|
||||
with:
|
||||
script: |
|
||||
function sleep(ms) {
|
||||
|
|
@ -204,7 +204,7 @@ jobs:
|
|||
name: 'Close Unsupported Feature Requests'
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- uses: actions/github-script@v8
|
||||
- uses: actions/github-script@ed597411d8f924073f98dfc5c65a23a2325f34cd # v8
|
||||
with:
|
||||
script: |
|
||||
function sleep(ms) {
|
||||
|
|
|
|||
8
.github/workflows/test.yml
vendored
8
.github/workflows/test.yml
vendored
|
|
@ -13,13 +13,13 @@ jobs:
|
|||
matrix:
|
||||
shard: [1, 2, 3, 4]
|
||||
steps:
|
||||
- uses: actions/checkout@v6
|
||||
- uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||
|
||||
- uses: pnpm/action-setup@v5
|
||||
- uses: pnpm/action-setup@a8198c4bff370c8506180b035930dea56dbd5288 # v5
|
||||
with:
|
||||
version: 9
|
||||
|
||||
- uses: actions/setup-node@v6
|
||||
- uses: actions/setup-node@53b83947a5a98c8d113130e565377fae1a50d02f # v6
|
||||
with:
|
||||
node-version: 20
|
||||
cache: pnpm
|
||||
|
|
@ -28,7 +28,7 @@ jobs:
|
|||
# Run Vitest directly so `--shard` is parsed as an option
|
||||
- run: pnpm -s exec vitest run --coverage --shard ${{ matrix.shard }}/4 --pool forks
|
||||
- name: Upload coverage reports to Codecov
|
||||
uses: codecov/codecov-action@v5
|
||||
uses: codecov/codecov-action@75cd11691c0faa626561e295848008c8a7dddffe # v5
|
||||
with:
|
||||
token: ${{ secrets.CODECOV_TOKEN }}
|
||||
files: ./coverage/lcov.info
|
||||
|
|
|
|||
|
|
@ -12,6 +12,7 @@ ARG CI
|
|||
ARG BUILDTIME
|
||||
ARG VERSION
|
||||
ARG REVISION
|
||||
ARG NEXT_PUBLIC_BASE_PATH
|
||||
ENV CI=$CI
|
||||
|
||||
# Install and build only outside CI
|
||||
|
|
@ -22,6 +23,7 @@ RUN if [ "$CI" != "true" ]; then \
|
|||
NEXT_PUBLIC_BUILDTIME=$BUILDTIME \
|
||||
NEXT_PUBLIC_VERSION=$VERSION \
|
||||
NEXT_PUBLIC_REVISION=$REVISION \
|
||||
NEXT_PUBLIC_BASE_PATH=$NEXT_PUBLIC_BASE_PATH \
|
||||
pnpm run build; \
|
||||
else \
|
||||
echo "✅ Using prebuilt app from CI context"; \
|
||||
|
|
|
|||
14
config/bookmarks.yaml
Normal file
14
config/bookmarks.yaml
Normal file
|
|
@ -0,0 +1,14 @@
|
|||
---
|
||||
- Repositories:
|
||||
- DeckForge:
|
||||
- abbr: DF
|
||||
href: https://bitbucket.org/zlalani/ppt-tool
|
||||
- GMAL Scope Builder:
|
||||
- abbr: GSB
|
||||
href: https://bitbucket.org/zlalani/gmal-scope-builder
|
||||
- Homepage:
|
||||
- abbr: HP
|
||||
href: https://bitbucket.org/zlalani/homepage
|
||||
- OliVAS:
|
||||
- abbr: OL
|
||||
href: https://bitbucket.org/zlalani/olivas
|
||||
3
config/docker.yaml
Normal file
3
config/docker.yaml
Normal file
|
|
@ -0,0 +1,3 @@
|
|||
---
|
||||
local:
|
||||
socket: /var/run/docker.sock
|
||||
96
config/services.yaml
Normal file
96
config/services.yaml
Normal file
|
|
@ -0,0 +1,96 @@
|
|||
---
|
||||
- AI Tools:
|
||||
- DeckForge:
|
||||
icon: mdi-presentation
|
||||
href: https://optical-dev.oliver.solutions/ppt-tool
|
||||
description: AI presentation generator
|
||||
container: ppt-tool-web-1
|
||||
server: local
|
||||
siteMonitor: https://optical-dev.oliver.solutions/ppt-tool
|
||||
showStats: true
|
||||
widget:
|
||||
type: deploy
|
||||
service: ppt-tool
|
||||
label: DeckForge
|
||||
|
||||
- GMAL Scope Builder:
|
||||
icon: mdi-briefcase-outline
|
||||
href: https://optical-dev.oliver.solutions/gsb
|
||||
description: AI ratecard & team scoping
|
||||
container: gmal-scope-builder-backend-1
|
||||
server: local
|
||||
siteMonitor: https://optical-dev.oliver.solutions/gsb/
|
||||
showStats: true
|
||||
widget:
|
||||
type: deploy
|
||||
service: gmal-scope-builder
|
||||
label: Scope Builder
|
||||
|
||||
- Semblance:
|
||||
icon: mdi-account-group-outline
|
||||
href: https://optical-dev.oliver.solutions/semblance
|
||||
description: Synthetic personas & focus groups
|
||||
container: semblance-backend-1
|
||||
server: local
|
||||
siteMonitor: https://optical-dev.oliver.solutions/semblance/
|
||||
showStats: true
|
||||
widget:
|
||||
type: deploy
|
||||
service: semblance
|
||||
label: Semblance
|
||||
|
||||
- CC Dashboard:
|
||||
icon: mdi-view-dashboard-outline
|
||||
href: https://optical-dev.oliver.solutions/cc-dashboard
|
||||
description: API key & project management
|
||||
container: cc-dashboard-app-1
|
||||
server: local
|
||||
siteMonitor: https://optical-dev.oliver.solutions/cc-dashboard/
|
||||
showStats: true
|
||||
widget:
|
||||
type: deploy
|
||||
service: cc-dashboard
|
||||
label: CC Dashboard
|
||||
|
||||
- OliVAS:
|
||||
icon: mdi-robot-outline
|
||||
href: https://optical-dev.oliver.solutions
|
||||
description: OliVAS backend API
|
||||
container: olivas-backend-1
|
||||
server: local
|
||||
siteMonitor: https://optical-dev.oliver.solutions/api/health
|
||||
showStats: true
|
||||
widget:
|
||||
type: deploy
|
||||
service: olivas
|
||||
label: OliVAS
|
||||
|
||||
- Infrastructure:
|
||||
- Homepage:
|
||||
icon: mdi-home-outline
|
||||
href: https://optical-dev.oliver.solutions/homepage
|
||||
description: This dashboard
|
||||
container: homepage-app-1
|
||||
server: local
|
||||
siteMonitor: https://optical-dev.oliver.solutions/homepage
|
||||
showStats: true
|
||||
widget:
|
||||
type: deploy
|
||||
service: homepage
|
||||
label: Homepage
|
||||
|
||||
- Deploy API:
|
||||
icon: mdi-rocket-launch-outline
|
||||
href: https://optical-dev.oliver.solutions/deploy-api/docs
|
||||
description: One-click deploy service
|
||||
siteMonitor: https://optical-dev.oliver.solutions/deploy-api/services
|
||||
|
||||
- PostgreSQL × 4:
|
||||
icon: mdi-database-outline
|
||||
description: "ppt-tool · olivas · cc-dashboard · gmal"
|
||||
href: https://optical-dev.oliver.solutions
|
||||
|
||||
- Redis + MongoDB:
|
||||
icon: mdi-server-outline
|
||||
description: ppt-tool (Redis) · semblance (Mongo)
|
||||
href: https://optical-dev.oliver.solutions
|
||||
33
config/settings.yaml
Normal file
33
config/settings.yaml
Normal file
|
|
@ -0,0 +1,33 @@
|
|||
---
|
||||
title: Optical Dev
|
||||
language: en
|
||||
theme: dark
|
||||
color: zinc
|
||||
|
||||
# Full-width layout, equal card heights, hide noise
|
||||
fullWidth: true
|
||||
useEqualHeights: true
|
||||
headerStyle: clean
|
||||
cardBlur: md
|
||||
statusStyle: dot
|
||||
hideVersion: true
|
||||
disableUpdateCheck: true
|
||||
target: _blank
|
||||
|
||||
# Layout sections
|
||||
layout:
|
||||
Widgets:
|
||||
style: row
|
||||
columns: 2
|
||||
AI Tools:
|
||||
style: row
|
||||
columns: 5
|
||||
useEqualHeights: true
|
||||
Infrastructure:
|
||||
style: row
|
||||
columns: 4
|
||||
useEqualHeights: true
|
||||
Repositories:
|
||||
style: row
|
||||
columns: 4
|
||||
iconsOnly: false
|
||||
15
config/widgets.yaml
Normal file
15
config/widgets.yaml
Normal file
|
|
@ -0,0 +1,15 @@
|
|||
---
|
||||
- resources:
|
||||
cpu: true
|
||||
memory: true
|
||||
disk: /
|
||||
label: optical-dev server
|
||||
expanded: true
|
||||
cputemp: false
|
||||
|
||||
- datetime:
|
||||
text_size: xl
|
||||
format:
|
||||
timeStyle: short
|
||||
dateStyle: short
|
||||
hourCycle: h23
|
||||
30
deploy.sh
Normal file
30
deploy.sh
Normal file
|
|
@ -0,0 +1,30 @@
|
|||
#!/bin/bash
|
||||
set -e
|
||||
|
||||
DEPLOY_DIR="/opt/homepage"
|
||||
REPO="git@bitbucket.org:zlalani/homepage.git"
|
||||
|
||||
echo "=== Homepage Deploy ==="
|
||||
|
||||
if [ ! -d "$DEPLOY_DIR/.git" ]; then
|
||||
echo "Cloning repository..."
|
||||
git clone "$REPO" "$DEPLOY_DIR"
|
||||
cd "$DEPLOY_DIR"
|
||||
else
|
||||
echo "Pulling latest changes..."
|
||||
cd "$DEPLOY_DIR"
|
||||
git pull origin dev
|
||||
fi
|
||||
|
||||
mkdir -p config
|
||||
|
||||
echo "Building and starting container..."
|
||||
docker compose build --no-cache
|
||||
docker compose up -d
|
||||
|
||||
echo "Waiting for healthcheck..."
|
||||
sleep 15
|
||||
docker compose ps
|
||||
|
||||
echo ""
|
||||
echo "Done. Homepage available at https://optical-dev.oliver.solutions/homepage"
|
||||
21
docker-compose.yml
Normal file
21
docker-compose.yml
Normal file
|
|
@ -0,0 +1,21 @@
|
|||
services:
|
||||
app:
|
||||
build:
|
||||
context: .
|
||||
args:
|
||||
NEXT_PUBLIC_BASE_PATH: /homepage
|
||||
ports:
|
||||
- "127.0.0.1:3001:3000"
|
||||
volumes:
|
||||
- ./config:/app/config
|
||||
- /var/run/docker.sock:/var/run/docker.sock:ro
|
||||
environment:
|
||||
HOSTNAME: "::"
|
||||
HOMEPAGE_ALLOWED_HOSTS: "optical-dev.oliver.solutions"
|
||||
restart: unless-stopped
|
||||
healthcheck:
|
||||
test: ["CMD-SHELL", "wget --no-verbose --tries=1 --spider http://127.0.0.1:3000/api/healthcheck || exit 1"]
|
||||
interval: 10s
|
||||
timeout: 3s
|
||||
start_period: 30s
|
||||
retries: 3
|
||||
24
docs/widgets/services/unifi-drive.md
Normal file
24
docs/widgets/services/unifi-drive.md
Normal file
|
|
@ -0,0 +1,24 @@
|
|||
---
|
||||
title: UniFi Drive
|
||||
description: UniFi Drive Widget Configuration
|
||||
---
|
||||
|
||||
Learn more about [UniFi Drive](https://ui.com/integrations/network-storage).
|
||||
|
||||
## Configuration
|
||||
|
||||
Displays storage statistics from your UniFi Network Attached Storage (UNAS) device. Requires a local UniFi account with at least read privileges.
|
||||
|
||||
Allowed fields: `["total", "used", "available", "status"]`
|
||||
|
||||
```yaml
|
||||
widget:
|
||||
type: unifi_drive
|
||||
url: https://unifi.host.or.ip
|
||||
username: your_username
|
||||
password: your_password
|
||||
```
|
||||
|
||||
!!! hint
|
||||
|
||||
If you enter incorrect credentials and receive an "API Error", you may need to recreate the container or restart the service to clear the cache.
|
||||
|
|
@ -171,6 +171,7 @@ nav:
|
|||
- widgets/services/truenas.md
|
||||
- widgets/services/tubearchivist.md
|
||||
- widgets/services/unifi-controller.md
|
||||
- widgets/services/unifi-drive.md
|
||||
- widgets/services/unmanic.md
|
||||
- widgets/services/unraid.md
|
||||
- widgets/services/uptime-kuma.md
|
||||
|
|
|
|||
|
|
@ -4,6 +4,7 @@ const { i18n } = require("./next-i18next.config");
|
|||
const nextConfig = {
|
||||
reactStrictMode: true,
|
||||
output: "standalone",
|
||||
basePath: process.env.NEXT_PUBLIC_BASE_PATH || "",
|
||||
images: {
|
||||
remotePatterns: [
|
||||
{
|
||||
|
|
|
|||
16
pnpm-lock.yaml
generated
16
pnpm-lock.yaml
generated
|
|
@ -1593,11 +1593,11 @@ packages:
|
|||
bl@4.1.0:
|
||||
resolution: {integrity: sha512-1W07cM9gS6DcLperZfFSj+bWLtaPGSOHWhPiGzXmvVJbRLdG82sH/Kn8EtW1VqWVA54AKf2h5k5BbnIbwF3h6w==}
|
||||
|
||||
brace-expansion@1.1.12:
|
||||
resolution: {integrity: sha512-9T9UjW3r0UW5c1Q7GTwllptXwhvYmEzFhzMfZ9H7FQWt+uZePjZPjBP/W1ZEyZ1twGWom5/56TF4lPcqjnDHcg==}
|
||||
brace-expansion@1.1.13:
|
||||
resolution: {integrity: sha512-9ZLprWS6EENmhEOpjCYW2c8VkmOvckIJZfkr7rBW6dObmfgJ/L1GpSYW5Hpo9lDz4D1+n0Ckz8rU7FwHDQiG/w==}
|
||||
|
||||
brace-expansion@2.0.2:
|
||||
resolution: {integrity: sha512-Jt0vHyM+jmUBqojB7E1NIYadt0vI0Qxjxd2TErW94wDz+E2LAm5vKMXXwg6ZZBTHPuUlDgQHKXvjGBdfcF1ZDQ==}
|
||||
brace-expansion@2.0.3:
|
||||
resolution: {integrity: sha512-MCV/fYJEbqx68aE58kv2cA/kiky1G8vux3OR6/jbS+jIMe/6fJWa0DTzJU7dqijOWYwHi1t29FlfYI9uytqlpA==}
|
||||
|
||||
braces@3.0.3:
|
||||
resolution: {integrity: sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==}
|
||||
|
|
@ -5162,12 +5162,12 @@ snapshots:
|
|||
inherits: 2.0.4
|
||||
readable-stream: 3.6.2
|
||||
|
||||
brace-expansion@1.1.12:
|
||||
brace-expansion@1.1.13:
|
||||
dependencies:
|
||||
balanced-match: 1.0.2
|
||||
concat-map: 0.0.1
|
||||
|
||||
brace-expansion@2.0.2:
|
||||
brace-expansion@2.0.3:
|
||||
dependencies:
|
||||
balanced-match: 1.0.2
|
||||
|
||||
|
|
@ -6598,11 +6598,11 @@ snapshots:
|
|||
|
||||
minimatch@3.1.2:
|
||||
dependencies:
|
||||
brace-expansion: 1.1.12
|
||||
brace-expansion: 1.1.13
|
||||
|
||||
minimatch@9.0.5:
|
||||
dependencies:
|
||||
brace-expansion: 2.0.2
|
||||
brace-expansion: 2.0.3
|
||||
|
||||
minimist@1.2.8: {}
|
||||
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Wag asseblief",
|
||||
"empty_data": "Substelsel status onbekend"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "حالة النظام الفرعي غير معروفة"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "استقبال",
|
||||
"tx": "ارسال",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Моля изчакайте",
|
||||
"empty_data": "Неизвестен статус на подсистема"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "ПЧ",
|
||||
"tx": "ИЗ",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Si us plau espera",
|
||||
"empty_data": "Estat del subsistema desconegut"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "Rebut",
|
||||
"tx": "Transmès",
|
||||
|
|
|
|||
|
|
@ -39,7 +39,7 @@
|
|||
"placeholder": "Hledat…"
|
||||
},
|
||||
"resources": {
|
||||
"cpu": "Zatížení procesoru",
|
||||
"cpu": "Využití procesoru",
|
||||
"mem": "Využití paměti",
|
||||
"total": "Celkem",
|
||||
"free": "Volné",
|
||||
|
|
@ -66,11 +66,16 @@
|
|||
"wait": "Čekejte prosím",
|
||||
"empty_data": "Stav podsystému neznámý"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Zdravý",
|
||||
"degraded": "Degradováno",
|
||||
"no_data": "Nejsou k dispozici žádná data úložiště"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
"mem": "Využití paměti",
|
||||
"cpu": "Zatížení procesoru",
|
||||
"cpu": "Využití procesoru",
|
||||
"running": "Běží",
|
||||
"offline": "Offline",
|
||||
"error": "Chyba",
|
||||
|
|
@ -232,7 +237,7 @@
|
|||
"seed": "Seedované"
|
||||
},
|
||||
"qnap": {
|
||||
"cpuUsage": "Zatížení procesoru",
|
||||
"cpuUsage": "Využití procesoru",
|
||||
"memUsage": "Využití paměti",
|
||||
"systemTempC": "Teplota systému",
|
||||
"poolUsage": "Využití fondu",
|
||||
|
|
@ -445,12 +450,12 @@
|
|||
},
|
||||
"proxmox": {
|
||||
"mem": "Využití paměti",
|
||||
"cpu": "Zatížení procesoru",
|
||||
"cpu": "Využití procesoru",
|
||||
"lxc": "LXC",
|
||||
"vms": "Virtuální Stroje"
|
||||
},
|
||||
"glances": {
|
||||
"cpu": "Zatížení procesoru",
|
||||
"cpu": "Využití procesoru",
|
||||
"load": "Zatížení",
|
||||
"wait": "Čekejte prosím",
|
||||
"temp": "TEPLOTA",
|
||||
|
|
@ -635,7 +640,7 @@
|
|||
"no_devices": "Žádná přijatá data zařízení"
|
||||
},
|
||||
"mikrotik": {
|
||||
"cpuLoad": "Zatížení procesoru",
|
||||
"cpuLoad": "Využití procesoru",
|
||||
"memoryUsed": "Využití paměti",
|
||||
"uptime": "Doba provozu",
|
||||
"numberOfLeases": "Pronájmy"
|
||||
|
|
@ -686,7 +691,7 @@
|
|||
"proxmoxbackupserver": {
|
||||
"datastore_usage": "Datové úložiště",
|
||||
"failed_tasks_24h": "Neúspěšné úlohy 24h",
|
||||
"cpu_usage": "Zatížení procesoru",
|
||||
"cpu_usage": "Využití procesoru",
|
||||
"memory_usage": "Využití paměti"
|
||||
},
|
||||
"immich": {
|
||||
|
|
@ -750,7 +755,7 @@
|
|||
"alertstriggered": "Spuštěné výstrahy"
|
||||
},
|
||||
"nextcloud": {
|
||||
"cpuload": "Zatížení procesoru",
|
||||
"cpuload": "Využití procesoru",
|
||||
"memoryusage": "Využití paměti",
|
||||
"freespace": "Volný prostor",
|
||||
"activeusers": "Aktivní uživatelé",
|
||||
|
|
@ -873,7 +878,7 @@
|
|||
},
|
||||
"openwrt": {
|
||||
"uptime": "Doba provozu",
|
||||
"cpuLoad": "Prům. zatížení procesoru (5m)",
|
||||
"cpuLoad": "Prům. využití procesoru (5m)",
|
||||
"up": "Běží",
|
||||
"down": "Výpadek",
|
||||
"bytesTx": "Přeneseno",
|
||||
|
|
@ -1037,7 +1042,7 @@
|
|||
"pending": "Čekající",
|
||||
"status": "Stav",
|
||||
"updated": "Aktualizováno",
|
||||
"cpu": "Zatížení procesoru",
|
||||
"cpu": "Využití procesoru",
|
||||
"memory": "Využití paměti",
|
||||
"disk": "Disk",
|
||||
"network": "Síť"
|
||||
|
|
@ -1133,7 +1138,7 @@
|
|||
"NO_DATA_DISKS": "Žádné datové disky",
|
||||
"notifications": "Upozornění",
|
||||
"status": "Stav",
|
||||
"cpu": "Zatížení procesoru",
|
||||
"cpu": "Využití procesoru",
|
||||
"memoryUsed": "Využití paměti",
|
||||
"memoryAvailable": "Volná paměť",
|
||||
"arrayUsed": "Využito pole",
|
||||
|
|
@ -1164,7 +1169,7 @@
|
|||
"dockhand": {
|
||||
"running": "Běží",
|
||||
"stopped": "Zastaveno",
|
||||
"cpu": "Zatížení procesoru",
|
||||
"cpu": "Využití procesoru",
|
||||
"memory": "Využití paměti",
|
||||
"images": "Obrazy",
|
||||
"volumes": "Úložiště",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status ukendt"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Bitte warten",
|
||||
"empty_data": "Subsystem-Status unbekannt"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Gesund",
|
||||
"degraded": "Beeinträchtigt",
|
||||
"no_data": "Keine Speicherdaten verfügbar"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
@ -115,7 +120,7 @@
|
|||
"movies": "Filme",
|
||||
"series": "Serien",
|
||||
"episodes": "Episoden",
|
||||
"songs": "Songs"
|
||||
"songs": "Titel"
|
||||
},
|
||||
"esphome": {
|
||||
"offline": "Offline",
|
||||
|
|
@ -185,10 +190,10 @@
|
|||
"plex_connection_error": "Prüfe Plex-Verbindung"
|
||||
},
|
||||
"tracearr": {
|
||||
"no_active": "No Active Streams",
|
||||
"no_active": "Keine aktiven Streams",
|
||||
"streams": "Streams",
|
||||
"transcodes": "Transcodes",
|
||||
"directplay": "Direct Play",
|
||||
"transcodes": "Transkodieren",
|
||||
"directplay": "Direkte Wiedergabe",
|
||||
"bitrate": "Bitrate"
|
||||
},
|
||||
"omada": {
|
||||
|
|
@ -290,12 +295,12 @@
|
|||
"available": "Verfügbar"
|
||||
},
|
||||
"seerr": {
|
||||
"pending": "Pending",
|
||||
"approved": "Approved",
|
||||
"available": "Available",
|
||||
"completed": "Completed",
|
||||
"processing": "Processing",
|
||||
"issues": "Open Issues"
|
||||
"pending": "Ausstehend",
|
||||
"approved": "Bestätigt",
|
||||
"available": "Verfügbar",
|
||||
"completed": "Abgeschlossen",
|
||||
"processing": "Wird verarbeitet",
|
||||
"issues": "Offene Probleme"
|
||||
},
|
||||
"netalertx": {
|
||||
"total": "Total",
|
||||
|
|
@ -615,7 +620,7 @@
|
|||
},
|
||||
"pangolin": {
|
||||
"orgs": "Orgs",
|
||||
"sites": "Sites",
|
||||
"sites": "Seiten",
|
||||
"resources": "Ressourcen",
|
||||
"targets": "Ziele",
|
||||
"traffic": "Traffic",
|
||||
|
|
@ -719,7 +724,7 @@
|
|||
"volumeAvailable": "Verfügbar"
|
||||
},
|
||||
"dispatcharr": {
|
||||
"channels": "Channels",
|
||||
"channels": "Kanäle",
|
||||
"streams": "Streams"
|
||||
},
|
||||
"mylar": {
|
||||
|
|
@ -811,10 +816,10 @@
|
|||
"series": "Serien"
|
||||
},
|
||||
"booklore": {
|
||||
"libraries": "Libraries",
|
||||
"libraries": "Bibliotheken",
|
||||
"books": "Bücher",
|
||||
"reading": "Reading",
|
||||
"finished": "Finished"
|
||||
"reading": "Am Lesen",
|
||||
"finished": "Fertig"
|
||||
},
|
||||
"jdownloader": {
|
||||
"downloadCount": "Warteschlange",
|
||||
|
|
@ -1155,11 +1160,11 @@
|
|||
"artists": "Künstler"
|
||||
},
|
||||
"arcane": {
|
||||
"containers": "Containers",
|
||||
"containers": "Container",
|
||||
"images": "Images",
|
||||
"image_updates": "Image Updates",
|
||||
"images_unused": "Unused",
|
||||
"environment_required": "Environment ID Required"
|
||||
"image_updates": "Image-Updates",
|
||||
"images_unused": "Ungenutzt",
|
||||
"environment_required": "Umgebungs-ID erforderlich"
|
||||
},
|
||||
"dockhand": {
|
||||
"running": "Wird ausgeführt",
|
||||
|
|
@ -1176,9 +1181,9 @@
|
|||
"environment_not_found": "Umgebung nicht gefunden"
|
||||
},
|
||||
"sparkyfitness": {
|
||||
"eaten": "Eaten",
|
||||
"burned": "Burned",
|
||||
"remaining": "Remaining",
|
||||
"steps": "Steps"
|
||||
"eaten": "",
|
||||
"burned": "Verbrannt",
|
||||
"remaining": "Verbleibend",
|
||||
"steps": "Schritte"
|
||||
}
|
||||
}
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Άγνωστη κατάσταση υποσυστήματος"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status unknown"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsistemostatuso nekonata"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Espere, por favor",
|
||||
"empty_data": "Se desconoce el estado del subsistema"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "Recibido",
|
||||
"tx": "Transmitido",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status unknown"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status unknown"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Veuillez patienter",
|
||||
"empty_data": "Statut du sous-système inconnu"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "Rx",
|
||||
"tx": "Tx",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "נא להמתין",
|
||||
"empty_data": "מצב תת-מערכת לא ידוע"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status unknown"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Pričekaj",
|
||||
"empty_data": "Stanje podsustava nepoznato"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Kérjük várjon",
|
||||
"empty_data": "Az alrendszer állapota ismeretlen"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Status subsistem tdk diketahui"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Stato del sottosistema sconosciuto"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "お待ちください",
|
||||
"empty_data": "サブシステムの状態は不明"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "受信済み",
|
||||
"tx": "送信済み",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "잠시만 기다려주세요",
|
||||
"empty_data": "서브시스템 상태 알 수 없음"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "수신",
|
||||
"tx": "송신",
|
||||
|
|
@ -108,14 +113,14 @@
|
|||
"songs": "음악"
|
||||
},
|
||||
"jellyfin": {
|
||||
"playing": "Playing",
|
||||
"transcoding": "Transcoding",
|
||||
"bitrate": "Bitrate",
|
||||
"no_active": "No Active Streams",
|
||||
"movies": "Movies",
|
||||
"series": "Series",
|
||||
"episodes": "Episodes",
|
||||
"songs": "Songs"
|
||||
"playing": "재생 중",
|
||||
"transcoding": "트랜스코딩 중",
|
||||
"bitrate": "비트레이트",
|
||||
"no_active": "활성 스트림 없음",
|
||||
"movies": "영상",
|
||||
"series": "시리즈",
|
||||
"episodes": "에피소드",
|
||||
"songs": "음악"
|
||||
},
|
||||
"esphome": {
|
||||
"offline": "오프라인",
|
||||
|
|
@ -185,11 +190,11 @@
|
|||
"plex_connection_error": "Plex 연결 확인"
|
||||
},
|
||||
"tracearr": {
|
||||
"no_active": "No Active Streams",
|
||||
"streams": "Streams",
|
||||
"transcodes": "Transcodes",
|
||||
"directplay": "Direct Play",
|
||||
"bitrate": "Bitrate"
|
||||
"no_active": "활성 스트림 없음",
|
||||
"streams": "스트림",
|
||||
"transcodes": "트랜스코드",
|
||||
"directplay": "다이렉트 플레이",
|
||||
"bitrate": "비트레이트"
|
||||
},
|
||||
"omada": {
|
||||
"connectedAp": "연결된 AP",
|
||||
|
|
@ -290,12 +295,12 @@
|
|||
"available": "이용 가능"
|
||||
},
|
||||
"seerr": {
|
||||
"pending": "Pending",
|
||||
"approved": "Approved",
|
||||
"available": "Available",
|
||||
"completed": "Completed",
|
||||
"processing": "Processing",
|
||||
"issues": "Open Issues"
|
||||
"pending": "대기 중",
|
||||
"approved": "승인됨",
|
||||
"available": "사용 가능",
|
||||
"completed": "완료됨",
|
||||
"processing": "처리 중",
|
||||
"issues": "열린 이슈"
|
||||
},
|
||||
"netalertx": {
|
||||
"total": "전체",
|
||||
|
|
@ -546,7 +551,7 @@
|
|||
"up": "업",
|
||||
"pending": "대기 중",
|
||||
"down": "다운",
|
||||
"ok": "Ok"
|
||||
"ok": "확인"
|
||||
},
|
||||
"healthchecks": {
|
||||
"new": "신규",
|
||||
|
|
@ -618,9 +623,9 @@
|
|||
"sites": "Sites",
|
||||
"resources": "Resources",
|
||||
"targets": "Targets",
|
||||
"traffic": "Traffic",
|
||||
"in": "In",
|
||||
"out": "Out"
|
||||
"traffic": "트래픽",
|
||||
"in": "수신",
|
||||
"out": "송신"
|
||||
},
|
||||
"peanut": {
|
||||
"battery_charge": "배터리 충전",
|
||||
|
|
@ -719,8 +724,8 @@
|
|||
"volumeAvailable": "사용 가능"
|
||||
},
|
||||
"dispatcharr": {
|
||||
"channels": "Channels",
|
||||
"streams": "Streams"
|
||||
"channels": "채널",
|
||||
"streams": "스트림"
|
||||
},
|
||||
"mylar": {
|
||||
"series": "시리즈",
|
||||
|
|
@ -787,7 +792,7 @@
|
|||
"gross_percent_today": "오늘",
|
||||
"gross_percent_1y": "1년",
|
||||
"gross_percent_max": "전체 기간",
|
||||
"net_worth": "Net Worth"
|
||||
"net_worth": "순자산"
|
||||
},
|
||||
"audiobookshelf": {
|
||||
"podcasts": "팟캐스트",
|
||||
|
|
@ -811,10 +816,10 @@
|
|||
"series": "시리즈"
|
||||
},
|
||||
"booklore": {
|
||||
"libraries": "Libraries",
|
||||
"books": "Books",
|
||||
"reading": "Reading",
|
||||
"finished": "Finished"
|
||||
"libraries": "라이브러리",
|
||||
"books": "책",
|
||||
"reading": "읽는 중",
|
||||
"finished": "완료"
|
||||
},
|
||||
"jdownloader": {
|
||||
"downloadCount": "대기열",
|
||||
|
|
@ -1150,30 +1155,30 @@
|
|||
"bytes_added_30": "추가된 용량"
|
||||
},
|
||||
"yourspotify": {
|
||||
"songs": "Songs",
|
||||
"time": "Time",
|
||||
"artists": "Artists"
|
||||
"songs": "음악",
|
||||
"time": "시간",
|
||||
"artists": "아티스트"
|
||||
},
|
||||
"arcane": {
|
||||
"containers": "Containers",
|
||||
"images": "Images",
|
||||
"image_updates": "Image Updates",
|
||||
"images_unused": "Unused",
|
||||
"environment_required": "Environment ID Required"
|
||||
"containers": "컨테이너",
|
||||
"images": "이미지",
|
||||
"image_updates": "이미지 업데이트",
|
||||
"images_unused": "미사용",
|
||||
"environment_required": "환경 ID 필요"
|
||||
},
|
||||
"dockhand": {
|
||||
"running": "Running",
|
||||
"stopped": "Stopped",
|
||||
"running": "실행 중",
|
||||
"stopped": "정지됨",
|
||||
"cpu": "CPU",
|
||||
"memory": "Memory",
|
||||
"images": "Images",
|
||||
"volumes": "Volumes",
|
||||
"events_today": "Events Today",
|
||||
"pending_updates": "Pending Updates",
|
||||
"stacks": "Stacks",
|
||||
"paused": "Paused",
|
||||
"total": "Total",
|
||||
"environment_not_found": "Environment Not Found"
|
||||
"memory": "메모리",
|
||||
"images": "이미지",
|
||||
"volumes": "볼륨",
|
||||
"events_today": "오늘의 이벤트",
|
||||
"pending_updates": "대기 중인 업데이트",
|
||||
"stacks": "스택",
|
||||
"paused": "일시정지됨",
|
||||
"total": "전체",
|
||||
"environment_not_found": "환경 없음"
|
||||
},
|
||||
"sparkyfitness": {
|
||||
"eaten": "Eaten",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status unknown"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Status subsistem tak diketahui"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Even geduld",
|
||||
"empty_data": "Subsysteem status onbekend"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Ukjent undersystemstatus"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Proszę czekać",
|
||||
"empty_data": "Status podsystemu nieznany"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "Rx",
|
||||
"tx": "Tx",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Status de Subsistema Desconhecido"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "Rx",
|
||||
"tx": "Tx",
|
||||
|
|
|
|||
|
|
@ -14,7 +14,7 @@
|
|||
"date": "{{value, date}}",
|
||||
"relativeDate": "{{value, relativeDate}}",
|
||||
"duration": "{{value, duration}}",
|
||||
"months": "M",
|
||||
"months": "mo",
|
||||
"days": "d",
|
||||
"hours": "h",
|
||||
"minutes": "m",
|
||||
|
|
@ -66,9 +66,14 @@
|
|||
"wait": "Por favor, aguarde",
|
||||
"empty_data": "Status do Subsistema desconhecido"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "Rx",
|
||||
"tx": "Tx",
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
"mem": "MEM",
|
||||
"cpu": "CPU",
|
||||
"running": "Executando",
|
||||
|
|
@ -101,21 +106,21 @@
|
|||
"playing": "A reproduzir",
|
||||
"transcoding": "Transcodificação",
|
||||
"bitrate": "Taxa de bits",
|
||||
"no_active": "Sem Streams Ativos",
|
||||
"no_active": "Sem Transmissões Ativas",
|
||||
"movies": "Filmes",
|
||||
"series": "Séries",
|
||||
"episodes": "Episódios",
|
||||
"songs": "Canções"
|
||||
},
|
||||
"jellyfin": {
|
||||
"playing": "Playing",
|
||||
"playing": "Jogando",
|
||||
"transcoding": "Transcoding",
|
||||
"bitrate": "Bitrate",
|
||||
"no_active": "No Active Streams",
|
||||
"movies": "Movies",
|
||||
"series": "Series",
|
||||
"episodes": "Episodes",
|
||||
"songs": "Songs"
|
||||
"movies": "Filmes",
|
||||
"series": "Séries",
|
||||
"episodes": "Episódios",
|
||||
"songs": "Músicas"
|
||||
},
|
||||
"esphome": {
|
||||
"offline": "Offline",
|
||||
|
|
@ -290,12 +295,12 @@
|
|||
"available": "Disponível"
|
||||
},
|
||||
"seerr": {
|
||||
"pending": "Pending",
|
||||
"approved": "Approved",
|
||||
"available": "Available",
|
||||
"completed": "Completed",
|
||||
"processing": "Processing",
|
||||
"issues": "Open Issues"
|
||||
"pending": "Pendente",
|
||||
"approved": "Aprovado",
|
||||
"available": "Disponível",
|
||||
"completed": "Concluído",
|
||||
"processing": "Processando",
|
||||
"issues": "Erros pendentes"
|
||||
},
|
||||
"netalertx": {
|
||||
"total": "Total",
|
||||
|
|
@ -616,7 +621,7 @@
|
|||
"pangolin": {
|
||||
"orgs": "Orgs",
|
||||
"sites": "Sites",
|
||||
"resources": "Resources",
|
||||
"resources": "Recursos",
|
||||
"targets": "Targets",
|
||||
"traffic": "Traffic",
|
||||
"in": "In",
|
||||
|
|
@ -719,8 +724,8 @@
|
|||
"volumeAvailable": "Disponível"
|
||||
},
|
||||
"dispatcharr": {
|
||||
"channels": "Channels",
|
||||
"streams": "Streams"
|
||||
"channels": "Canais",
|
||||
"streams": "Transmissões"
|
||||
},
|
||||
"mylar": {
|
||||
"series": "Séries",
|
||||
|
|
@ -811,10 +816,10 @@
|
|||
"series": "Séries"
|
||||
},
|
||||
"booklore": {
|
||||
"libraries": "Libraries",
|
||||
"books": "Books",
|
||||
"reading": "Reading",
|
||||
"finished": "Finished"
|
||||
"libraries": "Bibliotecas",
|
||||
"books": "Livros",
|
||||
"reading": "Lendo",
|
||||
"finished": "Finalizado"
|
||||
},
|
||||
"jdownloader": {
|
||||
"downloadCount": "Fila de espera",
|
||||
|
|
@ -1155,23 +1160,23 @@
|
|||
"artists": "Artistas"
|
||||
},
|
||||
"arcane": {
|
||||
"containers": "Containers",
|
||||
"images": "Images",
|
||||
"image_updates": "Image Updates",
|
||||
"images_unused": "Unused",
|
||||
"containers": "Recipientes",
|
||||
"images": "Imagens",
|
||||
"image_updates": "Atualizações de Imagem",
|
||||
"images_unused": "Não utilizado",
|
||||
"environment_required": "Environment ID Required"
|
||||
},
|
||||
"dockhand": {
|
||||
"running": "Running",
|
||||
"running": "Executando",
|
||||
"stopped": "Stopped",
|
||||
"cpu": "CPU",
|
||||
"memory": "Memory",
|
||||
"images": "Images",
|
||||
"volumes": "Volumes",
|
||||
"events_today": "Events Today",
|
||||
"pending_updates": "Pending Updates",
|
||||
"stacks": "Stacks",
|
||||
"paused": "Paused",
|
||||
"memory": "Memória",
|
||||
"images": "Imagens",
|
||||
"volumes": "Quantidades",
|
||||
"events_today": "Eventos hoje",
|
||||
"pending_updates": "Atualizações pendentes",
|
||||
"stacks": "Pilhas",
|
||||
"paused": "Pausado",
|
||||
"total": "Total",
|
||||
"environment_not_found": "Environment Not Found"
|
||||
},
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Starea subsistemului este necunoscut"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Пожалуйста, подождите",
|
||||
"empty_data": "Статус подсистемы неизвестен"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Čakajte, prosím",
|
||||
"empty_data": "Stav podsystému neznámy"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "Prijaté",
|
||||
"tx": "Odoslané",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Neznani status podsistema"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Молим сачекајте",
|
||||
"empty_data": "Статус подсистема непознат"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status unknown"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status unknown"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "Subsystem status unknown"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Lütfen bekleyin",
|
||||
"empty_data": "Alt sistem durumu bilinmiyor"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "Gelen Veri",
|
||||
"tx": "Giden Veri",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Будь ласка, зачекайте",
|
||||
"empty_data": "Статус підсистеми невідомий"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Vui lòng chờ",
|
||||
"empty_data": "Trạng thái hệ thống phụ không xác định"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "RX",
|
||||
"tx": "TX",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "子系統狀態未知"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "接收",
|
||||
"tx": "發送",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "请稍候",
|
||||
"empty_data": "子系统状态未知"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "接收",
|
||||
"tx": "发送",
|
||||
|
|
|
|||
|
|
@ -66,6 +66,11 @@
|
|||
"wait": "Please wait",
|
||||
"empty_data": "子系統狀態未知"
|
||||
},
|
||||
"unifi_drive": {
|
||||
"healthy": "Healthy",
|
||||
"degraded": "Degraded",
|
||||
"no_data": "No storage data available"
|
||||
},
|
||||
"docker": {
|
||||
"rx": "接收",
|
||||
"tx": "傳送",
|
||||
|
|
|
|||
|
|
@ -2,7 +2,7 @@ import { MdRefresh } from "react-icons/md";
|
|||
|
||||
export default function Revalidate() {
|
||||
const revalidate = () => {
|
||||
fetch("/api/revalidate").then((res) => {
|
||||
fetch(`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/revalidate`).then((res) => {
|
||||
if (res.ok) {
|
||||
window.location.reload();
|
||||
}
|
||||
|
|
|
|||
88
src/components/widgets/deploy/deploy.jsx
Normal file
88
src/components/widgets/deploy/deploy.jsx
Normal file
|
|
@ -0,0 +1,88 @@
|
|||
import { useState } from "react";
|
||||
import useSWR from "swr";
|
||||
|
||||
const STATUS_COLORS = {
|
||||
idle: "text-theme-500 dark:text-theme-400",
|
||||
running: "text-blue-500 dark:text-blue-400",
|
||||
success: "text-emerald-500 dark:text-emerald-400",
|
||||
failed: "text-red-500 dark:text-red-400",
|
||||
};
|
||||
|
||||
const STATUS_LABELS = {
|
||||
idle: "Never deployed",
|
||||
running: "Deploying...",
|
||||
success: "Deployed",
|
||||
failed: "Failed",
|
||||
};
|
||||
|
||||
function formatTime(iso) {
|
||||
if (!iso) return null;
|
||||
const d = new Date(iso);
|
||||
return d.toLocaleString("en-GB", { dateStyle: "short", timeStyle: "short", hourCycle: "h23" });
|
||||
}
|
||||
|
||||
export default function Deploy({ options }) {
|
||||
const { service, label, apiBase = "/deploy-api" } = options ?? {};
|
||||
const bp = process.env.NEXT_PUBLIC_BASE_PATH ?? "";
|
||||
// SWR middleware already prepends bp, so statusUrl must NOT include it
|
||||
const statusUrl = service ? `${apiBase}/status/${service}` : null;
|
||||
|
||||
const { data, mutate } = useSWR(statusUrl, {
|
||||
refreshInterval: (d) => (d?.status === "running" ? 2000 : 10000),
|
||||
});
|
||||
|
||||
const [triggering, setTriggering] = useState(false);
|
||||
|
||||
const status = data?.status ?? "idle";
|
||||
const lastRun = data?.last_run ? formatTime(data.last_run) : null;
|
||||
const isRunning = status === "running";
|
||||
|
||||
const handleDeploy = async () => {
|
||||
if (isRunning || triggering) return;
|
||||
setTriggering(true);
|
||||
try {
|
||||
await fetch(`${bp}${apiBase}/deploy/${service}`, { method: "POST" });
|
||||
await mutate();
|
||||
} finally {
|
||||
setTriggering(false);
|
||||
}
|
||||
};
|
||||
|
||||
if (!service) {
|
||||
return (
|
||||
<div className="flex flex-col items-center text-theme-500 text-xs p-2">
|
||||
<span>No service configured</span>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
||||
return (
|
||||
<div className="flex flex-col items-center justify-center p-2 gap-1 w-full">
|
||||
<div className={`text-xs font-semibold ${STATUS_COLORS[status] ?? STATUS_COLORS.idle}`}>
|
||||
{isRunning ? (
|
||||
<span className="animate-pulse">{STATUS_LABELS.running}</span>
|
||||
) : (
|
||||
STATUS_LABELS[status] ?? status
|
||||
)}
|
||||
</div>
|
||||
|
||||
{lastRun && (
|
||||
<div className="text-theme-500 dark:text-theme-400 text-xs opacity-75">{lastRun}</div>
|
||||
)}
|
||||
|
||||
<button
|
||||
type="button"
|
||||
onClick={handleDeploy}
|
||||
disabled={isRunning || triggering}
|
||||
className={[
|
||||
"mt-1 px-3 py-1 rounded text-xs font-medium transition-colors",
|
||||
isRunning || triggering
|
||||
? "bg-theme-300 dark:bg-theme-600 text-theme-500 dark:text-theme-400 cursor-not-allowed"
|
||||
: "bg-theme-500 hover:bg-theme-600 dark:bg-theme-600 dark:hover:bg-theme-500 text-white cursor-pointer",
|
||||
].join(" ")}
|
||||
>
|
||||
{isRunning ? "Running..." : `Deploy ${label ?? service}`}
|
||||
</button>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
|
@ -106,7 +106,7 @@ export default function Search({ options }) {
|
|||
query.trim().length > 0 &&
|
||||
query.trim() !== searchSuggestions[0]
|
||||
) {
|
||||
fetch(`/api/search/searchSuggestion?query=${encodeURIComponent(query)}&providerName=${selectedProvider.name}`, {
|
||||
fetch(`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/search/searchSuggestion?query=${encodeURIComponent(query)}&providerName=${selectedProvider.name}`, {
|
||||
signal: abortController.signal,
|
||||
})
|
||||
.then(async (searchSuggestionResult) => {
|
||||
|
|
|
|||
|
|
@ -15,6 +15,7 @@ const widgetMappings = {
|
|||
longhorn: dynamic(() => import("components/widgets/longhorn/longhorn")),
|
||||
kubernetes: dynamic(() => import("components/widgets/kubernetes/kubernetes")),
|
||||
stocks: dynamic(() => import("components/widgets/stocks/stocks")),
|
||||
deploy: dynamic(() => import("components/widgets/deploy/deploy"), { ssr: false }),
|
||||
};
|
||||
|
||||
export default function Widget({ widget, style }) {
|
||||
|
|
|
|||
|
|
@ -67,11 +67,18 @@ const tailwindSafelist = [
|
|||
"2xl:h-0 2xl:h-1 2xl:h-2 2xl:h-3 2xl:h-4 2xl:h-5 2xl:h-6 2xl:h-7 2xl:h-8 2xl:h-9 2xl:h-10 2xl:h-11 2xl:h-12 2xl:h-13 2xl:h-14 2xl:h-15 2xl:h-16 2xl:h-17 2xl:h-18 2xl:h-19 2xl:h-20 2xl:h-21 2xl:h-22 2xl:h-23 2xl:h-24 2xl:h-25 2xl:h-26 2xl:h-27 2xl:h-28 2xl:h-29 2xl:h-30 2xl:h-31 2xl:h-32 2xl:h-33 2xl:h-34 2xl:h-35 2xl:h-36 2xl:h-37 2xl:h-38 2xl:h-39 2xl:h-40 2xl:h-41 2xl:h-42 2xl:h-43 2xl:h-44 2xl:h-45 2xl:h-46 2xl:h-47 2xl:h-48 2xl:h-49 2xl:h-50 2xl:h-51 2xl:h-52 2xl:h-53 2xl:h-54 2xl:h-55 2xl:h-56 2xl:h-57 2xl:h-58 2xl:h-59 2xl:h-60 2xl:h-61 2xl:h-62 2xl:h-63 2xl:h-64 2xl:h-65 2xl:h-66 2xl:h-67 2xl:h-68 2xl:h-69 2xl:h-70 2xl:h-71 2xl:h-72 2xl:h-73 2xl:h-74 2xl:h-75 2xl:h-76 2xl:h-77 2xl:h-78 2xl:h-79 2xl:h-80 2xl:h-81 2xl:h-82 2xl:h-83 2xl:h-84 2xl:h-85 2xl:h-86 2xl:h-87 2xl:h-88 2xl:h-89 2xl:h-90 2xl:h-91 2xl:h-92 2xl:h-93 2xl:h-94 2xl:h-95 2xl:h-96",
|
||||
];
|
||||
|
||||
const basePathMiddleware = (useSWRNext) => (key, fetcher, config) => {
|
||||
const bp = process.env.NEXT_PUBLIC_BASE_PATH ?? "";
|
||||
const adjustedKey = bp && typeof key === "string" ? `${bp}${key}` : key;
|
||||
return useSWRNext(adjustedKey, fetcher, config);
|
||||
};
|
||||
|
||||
function MyApp({ Component, pageProps }) {
|
||||
return (
|
||||
<SWRConfig
|
||||
value={{
|
||||
fetcher: (resource, init) => fetch(resource, init).then((res) => res.json()),
|
||||
use: [basePathMiddleware],
|
||||
}}
|
||||
>
|
||||
<Head>
|
||||
|
|
|
|||
|
|
@ -6,8 +6,8 @@ export default function Document() {
|
|||
<Head>
|
||||
<meta name="mobile-web-app-capable" content="yes" />
|
||||
<link rel="manifest" href="/site.webmanifest?v=4" crossOrigin="use-credentials" />
|
||||
<link rel="preload" href="/api/config/custom.css" as="style" />
|
||||
<link rel="stylesheet" href="/api/config/custom.css" /> {/* eslint-disable-line @next/next/no-css-tags */}
|
||||
<link rel="preload" href={`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/config/custom.css`} as="style" />
|
||||
<link rel="stylesheet" href={`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/config/custom.css`} /> {/* eslint-disable-line @next/next/no-css-tags */}
|
||||
</Head>
|
||||
<body>
|
||||
<Main />
|
||||
|
|
|
|||
|
|
@ -63,14 +63,15 @@ export async function getStaticProps() {
|
|||
const widgets = await widgetsResponse();
|
||||
const language = normalizeLanguage(settings.language);
|
||||
|
||||
const bp = process.env.NEXT_PUBLIC_BASE_PATH ?? "";
|
||||
return {
|
||||
props: {
|
||||
initialSettings: settings,
|
||||
fallback: {
|
||||
"/api/services": services,
|
||||
"/api/bookmarks": bookmarks,
|
||||
"/api/widgets": widgets,
|
||||
"/api/hash": false,
|
||||
[`${bp}/api/services`]: services,
|
||||
[`${bp}/api/bookmarks`]: bookmarks,
|
||||
[`${bp}/api/widgets`]: widgets,
|
||||
[`${bp}/api/hash`]: false,
|
||||
},
|
||||
...(await serverSideTranslations(language)),
|
||||
},
|
||||
|
|
@ -83,10 +84,10 @@ export async function getStaticProps() {
|
|||
props: {
|
||||
initialSettings: {},
|
||||
fallback: {
|
||||
"/api/services": [],
|
||||
"/api/bookmarks": [],
|
||||
"/api/widgets": [],
|
||||
"/api/hash": false,
|
||||
[`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/services`]: [],
|
||||
[`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/bookmarks`]: [],
|
||||
[`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/widgets`]: [],
|
||||
[`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/hash`]: false,
|
||||
},
|
||||
...(await serverSideTranslations("en")),
|
||||
},
|
||||
|
|
@ -120,7 +121,7 @@ function Index({ initialSettings, fallback }) {
|
|||
setStale(true);
|
||||
localStorage.setItem("hash", hashData.hash);
|
||||
|
||||
fetch("/api/revalidate").then((res) => {
|
||||
fetch(`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/revalidate`).then((res) => {
|
||||
if (res.ok) {
|
||||
window.location.reload();
|
||||
}
|
||||
|
|
@ -434,7 +435,7 @@ function Home({ initialSettings }) {
|
|||
<meta name="color-scheme" content="dark light"></meta>
|
||||
</Head>
|
||||
|
||||
<Script src="/api/config/custom.js" />
|
||||
<Script src={`${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/config/custom.js`} />
|
||||
|
||||
<div
|
||||
className={classNames(
|
||||
|
|
|
|||
|
|
@ -435,6 +435,11 @@ export function cleanServiceGroups(groups) {
|
|||
|
||||
// grafana
|
||||
alerts,
|
||||
|
||||
// deploy
|
||||
service: deployService,
|
||||
label: deployLabel,
|
||||
apiBase: deployApiBase,
|
||||
} = widgetData;
|
||||
|
||||
let fieldsList = fields;
|
||||
|
|
@ -685,6 +690,11 @@ export function cleanServiceGroups(groups) {
|
|||
if (type === "grafana") {
|
||||
if (alerts) widget.alerts = alerts;
|
||||
}
|
||||
if (type === "deploy") {
|
||||
if (deployService) widget.service = deployService;
|
||||
if (deployLabel) widget.label = deployLabel;
|
||||
if (deployApiBase) widget.apiBase = deployApiBase;
|
||||
}
|
||||
if (type === "unraid") {
|
||||
if (pool1) widget.pool1 = pool1;
|
||||
if (pool2) widget.pool2 = pool2;
|
||||
|
|
|
|||
|
|
@ -29,7 +29,7 @@ export function formatProxyUrl(widget, endpoint, queryParams) {
|
|||
if (queryParams) {
|
||||
params.append("query", JSON.stringify(queryParams));
|
||||
}
|
||||
return `/api/services/proxy?${params.toString()}`;
|
||||
return `${process.env.NEXT_PUBLIC_BASE_PATH ?? ""}/api/services/proxy?${params.toString()}`;
|
||||
}
|
||||
|
||||
export function asJson(data) {
|
||||
|
|
|
|||
|
|
@ -26,6 +26,7 @@ const components = {
|
|||
iframe: dynamic(() => import("./iframe/component")),
|
||||
customapi: dynamic(() => import("./customapi/component")),
|
||||
deluge: dynamic(() => import("./deluge/component")),
|
||||
deploy: dynamic(() => import("./deploy/component")),
|
||||
develancacheui: dynamic(() => import("./develancacheui/component")),
|
||||
diskstation: dynamic(() => import("./diskstation/component")),
|
||||
dispatcharr: dynamic(() => import("./dispatcharr/component")),
|
||||
|
|
@ -147,6 +148,7 @@ const components = {
|
|||
tubearchivist: dynamic(() => import("./tubearchivist/component")),
|
||||
truenas: dynamic(() => import("./truenas/component")),
|
||||
unifi: dynamic(() => import("./unifi/component")),
|
||||
unifi_drive: dynamic(() => import("./unifi_drive/component")),
|
||||
unmanic: dynamic(() => import("./unmanic/component")),
|
||||
unraid: dynamic(() => import("./unraid/component")),
|
||||
uptimekuma: dynamic(() => import("./uptimekuma/component")),
|
||||
|
|
|
|||
78
src/widgets/deploy/component.jsx
Normal file
78
src/widgets/deploy/component.jsx
Normal file
|
|
@ -0,0 +1,78 @@
|
|||
import { useState } from "react";
|
||||
import useSWR from "swr";
|
||||
|
||||
const STATUS_COLORS = {
|
||||
idle: "text-theme-500 dark:text-theme-400",
|
||||
running: "text-blue-500 dark:text-blue-400",
|
||||
success: "text-emerald-500 dark:text-emerald-400",
|
||||
failed: "text-red-500 dark:text-red-400",
|
||||
};
|
||||
|
||||
const STATUS_LABELS = {
|
||||
idle: "Never deployed",
|
||||
running: "Deploying...",
|
||||
success: "Deployed",
|
||||
failed: "Failed",
|
||||
};
|
||||
|
||||
function formatTime(iso) {
|
||||
if (!iso) return null;
|
||||
const d = new Date(iso);
|
||||
return d.toLocaleString("en-GB", { dateStyle: "short", timeStyle: "short", hourCycle: "h23" });
|
||||
}
|
||||
|
||||
export default function DeployComponent({ service }) {
|
||||
const { service: svcName, label, apiBase = "/deploy-api" } = service?.widget ?? {};
|
||||
const bp = process.env.NEXT_PUBLIC_BASE_PATH ?? "";
|
||||
// SWR middleware already prepends bp, so statusUrl must NOT include it
|
||||
const statusUrl = svcName ? `${apiBase}/status/${svcName}` : null;
|
||||
|
||||
const { data, mutate } = useSWR(statusUrl, {
|
||||
refreshInterval: (d) => (d?.status === "running" ? 2000 : 10000),
|
||||
});
|
||||
|
||||
const [triggering, setTriggering] = useState(false);
|
||||
|
||||
const status = data?.status ?? "idle";
|
||||
const lastRun = data?.last_run ? formatTime(data.last_run) : null;
|
||||
const isRunning = status === "running";
|
||||
|
||||
const handleDeploy = async () => {
|
||||
if (isRunning || triggering) return;
|
||||
setTriggering(true);
|
||||
try {
|
||||
await fetch(`${bp}${apiBase}/deploy/${svcName}`, { method: "POST" });
|
||||
await mutate();
|
||||
} finally {
|
||||
setTriggering(false);
|
||||
}
|
||||
};
|
||||
|
||||
if (!svcName) return null;
|
||||
|
||||
return (
|
||||
<div className="flex flex-row items-center justify-between px-2 py-1 gap-2 w-full">
|
||||
<div className="flex flex-col">
|
||||
<span className={`text-xs font-semibold ${STATUS_COLORS[status] ?? STATUS_COLORS.idle}`}>
|
||||
{isRunning ? <span className="animate-pulse">{STATUS_LABELS.running}</span> : (STATUS_LABELS[status] ?? status)}
|
||||
</span>
|
||||
{lastRun && (
|
||||
<span className="text-theme-500 dark:text-theme-400 text-xs opacity-60">{lastRun}</span>
|
||||
)}
|
||||
</div>
|
||||
<button
|
||||
type="button"
|
||||
onClick={handleDeploy}
|
||||
disabled={isRunning || triggering}
|
||||
className={[
|
||||
"px-3 py-1 rounded text-xs font-medium transition-colors shrink-0",
|
||||
isRunning || triggering
|
||||
? "bg-theme-300 dark:bg-theme-600 text-theme-500 dark:text-theme-400 cursor-not-allowed"
|
||||
: "bg-theme-500 hover:bg-theme-600 dark:bg-theme-600 dark:hover:bg-theme-500 text-white cursor-pointer",
|
||||
].join(" ")}
|
||||
>
|
||||
{isRunning ? "Running..." : `Deploy ${label ?? svcName}`}
|
||||
</button>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
|
|
@ -12,7 +12,7 @@ async function login(widget) {
|
|||
const loginParams = {
|
||||
method: "POST",
|
||||
headers: { "Content-Type": "application/json" },
|
||||
body: null,
|
||||
body: "{}",
|
||||
};
|
||||
|
||||
if (widget.username && widget.password) {
|
||||
|
|
|
|||
|
|
@ -45,7 +45,7 @@ describe("widgets/flood/proxy", () => {
|
|||
expect(httpProxy).toHaveBeenCalledTimes(3);
|
||||
expect(httpProxy.mock.calls[0][0].toString()).toBe("http://flood/api/stats");
|
||||
expect(httpProxy.mock.calls[1][0]).toBe("http://flood/api/auth/authenticate");
|
||||
expect(httpProxy.mock.calls[1][1].body).toBeNull();
|
||||
expect(httpProxy.mock.calls[1][1].body).toBe("{}");
|
||||
expect(httpProxy.mock.calls[2][0].toString()).toBe("http://flood/api/stats");
|
||||
expect(res.statusCode).toBe(200);
|
||||
expect(res.body).toEqual(Buffer.from("data"));
|
||||
|
|
|
|||
58
src/widgets/unifi_drive/component.jsx
Normal file
58
src/widgets/unifi_drive/component.jsx
Normal file
|
|
@ -0,0 +1,58 @@
|
|||
import Block from "components/services/widget/block";
|
||||
import Container from "components/services/widget/container";
|
||||
import { useTranslation } from "next-i18next";
|
||||
|
||||
import useWidgetAPI from "utils/proxy/use-widget-api";
|
||||
|
||||
export default function Component({ service }) {
|
||||
const { t } = useTranslation();
|
||||
const { widget } = service;
|
||||
|
||||
const { data: storageData, error: storageError } = useWidgetAPI(widget, "storage");
|
||||
|
||||
if (storageError) {
|
||||
return <Container service={service} error={storageError} />;
|
||||
}
|
||||
|
||||
if (!storageData) {
|
||||
return (
|
||||
<Container service={service}>
|
||||
<Block field="unifi_drive.total" label="resources.total" />
|
||||
<Block field="unifi_drive.used" label="resources.used" />
|
||||
<Block field="unifi_drive.available" label="resources.free" />
|
||||
<Block field="unifi_drive.status" label="widget.status" />
|
||||
</Container>
|
||||
);
|
||||
}
|
||||
|
||||
const { data: storage } = storageData;
|
||||
|
||||
if (!storage) {
|
||||
return (
|
||||
<Container service={service}>
|
||||
<Block value={t("unifi_drive.no_data")} />
|
||||
</Container>
|
||||
);
|
||||
}
|
||||
|
||||
const { totalQuota, usage, status } = storage;
|
||||
const totalBytes = totalQuota ?? 0;
|
||||
const usedBytes = (usage?.system || 0) + (usage?.myDrives || 0) + (usage?.sharedDrives || 0);
|
||||
const availableBytes = Math.max(0, totalBytes - usedBytes);
|
||||
let statusValue = status;
|
||||
if (status === "healthy") statusValue = t("unifi_drive.healthy");
|
||||
else if (status === "degraded") statusValue = t("unifi_drive.degraded");
|
||||
|
||||
return (
|
||||
<Container service={service}>
|
||||
<Block field="unifi_drive.total" label="resources.total" value={t("common.bytes", { value: totalBytes })} />
|
||||
<Block field="unifi_drive.used" label="resources.used" value={t("common.bytes", { value: usedBytes })} />
|
||||
<Block
|
||||
field="unifi_drive.available"
|
||||
label="resources.free"
|
||||
value={t("common.bytes", { value: availableBytes })}
|
||||
/>
|
||||
<Block field="unifi_drive.status" label="widget.status" value={statusValue} />
|
||||
</Container>
|
||||
);
|
||||
}
|
||||
92
src/widgets/unifi_drive/component.test.jsx
Normal file
92
src/widgets/unifi_drive/component.test.jsx
Normal file
|
|
@ -0,0 +1,92 @@
|
|||
// @vitest-environment jsdom
|
||||
|
||||
import { screen } from "@testing-library/react";
|
||||
import { beforeEach, describe, expect, it, vi } from "vitest";
|
||||
|
||||
import { renderWithProviders } from "test-utils/render-with-providers";
|
||||
import { expectBlockValue } from "test-utils/widget-assertions";
|
||||
|
||||
const { useWidgetAPI } = vi.hoisted(() => ({ useWidgetAPI: vi.fn() }));
|
||||
|
||||
vi.mock("utils/proxy/use-widget-api", () => ({ default: useWidgetAPI }));
|
||||
|
||||
import Component from "./component";
|
||||
|
||||
describe("widgets/unifi_drive/component", () => {
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
});
|
||||
|
||||
it("renders placeholders while loading", () => {
|
||||
useWidgetAPI.mockReturnValue({ data: undefined, error: undefined });
|
||||
|
||||
const service = { widget: { type: "unifi_drive" } };
|
||||
const { container } = renderWithProviders(<Component service={service} />, { settings: { hideErrors: false } });
|
||||
|
||||
expect(container.querySelectorAll(".service-block")).toHaveLength(4);
|
||||
expect(screen.getByText("resources.total")).toBeInTheDocument();
|
||||
expect(screen.getByText("resources.used")).toBeInTheDocument();
|
||||
expect(screen.getByText("resources.free")).toBeInTheDocument();
|
||||
expect(screen.getByText("widget.status")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("renders error when API fails", () => {
|
||||
useWidgetAPI.mockReturnValue({ data: undefined, error: new Error("fail") });
|
||||
|
||||
const service = { widget: { type: "unifi_drive" } };
|
||||
renderWithProviders(<Component service={service} />, { settings: { hideErrors: false } });
|
||||
|
||||
expect(screen.getAllByText("widget.api_error", { exact: false }).length).toBeGreaterThan(0);
|
||||
});
|
||||
|
||||
it("renders no_data when storage data is missing", () => {
|
||||
useWidgetAPI.mockReturnValue({ data: { data: null }, error: undefined });
|
||||
|
||||
const service = { widget: { type: "unifi_drive" } };
|
||||
renderWithProviders(<Component service={service} />, { settings: { hideErrors: false } });
|
||||
|
||||
expect(screen.getByText("unifi_drive.no_data")).toBeInTheDocument();
|
||||
});
|
||||
|
||||
it("renders storage statistics when data is loaded", () => {
|
||||
useWidgetAPI.mockReturnValue({
|
||||
data: {
|
||||
data: {
|
||||
totalQuota: 1000000000000,
|
||||
usage: { system: 100000000000, myDrives: 200000000000, sharedDrives: 50000000000 },
|
||||
status: "healthy",
|
||||
},
|
||||
},
|
||||
error: undefined,
|
||||
});
|
||||
|
||||
const service = { widget: { type: "unifi_drive" } };
|
||||
const { container } = renderWithProviders(<Component service={service} />, { settings: { hideErrors: false } });
|
||||
|
||||
expect(container.querySelectorAll(".service-block")).toHaveLength(4);
|
||||
expectBlockValue(container, "resources.total", 1000000000000);
|
||||
expectBlockValue(container, "resources.used", 350000000000);
|
||||
expectBlockValue(container, "resources.free", 650000000000);
|
||||
expectBlockValue(container, "widget.status", "unifi_drive.healthy");
|
||||
});
|
||||
|
||||
it("renders degraded status", () => {
|
||||
useWidgetAPI.mockReturnValue({
|
||||
data: {
|
||||
data: {
|
||||
totalQuota: 100,
|
||||
usage: { system: 10, myDrives: 20, sharedDrives: 5 },
|
||||
status: "degraded",
|
||||
},
|
||||
},
|
||||
error: undefined,
|
||||
});
|
||||
|
||||
const service = { widget: { type: "unifi_drive" } };
|
||||
const { container } = renderWithProviders(<Component service={service} />, { settings: { hideErrors: false } });
|
||||
|
||||
expect(container.querySelectorAll(".service-block")).toHaveLength(4);
|
||||
expectBlockValue(container, "widget.status", "unifi_drive.degraded");
|
||||
expectBlockValue(container, "resources.free", 65);
|
||||
});
|
||||
});
|
||||
36
src/widgets/unifi_drive/proxy.js
Normal file
36
src/widgets/unifi_drive/proxy.js
Normal file
|
|
@ -0,0 +1,36 @@
|
|||
import getServiceWidget from "utils/config/service-helpers";
|
||||
import createUnifiProxyHandler from "utils/proxy/handlers/unifi";
|
||||
import { httpProxy } from "utils/proxy/http";
|
||||
|
||||
const drivePrefix = "/proxy/drive";
|
||||
|
||||
async function getWidget(req, logger) {
|
||||
const { group, service, index } = req.query;
|
||||
if (!group || !service) return null;
|
||||
|
||||
const widget = await getServiceWidget(group, service, index);
|
||||
if (!widget) {
|
||||
logger.debug("Invalid or missing widget for service '%s' in group '%s'", service, group);
|
||||
return null;
|
||||
}
|
||||
return widget;
|
||||
}
|
||||
|
||||
async function resolveRequestContext({ cachedPrefix, widget }) {
|
||||
if (cachedPrefix !== null) {
|
||||
return { prefix: cachedPrefix };
|
||||
}
|
||||
|
||||
const [, , , responseHeaders] = await httpProxy(widget.url);
|
||||
|
||||
return {
|
||||
prefix: drivePrefix,
|
||||
csrfToken: responseHeaders?.["x-csrf-token"],
|
||||
};
|
||||
}
|
||||
|
||||
export default createUnifiProxyHandler({
|
||||
proxyName: "unifiDriveProxyHandler",
|
||||
resolveWidget: getWidget,
|
||||
resolveRequestContext,
|
||||
});
|
||||
82
src/widgets/unifi_drive/proxy.test.js
Normal file
82
src/widgets/unifi_drive/proxy.test.js
Normal file
|
|
@ -0,0 +1,82 @@
|
|||
import { beforeEach, describe, expect, it, vi } from "vitest";
|
||||
|
||||
import createMockRes from "test-utils/create-mock-res";
|
||||
|
||||
const { httpProxy, getServiceWidget, cache, logger } = vi.hoisted(() => {
|
||||
const store = new Map();
|
||||
return {
|
||||
httpProxy: vi.fn(),
|
||||
getServiceWidget: vi.fn(),
|
||||
cache: {
|
||||
get: vi.fn((k) => (store.has(k) ? store.get(k) : null)),
|
||||
put: vi.fn((k, v) => store.set(k, v)),
|
||||
del: vi.fn((k) => store.delete(k)),
|
||||
_reset: () => store.clear(),
|
||||
},
|
||||
logger: { debug: vi.fn(), error: vi.fn() },
|
||||
};
|
||||
});
|
||||
|
||||
vi.mock("memory-cache", () => ({ default: cache, ...cache }));
|
||||
vi.mock("utils/logger", () => ({ default: () => logger }));
|
||||
vi.mock("utils/config/service-helpers", () => ({ default: getServiceWidget }));
|
||||
vi.mock("utils/proxy/http", () => ({ httpProxy }));
|
||||
vi.mock("widgets/widgets", () => ({
|
||||
default: { unifi_drive: { api: "{url}{prefix}/api/{endpoint}" } },
|
||||
}));
|
||||
|
||||
import unifiDriveProxyHandler from "./proxy";
|
||||
|
||||
const widgetConfig = { type: "unifi_drive", url: "http://unifi", username: "u", password: "p" };
|
||||
|
||||
describe("widgets/unifi_drive/proxy", () => {
|
||||
beforeEach(() => {
|
||||
vi.clearAllMocks();
|
||||
cache._reset();
|
||||
});
|
||||
|
||||
it("returns 400 when widget config is missing", async () => {
|
||||
getServiceWidget.mockResolvedValue(null);
|
||||
const res = createMockRes();
|
||||
await unifiDriveProxyHandler(
|
||||
{ query: { group: "g", service: "s", endpoint: "v1/systems/storage?type=detail" } },
|
||||
res,
|
||||
);
|
||||
expect(res.statusCode).toBe(400);
|
||||
});
|
||||
|
||||
it("returns 403 when widget type has no API config", async () => {
|
||||
getServiceWidget.mockResolvedValue({ ...widgetConfig, type: "unknown" });
|
||||
const res = createMockRes();
|
||||
await unifiDriveProxyHandler({ query: { group: "g", service: "s", endpoint: "storage" } }, res);
|
||||
expect(res.statusCode).toBe(403);
|
||||
});
|
||||
|
||||
it("uses /proxy/drive prefix and returns data on success", async () => {
|
||||
getServiceWidget.mockResolvedValue({ ...widgetConfig });
|
||||
httpProxy
|
||||
.mockResolvedValueOnce([200, "text/html", Buffer.from(""), {}])
|
||||
.mockResolvedValueOnce([200, "application/json", Buffer.from('{"data":{}}'), {}]);
|
||||
|
||||
const res = createMockRes();
|
||||
await unifiDriveProxyHandler({ query: { group: "g", service: "s", endpoint: "storage" } }, res);
|
||||
|
||||
expect(httpProxy.mock.calls[0][0]).toBe("http://unifi");
|
||||
expect(httpProxy.mock.calls[1][0].toString()).toContain("/proxy/drive/api/");
|
||||
expect(cache.put).toHaveBeenCalledWith("unifiDriveProxyHandler__prefix.s", "/proxy/drive");
|
||||
expect(res.statusCode).toBe(200);
|
||||
});
|
||||
|
||||
it("skips prefix detection when cached", async () => {
|
||||
getServiceWidget.mockResolvedValue({ ...widgetConfig });
|
||||
cache.put("unifiDriveProxyHandler__prefix.s", "/proxy/drive");
|
||||
httpProxy.mockResolvedValueOnce([200, "application/json", Buffer.from('{"data":{}}'), {}]);
|
||||
|
||||
const res = createMockRes();
|
||||
await unifiDriveProxyHandler({ query: { group: "g", service: "s", endpoint: "storage" } }, res);
|
||||
|
||||
expect(httpProxy).toHaveBeenCalledTimes(1);
|
||||
expect(httpProxy.mock.calls[0][0].toString()).toContain("/proxy/drive/api/");
|
||||
expect(res.statusCode).toBe(200);
|
||||
});
|
||||
});
|
||||
14
src/widgets/unifi_drive/widget.js
Normal file
14
src/widgets/unifi_drive/widget.js
Normal file
|
|
@ -0,0 +1,14 @@
|
|||
import unifiDriveProxyHandler from "./proxy";
|
||||
|
||||
const widget = {
|
||||
api: "{url}{prefix}/api/{endpoint}",
|
||||
proxyHandler: unifiDriveProxyHandler,
|
||||
|
||||
mappings: {
|
||||
storage: {
|
||||
endpoint: "v1/systems/storage?type=detail",
|
||||
},
|
||||
},
|
||||
};
|
||||
|
||||
export default widget;
|
||||
11
src/widgets/unifi_drive/widget.test.js
Normal file
11
src/widgets/unifi_drive/widget.test.js
Normal file
|
|
@ -0,0 +1,11 @@
|
|||
import { describe, it } from "vitest";
|
||||
|
||||
import { expectWidgetConfigShape } from "test-utils/widget-config";
|
||||
|
||||
import widget from "./widget";
|
||||
|
||||
describe("unifi_drive widget config", () => {
|
||||
it("exports a valid widget config", () => {
|
||||
expectWidgetConfigShape(widget);
|
||||
});
|
||||
});
|
||||
|
|
@ -137,6 +137,7 @@ import trilium from "./trilium/widget";
|
|||
import truenas from "./truenas/widget";
|
||||
import tubearchivist from "./tubearchivist/widget";
|
||||
import unifi from "./unifi/widget";
|
||||
import unifi_drive from "./unifi_drive/widget";
|
||||
import unmanic from "./unmanic/widget";
|
||||
import unraid from "./unraid/widget";
|
||||
import uptimekuma from "./uptimekuma/widget";
|
||||
|
|
@ -296,6 +297,7 @@ const widgets = {
|
|||
truenas,
|
||||
unifi,
|
||||
unifi_console: unifi,
|
||||
unifi_drive,
|
||||
unmanic,
|
||||
unraid,
|
||||
uptimekuma,
|
||||
|
|
|
|||
Loading…
Add table
Reference in a new issue