Compare commits
2 Commits
v1.0.3
...
e103829870
| Author | SHA1 | Date | |
|---|---|---|---|
| e103829870 | |||
| 2714907ff4 |
@@ -1,46 +0,0 @@
|
|||||||
name: Build and Publish Docker Image
|
|
||||||
run-name: Publishing ${{ gitea.ref_name }}
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
tags:
|
|
||||||
- "v*"
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
deploy:
|
|
||||||
runs-on: ubuntu-latest
|
|
||||||
steps:
|
|
||||||
- name: Checkout
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
|
|
||||||
- name: Setup Go
|
|
||||||
uses: actions/setup-go@v5
|
|
||||||
with:
|
|
||||||
go-version-file: go.mod
|
|
||||||
cache: false
|
|
||||||
|
|
||||||
- name: Run Tests
|
|
||||||
run: go test ./...
|
|
||||||
|
|
||||||
- name: Install Docker
|
|
||||||
run: curl -fsSL https://get.docker.com | sh
|
|
||||||
|
|
||||||
- name: Build Docker Image
|
|
||||||
run: |
|
|
||||||
docker build \
|
|
||||||
--build-arg APP_VERSION=${{ gitea.ref_name }} \
|
|
||||||
-t tea.chunkbyte.com/kato/warpbox:${{ gitea.ref_name }} \
|
|
||||||
-t tea.chunkbyte.com/kato/warpbox:latest \
|
|
||||||
.
|
|
||||||
|
|
||||||
- name: Login to Gitea Container Registry
|
|
||||||
uses: docker/login-action@v3
|
|
||||||
with:
|
|
||||||
registry: tea.chunkbyte.com
|
|
||||||
username: ${{ secrets.REGISTRY_USERNAME }}
|
|
||||||
password: ${{ secrets.REGISTRY_PASSWORD }}
|
|
||||||
|
|
||||||
- name: Push Docker Image
|
|
||||||
run: |
|
|
||||||
docker push tea.chunkbyte.com/kato/warpbox:${{ gitea.ref_name }}
|
|
||||||
docker push tea.chunkbyte.com/kato/warpbox:latest
|
|
||||||
20
Dockerfile
20
Dockerfile
@@ -1,8 +1,6 @@
|
|||||||
# Stage 1: Build
|
# Stage 1: Build
|
||||||
FROM golang:1.23-alpine AS builder
|
FROM golang:1.23-alpine AS builder
|
||||||
|
|
||||||
ARG APP_VERSION=""
|
|
||||||
|
|
||||||
RUN apk add --no-cache git ca-certificates
|
RUN apk add --no-cache git ca-certificates
|
||||||
|
|
||||||
WORKDIR /build
|
WORKDIR /build
|
||||||
@@ -18,19 +16,12 @@ COPY static/ static/
|
|||||||
COPY templates/ templates/
|
COPY templates/ templates/
|
||||||
|
|
||||||
# Build the binary
|
# Build the binary
|
||||||
RUN CGO_ENABLED=0 GOOS=linux go build -ldflags="-s -w" -o warpbox ./cmd/
|
RUN CGO_ENABLED=0 GOOS=linux go build -ldflags="-s -w" -o warpbox ./cmd/main.go
|
||||||
|
|
||||||
# Stage 2: Runtime
|
# Stage 2: Runtime
|
||||||
FROM alpine:3.21
|
FROM alpine:3.21
|
||||||
|
|
||||||
ARG APP_VERSION=""
|
RUN apk add --no-cache ca-certificates tzdata
|
||||||
ENV APP_VERSION=${APP_VERSION}
|
|
||||||
|
|
||||||
RUN apk add \
|
|
||||||
--no-cache \
|
|
||||||
ca-certificates \
|
|
||||||
tzdata \
|
|
||||||
wget
|
|
||||||
|
|
||||||
# Create non-root user
|
# Create non-root user
|
||||||
RUN addgroup -S warpbox && adduser -S warpbox -G warpbox
|
RUN addgroup -S warpbox && adduser -S warpbox -G warpbox
|
||||||
@@ -59,8 +50,8 @@ ENV WARPBOX_DATA_DIR=/app/data \
|
|||||||
WARPBOX_ONE_TIME_DOWNLOAD_EXPIRY_SECONDS=604800 \
|
WARPBOX_ONE_TIME_DOWNLOAD_EXPIRY_SECONDS=604800 \
|
||||||
WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE=false \
|
WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE=false \
|
||||||
WARPBOX_ADMIN_ENABLED=true \
|
WARPBOX_ADMIN_ENABLED=true \
|
||||||
WARPBOX_GLOBAL_MAX_FILE_SIZE_GB=2 \
|
WARPBOX_GLOBAL_MAX_FILE_SIZE_MB=2048 \
|
||||||
WARPBOX_GLOBAL_MAX_BOX_SIZE_GB=4 \
|
WARPBOX_GLOBAL_MAX_BOX_SIZE_MB=4096 \
|
||||||
WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS=3600 \
|
WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS=3600 \
|
||||||
WARPBOX_MAX_GUEST_EXPIRY_SECONDS=172800 \
|
WARPBOX_MAX_GUEST_EXPIRY_SECONDS=172800 \
|
||||||
WARPBOX_BOX_POLL_INTERVAL_MS=5000 \
|
WARPBOX_BOX_POLL_INTERVAL_MS=5000 \
|
||||||
@@ -69,9 +60,6 @@ ENV WARPBOX_DATA_DIR=/app/data \
|
|||||||
|
|
||||||
EXPOSE 8080
|
EXPOSE 8080
|
||||||
|
|
||||||
HEALTHCHECK --interval=30s --timeout=3s --start-period=10s --retries=3 \
|
|
||||||
CMD wget -qO- http://127.0.0.1:8080/health >/dev/null || exit 1
|
|
||||||
|
|
||||||
VOLUME ["/app/data"]
|
VOLUME ["/app/data"]
|
||||||
|
|
||||||
CMD ["./warpbox", "run", "--addr", ":8080"]
|
CMD ["./warpbox", "run", "--addr", ":8080"]
|
||||||
|
|||||||
4
NOTICE
Normal file
4
NOTICE
Normal file
@@ -0,0 +1,4 @@
|
|||||||
|
WarpBox
|
||||||
|
Copyright (c) 2026 Daniel Legt
|
||||||
|
|
||||||
|
This product includes software developed by Daniel Legt.
|
||||||
36
README.md
36
README.md
@@ -86,8 +86,8 @@ go run ./cmd run --addr :3000
|
|||||||
## Configuration
|
## Configuration
|
||||||
|
|
||||||
WarpBox loads defaults, applies environment variables at startup, then applies
|
WarpBox loads defaults, applies environment variables at startup, then applies
|
||||||
safe admin settings overrides from BadgerDB. Storage path settings remain
|
safe admin settings overrides from BadgerDB. Hard storage and global limit
|
||||||
environment controlled.
|
settings remain environment controlled.
|
||||||
|
|
||||||
| Variable | Default | What it does |
|
| Variable | Default | What it does |
|
||||||
| --- | ---: | --- |
|
| --- | ---: | --- |
|
||||||
@@ -108,16 +108,16 @@ environment controlled.
|
|||||||
| `WARPBOX_RENEW_ON_DOWNLOAD_ENABLED` | `false` | Renews expiring boxes on download. |
|
| `WARPBOX_RENEW_ON_DOWNLOAD_ENABLED` | `false` | Renews expiring boxes on download. |
|
||||||
| `WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS` | `10` | Default guest retention. |
|
| `WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS` | `10` | Default guest retention. |
|
||||||
| `WARPBOX_MAX_GUEST_EXPIRY_SECONDS` | `172800` | Max guest retention shown/accepted. |
|
| `WARPBOX_MAX_GUEST_EXPIRY_SECONDS` | `172800` | Max guest retention shown/accepted. |
|
||||||
| `WARPBOX_GLOBAL_MAX_FILE_SIZE_GB` | `0` | Per-file cap in GB using `1024^3` conversion; `0` means unlimited. Decimals allowed, like `0.5`. |
|
| `WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES` | `0` | Hard per-file cap; `0` means unlimited. |
|
||||||
| `WARPBOX_GLOBAL_MAX_BOX_SIZE_GB` | `0` | Per-box cap in GB using `1024^3` conversion; `0` means unlimited. Decimals allowed. |
|
| `WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES` | `0` | Hard per-box cap; `0` means unlimited. |
|
||||||
| `WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_GB` | `0` | Default user file cap in GB using `1024^3` conversion. |
|
| `WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES` | `0` | Default user file cap. |
|
||||||
| `WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_GB` | `0` | Default user box cap in GB using `1024^3` conversion. |
|
| `WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES` | `0` | Default user box cap. |
|
||||||
| `WARPBOX_SESSION_TTL_SECONDS` | `86400` | Admin session lifetime. |
|
| `WARPBOX_SESSION_TTL_SECONDS` | `86400` | Admin session lifetime. |
|
||||||
| `WARPBOX_BOX_POLL_INTERVAL_MS` | `5000` | Browser polling interval for box/file status updates. |
|
| `WARPBOX_BOX_POLL_INTERVAL_MS` | `5000` | Browser polling interval for box/file status updates. |
|
||||||
| `WARPBOX_THUMBNAIL_BATCH_SIZE` | `10` | Number of pending thumbnails processed per worker pass. |
|
| `WARPBOX_THUMBNAIL_BATCH_SIZE` | `10` | Number of pending thumbnails processed per worker pass. |
|
||||||
| `WARPBOX_THUMBNAIL_INTERVAL_SECONDS` | `30` | Delay between thumbnail worker passes. |
|
| `WARPBOX_THUMBNAIL_INTERVAL_SECONDS` | `30` | Delay between thumbnail worker passes. |
|
||||||
|
|
||||||
Legacy `_MB` and `_BYTES` size env names are still accepted for compatibility, but GB env names are the intended format now. GB input uses `1024^3` bytes so UI limits and displayed space stay consistent.
|
Size limits also accept `_MB` variants for the same settings.
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
|
|
||||||
@@ -189,25 +189,3 @@ keeps most behavior easy to follow from the Go handlers and the small browser
|
|||||||
scripts.
|
scripts.
|
||||||
|
|
||||||
For a short implementation overview, see [docs/tech.md](docs/tech.md).
|
For a short implementation overview, see [docs/tech.md](docs/tech.md).
|
||||||
|
|
||||||
## Docker / Podman
|
|
||||||
|
|
||||||
If you are using Podman, please pay attention in the [docker-compose.yml](./docker-compose.example.yml) example
|
|
||||||
file that has been provided, there's comments in regards to differences between the two.
|
|
||||||
|
|
||||||
|
|
||||||
When it comes to building the image, please make sure that you basically set the `--format docker` in the podman
|
|
||||||
build command, otherwise it won't have HealthChecks and other issues might arise.
|
|
||||||
|
|
||||||
Tip: Put the following in `~/.config/containers/containers.conf`
|
|
||||||
```toml
|
|
||||||
[engine]
|
|
||||||
image_default_format = "docker"
|
|
||||||
```
|
|
||||||
|
|
||||||
For just running the docker-compose.yml with docker image format:
|
|
||||||
```bash
|
|
||||||
BUILDAH_FORMAT=docker podman compose up --build
|
|
||||||
```
|
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
114
TO-DO.md
114
TO-DO.md
@@ -1,114 +0,0 @@
|
|||||||
# WarpBox Security TO-DO
|
|
||||||
|
|
||||||
## 1) High Priority (Do Next)
|
|
||||||
|
|
||||||
- [ ] Persist IP bans across restarts
|
|
||||||
- Current: bans stored in-memory (`lib/security/guard.go`)
|
|
||||||
- Target: durable store in `DBDir` (similar style to `activity`/`alerts`)
|
|
||||||
- Include: startup load, expiry cleanup, atomic writes, corruption-safe fallback
|
|
||||||
|
|
||||||
- [ ] Add trusted proxy CIDR config
|
|
||||||
- Current: forwarded headers trusted only when remote hop is private/local (`lib/server/ip.go`)
|
|
||||||
- Risk: heuristic-only trust model
|
|
||||||
- Target:
|
|
||||||
- `WARPBOX_TRUSTED_PROXY_CIDRS` setting
|
|
||||||
- trust `X-Forwarded-For` only when `RemoteAddr` in trusted CIDR
|
|
||||||
- fallback to direct remote IP otherwise
|
|
||||||
|
|
||||||
- [ ] Add CIDR/range support for whitelists
|
|
||||||
- Current: exact IP match only (`WARPBOX_SECURITY_IP_WHITELIST`, `WARPBOX_SECURITY_ADMIN_IP_WHITELIST`)
|
|
||||||
- Target: support exact IP + CIDR entries
|
|
||||||
- Include strict parser + validation errors in settings save
|
|
||||||
|
|
||||||
- [ ] Add unban / ban edit API audit trail hardening
|
|
||||||
- Ensure all manual ban/unban/ban-until actions always write:
|
|
||||||
- activity event
|
|
||||||
- alert (or policy-based selective alerting)
|
|
||||||
- Add tests for these paths
|
|
||||||
|
|
||||||
## 2) Medium Priority
|
|
||||||
|
|
||||||
- [ ] GeoIP integration for security detail pane
|
|
||||||
- Current: placeholder fields in `/admin/security`
|
|
||||||
- Target: wire geoipfast provider for country/region/ASN fields
|
|
||||||
- Add caching + timeout/failure-safe behavior
|
|
||||||
|
|
||||||
- [ ] Expand malicious path detection rules
|
|
||||||
- Current: simple substring checks in `handleNoRoute`
|
|
||||||
- Target:
|
|
||||||
- rule list/pattern config
|
|
||||||
- normalize URL + decode checks
|
|
||||||
- classify severity by signature group
|
|
||||||
|
|
||||||
- [ ] Add global abuse score per IP
|
|
||||||
- Combine signals:
|
|
||||||
- failed admin auth
|
|
||||||
- malicious path scans
|
|
||||||
- upload abuse
|
|
||||||
- Use score to escalate ban duration automatically
|
|
||||||
|
|
||||||
- [ ] Ban duration policy ladder
|
|
||||||
- Current: fixed `WARPBOX_SECURITY_BAN_SECONDS`
|
|
||||||
- Target:
|
|
||||||
- progressive durations (e.g., 30m, 2h, 24h)
|
|
||||||
- reset after quiet period
|
|
||||||
|
|
||||||
- [ ] Add security settings validation UX
|
|
||||||
- Ensure invalid values (negative, malformed lists, invalid CIDR) rejected with clear UI errors
|
|
||||||
- Add server tests for malformed security override payloads
|
|
||||||
|
|
||||||
## 3) Admin UX Follow-Ups
|
|
||||||
|
|
||||||
- [ ] Add dedicated “Active Bans” page-level controls
|
|
||||||
- bulk unban
|
|
||||||
- filter/sort by expiry and IP
|
|
||||||
- copy IP and quick search in activity/alerts
|
|
||||||
|
|
||||||
- [ ] Add “why banned” detail
|
|
||||||
- link ban entry to latest triggering events and alerts
|
|
||||||
- show counts in active windows (login/scan/upload)
|
|
||||||
|
|
||||||
- [ ] Add optional confirmation modal for destructive security actions
|
|
||||||
- unban all / bulk unban / long custom bans
|
|
||||||
|
|
||||||
## 4) Testing & QA
|
|
||||||
|
|
||||||
- [ ] Add unit tests for `lib/security/guard.go`
|
|
||||||
- `Ban`, `BanUntil`, `Unban`, `BanList` expiry pruning
|
|
||||||
- login/scan threshold behavior
|
|
||||||
- upload rate limiting behavior
|
|
||||||
|
|
||||||
- [ ] Add tests for real-IP resolution edge cases (`lib/server/ip.go`)
|
|
||||||
- direct client
|
|
||||||
- trusted proxy chain
|
|
||||||
- spoofed forwarding headers from untrusted remote
|
|
||||||
|
|
||||||
- [ ] Add integration tests for security endpoints
|
|
||||||
- `/admin/security/actions` ban/ban_until/unban
|
|
||||||
- `/admin/alerts/actions`
|
|
||||||
- admin login brute-force auto-ban flow
|
|
||||||
|
|
||||||
- [ ] Add concurrency/race test pass in CI
|
|
||||||
- run `go test ./... -race` in workflow (where Go toolchain available)
|
|
||||||
|
|
||||||
## 5) Operational / Deployment
|
|
||||||
|
|
||||||
- [ ] Document reverse-proxy setup requirements
|
|
||||||
- Caddy / ingress config examples for forwarding headers
|
|
||||||
- guidance for trusted proxy CIDRs
|
|
||||||
|
|
||||||
- [ ] Add security runbook
|
|
||||||
- how to investigate alerts
|
|
||||||
- how to ban/unban safely
|
|
||||||
- how to tune thresholds for low/high traffic environments
|
|
||||||
|
|
||||||
- [ ] Add metrics hooks (future)
|
|
||||||
- counts: blocked requests, bans issued, unbans, alert volume
|
|
||||||
- expose to Prometheus-compatible endpoint later
|
|
||||||
|
|
||||||
## 6) Nice-to-Have (Later)
|
|
||||||
|
|
||||||
- [ ] Optional external enforcement bridge (fail2ban-compatible log format)
|
|
||||||
- [ ] Webhook notifications for high-severity security alerts
|
|
||||||
- [ ] Per-account/API-key limits once account system matures
|
|
||||||
|
|
||||||
2
TRADEMARK.md
Normal file
2
TRADEMARK.md
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
The name "WarpBox" and associated branding are not licensed under the Apache License 2.0.
|
||||||
|
You may not use them without permission.
|
||||||
@@ -175,6 +175,22 @@ func buildExtraEnvRows(includeHidden bool) []envRow {
|
|||||||
{EnvName: "WARPBOX_ALLOW_ADMIN_SETTINGS_OVERRIDE", Key: "allow_admin_override", Label: "Allow admin UI to override settings", Type: config.SettingTypeBool, Editable: false, HardLimit: true, Default: "true"},
|
{EnvName: "WARPBOX_ALLOW_ADMIN_SETTINGS_OVERRIDE", Key: "allow_admin_override", Label: "Allow admin UI to override settings", Type: config.SettingTypeBool, Editable: false, HardLimit: true, Default: "true"},
|
||||||
}
|
}
|
||||||
|
|
||||||
|
sizePairs := []struct {
|
||||||
|
bytesEnv string
|
||||||
|
mbEnv string
|
||||||
|
label string
|
||||||
|
}{
|
||||||
|
{"WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES", "WARPBOX_GLOBAL_MAX_FILE_SIZE_MB", "Global max file size"},
|
||||||
|
{"WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES", "WARPBOX_GLOBAL_MAX_BOX_SIZE_MB", "Global max box size"},
|
||||||
|
{"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES", "WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_MB", "Default user max file size"},
|
||||||
|
{"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES", "WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_MB", "Default user max box size"},
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, pair := range sizePairs {
|
||||||
|
extra = append(extra, envRow{EnvName: pair.bytesEnv, Key: pair.bytesEnv, Label: pair.label + " (bytes)", Type: config.SettingTypeInt64, Editable: false, HardLimit: true, Minimum: 0, Default: "(use bytes or MB variant)"})
|
||||||
|
extra = append(extra, envRow{EnvName: pair.mbEnv, Key: pair.mbEnv, Label: pair.label + " (MB)", Type: config.SettingTypeInt64, Editable: false, HardLimit: true, Minimum: 0, Default: "(use bytes or MB variant)"})
|
||||||
|
}
|
||||||
|
|
||||||
return extra
|
return extra
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -5,8 +5,6 @@ services:
|
|||||||
ports:
|
ports:
|
||||||
- "8080:8080"
|
- "8080:8080"
|
||||||
volumes:
|
volumes:
|
||||||
# For podman please use :Z
|
|
||||||
# - ./data:/app/data:Z
|
|
||||||
- ./data:/app/data
|
- ./data:/app/data
|
||||||
env_file:
|
env_file:
|
||||||
- .env
|
- .env
|
||||||
|
|||||||
@@ -1,19 +0,0 @@
|
|||||||
# GeoIP Guide (Planning)
|
|
||||||
|
|
||||||
This project intentionally does not enable GeoIP enforcement yet.
|
|
||||||
|
|
||||||
Planned integration target: `github.com/rabuchaim/geoip2fast`.
|
|
||||||
|
|
||||||
## Recommended approach
|
|
||||||
|
|
||||||
1. Load one shared GeoIP provider instance at startup.
|
|
||||||
2. Add a small in-memory cache keyed by IP with TTL.
|
|
||||||
3. Apply lookup timeout and fallback to `unknown` values on failures.
|
|
||||||
4. Use results first in the admin security detail pane.
|
|
||||||
5. Add aggregated statistics only after detail pane behavior is stable.
|
|
||||||
|
|
||||||
## Why this is safe
|
|
||||||
|
|
||||||
- No request path should fail because GeoIP lookup fails.
|
|
||||||
- Lookup cost stays bounded with caching.
|
|
||||||
- Security decisions remain independent from GeoIP quality.
|
|
||||||
@@ -1,40 +0,0 @@
|
|||||||
# Security Runbook
|
|
||||||
|
|
||||||
## Trusted Proxy Setup (Caddy)
|
|
||||||
|
|
||||||
Set `WARPBOX_TRUSTED_PROXY_CIDRS` to only the CIDRs of your reverse proxies/load balancers.
|
|
||||||
|
|
||||||
Example:
|
|
||||||
|
|
||||||
```bash
|
|
||||||
WARPBOX_TRUSTED_PROXY_CIDRS=10.0.0.0/8,192.168.0.0/16
|
|
||||||
```
|
|
||||||
|
|
||||||
Caddy example:
|
|
||||||
|
|
||||||
```caddyfile
|
|
||||||
:443 {
|
|
||||||
reverse_proxy 127.0.0.1:8080 {
|
|
||||||
header_up X-Forwarded-For {http.request.remote.host}
|
|
||||||
header_up X-Real-IP {http.request.remote.host}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
```
|
|
||||||
|
|
||||||
WarpBox will trust `X-Forwarded-For` only if the direct remote IP is inside `WARPBOX_TRUSTED_PROXY_CIDRS`.
|
|
||||||
|
|
||||||
## IP Ban Operations
|
|
||||||
|
|
||||||
- Use temporary bans by default.
|
|
||||||
- Use `ban_until` only for active incidents requiring explicit windows.
|
|
||||||
- Before unbanning, inspect related activity and alerts for repeated abuse patterns.
|
|
||||||
- For destructive actions (`bulk_unban`, `unban_all`), require explicit confirmation.
|
|
||||||
|
|
||||||
## Tuning Guidance
|
|
||||||
|
|
||||||
- Low traffic deployments: reduce max-attempt thresholds to catch abuse faster.
|
|
||||||
- High traffic deployments: increase windows and max-attempts incrementally to reduce false positives.
|
|
||||||
- Watch for:
|
|
||||||
- repeated `auth.admin.failed`
|
|
||||||
- repeated `security.scan`
|
|
||||||
- frequent `security.upload_limit`
|
|
||||||
10
docs/tech.md
10
docs/tech.md
@@ -150,16 +150,16 @@ Primary environment variables:
|
|||||||
- `WARPBOX_RENEW_ON_DOWNLOAD_ENABLED`
|
- `WARPBOX_RENEW_ON_DOWNLOAD_ENABLED`
|
||||||
- `WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS`
|
- `WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS`
|
||||||
- `WARPBOX_MAX_GUEST_EXPIRY_SECONDS`
|
- `WARPBOX_MAX_GUEST_EXPIRY_SECONDS`
|
||||||
- `WARPBOX_GLOBAL_MAX_FILE_SIZE_GB`
|
- `WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES`
|
||||||
- `WARPBOX_GLOBAL_MAX_BOX_SIZE_GB`
|
- `WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES`
|
||||||
- `WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_GB`
|
- `WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES`
|
||||||
- `WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_GB`
|
- `WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES`
|
||||||
- `WARPBOX_SESSION_TTL_SECONDS`
|
- `WARPBOX_SESSION_TTL_SECONDS`
|
||||||
- `WARPBOX_BOX_POLL_INTERVAL_MS`
|
- `WARPBOX_BOX_POLL_INTERVAL_MS`
|
||||||
- `WARPBOX_THUMBNAIL_BATCH_SIZE`
|
- `WARPBOX_THUMBNAIL_BATCH_SIZE`
|
||||||
- `WARPBOX_THUMBNAIL_INTERVAL_SECONDS`
|
- `WARPBOX_THUMBNAIL_INTERVAL_SECONDS`
|
||||||
|
|
||||||
Size limit settings use `_GB` env names with `1024^3` conversion. Legacy `_MB` and `_BYTES` names remain accepted for compatibility. `WARPBOX_ADMIN_ENABLED`
|
Size limit settings accept `_MB` or `_BYTES` env names. `WARPBOX_ADMIN_ENABLED`
|
||||||
accepts `auto`, `true`, or `false`.
|
accepts `auto`, `true`, or `false`.
|
||||||
|
|
||||||
The HTTP listen address is configured through the CLI flag:
|
The HTTP listen address is configured through the CLI flag:
|
||||||
|
|||||||
14
go.mod
14
go.mod
@@ -3,12 +3,11 @@ module warpbox
|
|||||||
go 1.23.0
|
go 1.23.0
|
||||||
|
|
||||||
require (
|
require (
|
||||||
github.com/dgraph-io/badger/v4 v4.9.1
|
github.com/dgraph-io/badger/v4 v4.8.0
|
||||||
github.com/gin-contrib/gzip v1.0.1
|
github.com/gin-contrib/gzip v1.0.1
|
||||||
github.com/gin-gonic/gin v1.10.0
|
github.com/gin-gonic/gin v1.10.0
|
||||||
github.com/spf13/cobra v1.9.1
|
github.com/spf13/cobra v1.9.1
|
||||||
github.com/spf13/pflag v1.0.6
|
golang.org/x/crypto v0.39.0
|
||||||
golang.org/x/crypto v0.41.0
|
|
||||||
)
|
)
|
||||||
|
|
||||||
require (
|
require (
|
||||||
@@ -37,6 +36,7 @@ require (
|
|||||||
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd // indirect
|
||||||
github.com/modern-go/reflect2 v1.0.2 // indirect
|
github.com/modern-go/reflect2 v1.0.2 // indirect
|
||||||
github.com/pelletier/go-toml/v2 v2.2.2 // indirect
|
github.com/pelletier/go-toml/v2 v2.2.2 // indirect
|
||||||
|
github.com/spf13/pflag v1.0.6 // indirect
|
||||||
github.com/stretchr/testify v1.11.1 // indirect
|
github.com/stretchr/testify v1.11.1 // indirect
|
||||||
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
github.com/twitchyliquid64/golang-asm v0.15.1 // indirect
|
||||||
github.com/ugorji/go/codec v1.2.12 // indirect
|
github.com/ugorji/go/codec v1.2.12 // indirect
|
||||||
@@ -45,9 +45,9 @@ require (
|
|||||||
go.opentelemetry.io/otel/metric v1.37.0 // indirect
|
go.opentelemetry.io/otel/metric v1.37.0 // indirect
|
||||||
go.opentelemetry.io/otel/trace v1.37.0 // indirect
|
go.opentelemetry.io/otel/trace v1.37.0 // indirect
|
||||||
golang.org/x/arch v0.8.0 // indirect
|
golang.org/x/arch v0.8.0 // indirect
|
||||||
golang.org/x/net v0.43.0 // indirect
|
golang.org/x/net v0.41.0 // indirect
|
||||||
golang.org/x/sys v0.35.0 // indirect
|
golang.org/x/sys v0.34.0 // indirect
|
||||||
golang.org/x/text v0.28.0 // indirect
|
golang.org/x/text v0.26.0 // indirect
|
||||||
google.golang.org/protobuf v1.36.7 // indirect
|
google.golang.org/protobuf v1.36.6 // indirect
|
||||||
gopkg.in/yaml.v3 v3.0.1 // indirect
|
gopkg.in/yaml.v3 v3.0.1 // indirect
|
||||||
)
|
)
|
||||||
|
|||||||
24
go.sum
24
go.sum
@@ -12,8 +12,8 @@ github.com/cpuguy83/go-md2man/v2 v2.0.6/go.mod h1:oOW0eioCTA6cOiMLiUPZOpcVxMig6N
|
|||||||
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c=
|
||||||
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38=
|
||||||
github.com/dgraph-io/badger/v4 v4.9.1 h1:DocZXZkg5JJHJPtUErA0ibyHxOVUDVoXLSCV6t8NC8w=
|
github.com/dgraph-io/badger/v4 v4.8.0 h1:JYph1ChBijCw8SLeybvPINizbDKWZ5n/GYbz2yhN/bs=
|
||||||
github.com/dgraph-io/badger/v4 v4.9.1/go.mod h1:5/MEx97uzdPUHR4KtkNt8asfI2T4JiEiQlV7kWUo8c0=
|
github.com/dgraph-io/badger/v4 v4.8.0/go.mod h1:U6on6e8k/RTbUWxqKR0MvugJuVmkxSNc79ap4917h4w=
|
||||||
github.com/dgraph-io/ristretto/v2 v2.2.0 h1:bkY3XzJcXoMuELV8F+vS8kzNgicwQFAaGINAEJdWGOM=
|
github.com/dgraph-io/ristretto/v2 v2.2.0 h1:bkY3XzJcXoMuELV8F+vS8kzNgicwQFAaGINAEJdWGOM=
|
||||||
github.com/dgraph-io/ristretto/v2 v2.2.0/go.mod h1:RZrm63UmcBAaYWC1DotLYBmTvgkrs0+XhBd7Npn7/zI=
|
github.com/dgraph-io/ristretto/v2 v2.2.0/go.mod h1:RZrm63UmcBAaYWC1DotLYBmTvgkrs0+XhBd7Npn7/zI=
|
||||||
github.com/dgryski/go-farm v0.0.0-20240924180020-3414d57e47da h1:aIftn67I1fkbMa512G+w+Pxci9hJPB8oMnkcP3iZF38=
|
github.com/dgryski/go-farm v0.0.0-20240924180020-3414d57e47da h1:aIftn67I1fkbMa512G+w+Pxci9hJPB8oMnkcP3iZF38=
|
||||||
@@ -110,18 +110,18 @@ go.opentelemetry.io/otel/trace v1.37.0/go.mod h1:TlgrlQ+PtQO5XFerSPUYG0JSgGyryXe
|
|||||||
golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
golang.org/x/arch v0.0.0-20210923205945-b76863e36670/go.mod h1:5om86z9Hs0C8fWVUuoMHwpExlXzs5Tkyp9hOrfG7pp8=
|
||||||
golang.org/x/arch v0.8.0 h1:3wRIsP3pM4yUptoR96otTUOXI367OS0+c9eeRi9doIc=
|
golang.org/x/arch v0.8.0 h1:3wRIsP3pM4yUptoR96otTUOXI367OS0+c9eeRi9doIc=
|
||||||
golang.org/x/arch v0.8.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys=
|
golang.org/x/arch v0.8.0/go.mod h1:FEVrYAQjsQXMVJ1nsMoVVXPZg6p2JE2mx8psSWTDQys=
|
||||||
golang.org/x/crypto v0.41.0 h1:WKYxWedPGCTVVl5+WHSSrOBT0O8lx32+zxmHxijgXp4=
|
golang.org/x/crypto v0.39.0 h1:SHs+kF4LP+f+p14esP5jAoDpHU8Gu/v9lFRK6IT5imM=
|
||||||
golang.org/x/crypto v0.41.0/go.mod h1:pO5AFd7FA68rFak7rOAGVuygIISepHftHnr8dr6+sUc=
|
golang.org/x/crypto v0.39.0/go.mod h1:L+Xg3Wf6HoL4Bn4238Z6ft6KfEpN0tJGo53AAPC632U=
|
||||||
golang.org/x/net v0.43.0 h1:lat02VYK2j4aLzMzecihNvTlJNQUq316m2Mr9rnM6YE=
|
golang.org/x/net v0.41.0 h1:vBTly1HeNPEn3wtREYfy4GZ/NECgw2Cnl+nK6Nz3uvw=
|
||||||
golang.org/x/net v0.43.0/go.mod h1:vhO1fvI4dGsIjh73sWfUVjj3N7CA9WkKJNQm2svM6Jg=
|
golang.org/x/net v0.41.0/go.mod h1:B/K4NNqkfmg07DQYrbwvSluqCJOOXwUjeb/5lOisjbA=
|
||||||
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.5.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg=
|
||||||
golang.org/x/sys v0.35.0 h1:vz1N37gP5bs89s7He8XuIYXpyY0+QlsKmzipCbUtyxI=
|
golang.org/x/sys v0.34.0 h1:H5Y5sJ2L2JRdyv7ROF1he/lPdvFsd0mJHFw2ThKHxLA=
|
||||||
golang.org/x/sys v0.35.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
|
golang.org/x/sys v0.34.0/go.mod h1:BJP2sWEmIv4KK5OTEluFJCKSidICx8ciO85XgH3Ak8k=
|
||||||
golang.org/x/text v0.28.0 h1:rhazDwis8INMIwQ4tpjLDzUhx6RlXqZNPEM0huQojng=
|
golang.org/x/text v0.26.0 h1:P42AVeLghgTYr4+xUnTRKDMqpar+PtX7KWuNQL21L8M=
|
||||||
golang.org/x/text v0.28.0/go.mod h1:U8nCwOR8jO/marOQ0QbDiOngZVEBB7MAiitBuMjXiNU=
|
golang.org/x/text v0.26.0/go.mod h1:QK15LZJUUQVJxhz7wXgxSy/CJaTFjd0G+YLonydOVQA=
|
||||||
google.golang.org/protobuf v1.36.7 h1:IgrO7UwFQGJdRNXH/sQux4R1Dj1WAKcLElzeeRaXV2A=
|
google.golang.org/protobuf v1.36.6 h1:z1NpPI8ku2WgiWnf+t9wTPsn6eP1L7ksHUlkfLvd9xY=
|
||||||
google.golang.org/protobuf v1.36.7/go.mod h1:jduwjTPXsFjZGTmRluh+L6NjiWu7pchiJ2/5YcXBHnY=
|
google.golang.org/protobuf v1.36.6/go.mod h1:jduwjTPXsFjZGTmRluh+L6NjiWu7pchiJ2/5YcXBHnY=
|
||||||
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0=
|
||||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
|
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk=
|
||||||
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
|
gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q=
|
||||||
|
|||||||
@@ -1,116 +0,0 @@
|
|||||||
package activity
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"sort"
|
|
||||||
"sync"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Event struct {
|
|
||||||
ID string `json:"id"`
|
|
||||||
Kind string `json:"kind"`
|
|
||||||
Severity string `json:"severity"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
Actor string `json:"actor"`
|
|
||||||
IP string `json:"ip"`
|
|
||||||
Path string `json:"path"`
|
|
||||||
Method string `json:"method"`
|
|
||||||
CreatedAt time.Time `json:"created_at"`
|
|
||||||
Meta map[string]string `json:"meta,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Store struct {
|
|
||||||
path string
|
|
||||||
mu sync.Mutex
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewStore(path string) *Store {
|
|
||||||
return &Store{path: path}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) Append(event Event, retentionSeconds int64) error {
|
|
||||||
s.mu.Lock()
|
|
||||||
defer s.mu.Unlock()
|
|
||||||
|
|
||||||
events, err := s.readLocked()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if event.CreatedAt.IsZero() {
|
|
||||||
event.CreatedAt = time.Now().UTC()
|
|
||||||
}
|
|
||||||
if event.ID == "" {
|
|
||||||
event.ID = event.CreatedAt.Format("20060102T150405.000000000")
|
|
||||||
}
|
|
||||||
|
|
||||||
events = append(events, event)
|
|
||||||
events = pruneByRetention(events, retentionSeconds)
|
|
||||||
return s.writeLocked(events)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) List(limit int, retentionSeconds int64) ([]Event, error) {
|
|
||||||
s.mu.Lock()
|
|
||||||
defer s.mu.Unlock()
|
|
||||||
|
|
||||||
events, err := s.readLocked()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
events = pruneByRetention(events, retentionSeconds)
|
|
||||||
if err := s.writeLocked(events); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
sort.Slice(events, func(i, j int) bool {
|
|
||||||
return events[i].CreatedAt.After(events[j].CreatedAt)
|
|
||||||
})
|
|
||||||
if limit > 0 && len(events) > limit {
|
|
||||||
return events[:limit], nil
|
|
||||||
}
|
|
||||||
return events, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func pruneByRetention(events []Event, retentionSeconds int64) []Event {
|
|
||||||
if retentionSeconds <= 0 {
|
|
||||||
return events
|
|
||||||
}
|
|
||||||
cutoff := time.Now().UTC().Add(-time.Duration(retentionSeconds) * time.Second)
|
|
||||||
out := make([]Event, 0, len(events))
|
|
||||||
for _, event := range events {
|
|
||||||
if event.CreatedAt.IsZero() || event.CreatedAt.After(cutoff) {
|
|
||||||
out = append(out, event)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) readLocked() ([]Event, error) {
|
|
||||||
data, err := os.ReadFile(s.path)
|
|
||||||
if err != nil {
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
return []Event{}, nil
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if len(data) == 0 {
|
|
||||||
return []Event{}, nil
|
|
||||||
}
|
|
||||||
var events []Event
|
|
||||||
if err := json.Unmarshal(data, &events); err != nil {
|
|
||||||
return []Event{}, nil
|
|
||||||
}
|
|
||||||
return events, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) writeLocked(events []Event) error {
|
|
||||||
if err := os.MkdirAll(filepath.Dir(s.path), 0755); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
data, err := json.MarshalIndent(events, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return os.WriteFile(s.path, data, 0644)
|
|
||||||
}
|
|
||||||
@@ -1,151 +0,0 @@
|
|||||||
package alerts
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"sort"
|
|
||||||
"strconv"
|
|
||||||
"sync"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Status string
|
|
||||||
|
|
||||||
const (
|
|
||||||
StatusOpen Status = "open"
|
|
||||||
StatusAcked Status = "acked"
|
|
||||||
StatusClosed Status = "closed"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Alert struct {
|
|
||||||
ID string `json:"id"`
|
|
||||||
Title string `json:"title"`
|
|
||||||
Severity string `json:"severity"`
|
|
||||||
Status Status `json:"status"`
|
|
||||||
Group string `json:"group"`
|
|
||||||
Code string `json:"code"`
|
|
||||||
Trace string `json:"trace"`
|
|
||||||
Message string `json:"message"`
|
|
||||||
CreatedAt time.Time `json:"created_at"`
|
|
||||||
UpdatedAt time.Time `json:"updated_at"`
|
|
||||||
Meta map[string]string `json:"meta,omitempty"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type Store struct {
|
|
||||||
path string
|
|
||||||
mu sync.Mutex
|
|
||||||
}
|
|
||||||
|
|
||||||
func NewStore(path string) *Store {
|
|
||||||
return &Store{path: path}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) Add(alert Alert) error {
|
|
||||||
s.mu.Lock()
|
|
||||||
defer s.mu.Unlock()
|
|
||||||
|
|
||||||
alertsList, err := s.readLocked()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
now := time.Now().UTC()
|
|
||||||
if alert.ID == "" {
|
|
||||||
alert.ID = strconv.FormatInt(now.UnixNano(), 10)
|
|
||||||
}
|
|
||||||
if alert.Status == "" {
|
|
||||||
alert.Status = StatusOpen
|
|
||||||
}
|
|
||||||
if alert.CreatedAt.IsZero() {
|
|
||||||
alert.CreatedAt = now
|
|
||||||
}
|
|
||||||
alert.UpdatedAt = now
|
|
||||||
alertsList = append(alertsList, alert)
|
|
||||||
return s.writeLocked(alertsList)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) List(limit int) ([]Alert, error) {
|
|
||||||
s.mu.Lock()
|
|
||||||
defer s.mu.Unlock()
|
|
||||||
alertsList, err := s.readLocked()
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
sort.Slice(alertsList, func(i, j int) bool {
|
|
||||||
return alertsList[i].CreatedAt.After(alertsList[j].CreatedAt)
|
|
||||||
})
|
|
||||||
if limit > 0 && len(alertsList) > limit {
|
|
||||||
return alertsList[:limit], nil
|
|
||||||
}
|
|
||||||
return alertsList, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) SetStatus(ids []string, status Status) error {
|
|
||||||
s.mu.Lock()
|
|
||||||
defer s.mu.Unlock()
|
|
||||||
alertsList, err := s.readLocked()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
target := map[string]bool{}
|
|
||||||
for _, id := range ids {
|
|
||||||
target[id] = true
|
|
||||||
}
|
|
||||||
now := time.Now().UTC()
|
|
||||||
for i := range alertsList {
|
|
||||||
if target[alertsList[i].ID] {
|
|
||||||
alertsList[i].Status = status
|
|
||||||
alertsList[i].UpdatedAt = now
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return s.writeLocked(alertsList)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) Delete(ids []string) error {
|
|
||||||
s.mu.Lock()
|
|
||||||
defer s.mu.Unlock()
|
|
||||||
alertsList, err := s.readLocked()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
target := map[string]bool{}
|
|
||||||
for _, id := range ids {
|
|
||||||
target[id] = true
|
|
||||||
}
|
|
||||||
kept := make([]Alert, 0, len(alertsList))
|
|
||||||
for _, alert := range alertsList {
|
|
||||||
if !target[alert.ID] {
|
|
||||||
kept = append(kept, alert)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return s.writeLocked(kept)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) readLocked() ([]Alert, error) {
|
|
||||||
data, err := os.ReadFile(s.path)
|
|
||||||
if err != nil {
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
return []Alert{}, nil
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if len(data) == 0 {
|
|
||||||
return []Alert{}, nil
|
|
||||||
}
|
|
||||||
var alertsList []Alert
|
|
||||||
if err := json.Unmarshal(data, &alertsList); err != nil {
|
|
||||||
return []Alert{}, nil
|
|
||||||
}
|
|
||||||
return alertsList, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (s *Store) writeLocked(alertsList []Alert) error {
|
|
||||||
if err := os.MkdirAll(filepath.Dir(s.path), 0755); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
data, err := json.MarshalIndent(alertsList, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return os.WriteFile(s.path, data, 0644)
|
|
||||||
}
|
|
||||||
@@ -1,60 +0,0 @@
|
|||||||
package boxstore
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"os"
|
|
||||||
)
|
|
||||||
|
|
||||||
type CleanupExpiredResult struct {
|
|
||||||
Scanned int
|
|
||||||
Deleted int
|
|
||||||
Skipped int
|
|
||||||
DeletedIDs []string
|
|
||||||
Warnings []string
|
|
||||||
}
|
|
||||||
|
|
||||||
func CleanupExpiredBoxes() (CleanupExpiredResult, error) {
|
|
||||||
entries, err := os.ReadDir(uploadRoot)
|
|
||||||
if err != nil {
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
return CleanupExpiredResult{}, nil
|
|
||||||
}
|
|
||||||
return CleanupExpiredResult{}, err
|
|
||||||
}
|
|
||||||
|
|
||||||
result := CleanupExpiredResult{
|
|
||||||
DeletedIDs: make([]string, 0),
|
|
||||||
Warnings: make([]string, 0),
|
|
||||||
}
|
|
||||||
for _, entry := range entries {
|
|
||||||
if !entry.IsDir() {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
boxID := entry.Name()
|
|
||||||
if !ValidBoxID(boxID) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
result.Scanned++
|
|
||||||
|
|
||||||
manifest, err := ReadManifest(boxID)
|
|
||||||
if err != nil {
|
|
||||||
result.Skipped++
|
|
||||||
if !os.IsNotExist(err) {
|
|
||||||
result.Warnings = append(result.Warnings, fmt.Sprintf("%s: %v", boxID, err))
|
|
||||||
}
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if !IsExpired(manifest) {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if err := DeleteBox(boxID); err != nil {
|
|
||||||
result.Skipped++
|
|
||||||
result.Warnings = append(result.Warnings, fmt.Sprintf("%s: %v", boxID, err))
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
result.Deleted++
|
|
||||||
result.DeletedIDs = append(result.DeletedIDs, boxID)
|
|
||||||
}
|
|
||||||
|
|
||||||
return result, nil
|
|
||||||
}
|
|
||||||
@@ -1,58 +0,0 @@
|
|||||||
package boxstore
|
|
||||||
|
|
||||||
import (
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"testing"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"warpbox/lib/models"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestCleanupExpiredBoxesDeletesOnlyExpiredManifestBoxes(t *testing.T) {
|
|
||||||
root := filepath.Join(t.TempDir(), "uploads")
|
|
||||||
previousRoot := UploadRoot()
|
|
||||||
t.Cleanup(func() { SetUploadRoot(previousRoot) })
|
|
||||||
SetUploadRoot(root)
|
|
||||||
|
|
||||||
expiredID := "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa"
|
|
||||||
activeID := "bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb"
|
|
||||||
legacyID := "cccccccccccccccccccccccccccccccc"
|
|
||||||
|
|
||||||
if err := os.MkdirAll(BoxPath(expiredID), 0755); err != nil {
|
|
||||||
t.Fatalf("mkdir expired: %v", err)
|
|
||||||
}
|
|
||||||
if err := os.MkdirAll(BoxPath(activeID), 0755); err != nil {
|
|
||||||
t.Fatalf("mkdir active: %v", err)
|
|
||||||
}
|
|
||||||
if err := os.MkdirAll(BoxPath(legacyID), 0755); err != nil {
|
|
||||||
t.Fatalf("mkdir legacy: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := WriteManifest(expiredID, models.BoxManifest{CreatedAt: time.Now().UTC().Add(-2 * time.Hour), ExpiresAt: time.Now().UTC().Add(-time.Minute)}); err != nil {
|
|
||||||
t.Fatalf("write expired manifest: %v", err)
|
|
||||||
}
|
|
||||||
if err := WriteManifest(activeID, models.BoxManifest{CreatedAt: time.Now().UTC(), ExpiresAt: time.Now().UTC().Add(time.Hour)}); err != nil {
|
|
||||||
t.Fatalf("write active manifest: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
result, err := CleanupExpiredBoxes()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("cleanup failed: %v", err)
|
|
||||||
}
|
|
||||||
if result.Deleted != 1 {
|
|
||||||
t.Fatalf("expected 1 deleted box, got %d", result.Deleted)
|
|
||||||
}
|
|
||||||
if len(result.DeletedIDs) != 1 || result.DeletedIDs[0] != expiredID {
|
|
||||||
t.Fatalf("expected deleted id %s, got %#v", expiredID, result.DeletedIDs)
|
|
||||||
}
|
|
||||||
if _, err := os.Stat(BoxPath(expiredID)); !os.IsNotExist(err) {
|
|
||||||
t.Fatalf("expected expired box dir removed, stat err=%v", err)
|
|
||||||
}
|
|
||||||
if _, err := os.Stat(BoxPath(activeID)); err != nil {
|
|
||||||
t.Fatalf("expected active box to remain, stat err=%v", err)
|
|
||||||
}
|
|
||||||
if _, err := os.Stat(BoxPath(legacyID)); err != nil {
|
|
||||||
t.Fatalf("expected legacy box to remain, stat err=%v", err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -153,43 +153,6 @@ func RenewManifest(boxID string, seconds int64) (models.BoxManifest, error) {
|
|||||||
manifest.ExpiresAt = time.Now().UTC().Add(time.Duration(seconds) * time.Second)
|
manifest.ExpiresAt = time.Now().UTC().Add(time.Duration(seconds) * time.Second)
|
||||||
return manifest, writeManifestUnlocked(boxID, manifest)
|
return manifest, writeManifestUnlocked(boxID, manifest)
|
||||||
}
|
}
|
||||||
|
|
||||||
func ExpireBox(boxID string) (models.BoxManifest, error) {
|
|
||||||
manifestMu.Lock()
|
|
||||||
defer manifestMu.Unlock()
|
|
||||||
|
|
||||||
manifest, err := readManifestUnlocked(boxID)
|
|
||||||
if err != nil {
|
|
||||||
return manifest, err
|
|
||||||
}
|
|
||||||
manifest.ExpiresAt = time.Now().UTC().Add(-time.Second)
|
|
||||||
return manifest, writeManifestUnlocked(boxID, manifest)
|
|
||||||
}
|
|
||||||
|
|
||||||
func BumpBoxExpiry(boxID string, delta time.Duration) (models.BoxManifest, error) {
|
|
||||||
manifestMu.Lock()
|
|
||||||
defer manifestMu.Unlock()
|
|
||||||
|
|
||||||
manifest, err := readManifestUnlocked(boxID)
|
|
||||||
if err != nil {
|
|
||||||
return manifest, err
|
|
||||||
}
|
|
||||||
if delta <= 0 {
|
|
||||||
return manifest, fmt.Errorf("Invalid bump duration")
|
|
||||||
}
|
|
||||||
if manifest.OneTimeDownload {
|
|
||||||
return manifest, fmt.Errorf("One-time boxes cannot be extended")
|
|
||||||
}
|
|
||||||
|
|
||||||
base := manifest.ExpiresAt
|
|
||||||
now := time.Now().UTC()
|
|
||||||
if base.IsZero() || base.Before(now) {
|
|
||||||
base = now
|
|
||||||
}
|
|
||||||
manifest.ExpiresAt = base.Add(delta)
|
|
||||||
return manifest, writeManifestUnlocked(boxID, manifest)
|
|
||||||
}
|
|
||||||
|
|
||||||
func reconcileManifest(boxID string) (models.BoxManifest, error) {
|
func reconcileManifest(boxID string) (models.BoxManifest, error) {
|
||||||
manifestMu.Lock()
|
manifestMu.Lock()
|
||||||
defer manifestMu.Unlock()
|
defer manifestMu.Unlock()
|
||||||
|
|||||||
@@ -204,57 +204,3 @@ func TestBoxPasswordUsesBcryptAndVerifiesLegacy(t *testing.T) {
|
|||||||
t.Fatal("expected legacy password hash to verify")
|
t.Fatal("expected legacy password hash to verify")
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestExpireBoxMarksManifestExpired(t *testing.T) {
|
|
||||||
restoreUploadRoot := UploadRoot()
|
|
||||||
defer SetUploadRoot(restoreUploadRoot)
|
|
||||||
SetUploadRoot(t.TempDir())
|
|
||||||
|
|
||||||
boxID := "0123456789abcdef0123456789abcdef"
|
|
||||||
if err := os.MkdirAll(BoxPath(boxID), 0755); err != nil {
|
|
||||||
t.Fatalf("MkdirAll returned error: %v", err)
|
|
||||||
}
|
|
||||||
manifest := models.BoxManifest{
|
|
||||||
CreatedAt: time.Now().UTC().Add(-time.Hour),
|
|
||||||
ExpiresAt: time.Now().UTC().Add(time.Hour),
|
|
||||||
}
|
|
||||||
if err := WriteManifest(boxID, manifest); err != nil {
|
|
||||||
t.Fatalf("WriteManifest returned error: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
expired, err := ExpireBox(boxID)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("ExpireBox returned error: %v", err)
|
|
||||||
}
|
|
||||||
if !expired.ExpiresAt.Before(time.Now().UTC()) {
|
|
||||||
t.Fatalf("expected expired manifest time in past, got %s", expired.ExpiresAt)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestBumpBoxExpiryExtendsFutureExpiry(t *testing.T) {
|
|
||||||
restoreUploadRoot := UploadRoot()
|
|
||||||
defer SetUploadRoot(restoreUploadRoot)
|
|
||||||
SetUploadRoot(t.TempDir())
|
|
||||||
|
|
||||||
boxID := "fedcba9876543210fedcba9876543210"
|
|
||||||
if err := os.MkdirAll(BoxPath(boxID), 0755); err != nil {
|
|
||||||
t.Fatalf("MkdirAll returned error: %v", err)
|
|
||||||
}
|
|
||||||
base := time.Now().UTC().Add(time.Hour).Truncate(time.Second)
|
|
||||||
manifest := models.BoxManifest{
|
|
||||||
CreatedAt: time.Now().UTC().Add(-time.Hour),
|
|
||||||
ExpiresAt: base,
|
|
||||||
}
|
|
||||||
if err := WriteManifest(boxID, manifest); err != nil {
|
|
||||||
t.Fatalf("WriteManifest returned error: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
bumped, err := BumpBoxExpiry(boxID, 24*time.Hour)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("BumpBoxExpiry returned error: %v", err)
|
|
||||||
}
|
|
||||||
expected := base.Add(24 * time.Hour)
|
|
||||||
if bumped.ExpiresAt.Before(expected.Add(-time.Second)) || bumped.ExpiresAt.After(expected.Add(time.Second)) {
|
|
||||||
t.Fatalf("expected bumped expiry near %s, got %s", expected, bumped.ExpiresAt)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -22,15 +22,18 @@ func TestDefaults(t *testing.T) {
|
|||||||
if !cfg.GuestUploadsEnabled || !cfg.APIEnabled || !cfg.ZipDownloadsEnabled || !cfg.OneTimeDownloadsEnabled {
|
if !cfg.GuestUploadsEnabled || !cfg.APIEnabled || !cfg.ZipDownloadsEnabled || !cfg.OneTimeDownloadsEnabled {
|
||||||
t.Fatal("expected default guest/API/download toggles to be enabled")
|
t.Fatal("expected default guest/API/download toggles to be enabled")
|
||||||
}
|
}
|
||||||
if !cfg.SecurityEnabled {
|
|
||||||
t.Fatal("expected security features to be enabled by default")
|
|
||||||
}
|
|
||||||
if cfg.AdminUsername != "admin" {
|
if cfg.AdminUsername != "admin" {
|
||||||
t.Fatalf("unexpected admin username: %s", cfg.AdminUsername)
|
t.Fatalf("unexpected admin username: %s", cfg.AdminUsername)
|
||||||
}
|
}
|
||||||
if cfg.AdminPassword != "" {
|
if cfg.AdminPassword != "" {
|
||||||
t.Fatal("expected default admin password to be empty")
|
t.Fatal("expected default admin password to be empty")
|
||||||
}
|
}
|
||||||
|
if !cfg.BoxOwnerEditEnabled || !cfg.BoxOwnerRefreshEnabled || !cfg.BoxOwnerPasswordEditEnabled {
|
||||||
|
t.Fatal("expected box owner policy defaults to be enabled")
|
||||||
|
}
|
||||||
|
if cfg.BoxOwnerMaxRefreshCount != 3 || cfg.BoxOwnerMaxRefreshAmountSeconds != 86400 || cfg.BoxOwnerMaxTotalExpirySeconds != 604800 {
|
||||||
|
t.Fatalf("unexpected box owner policy defaults: %#v", cfg)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestEnvironmentOverrides(t *testing.T) {
|
func TestEnvironmentOverrides(t *testing.T) {
|
||||||
@@ -38,11 +41,12 @@ func TestEnvironmentOverrides(t *testing.T) {
|
|||||||
t.Setenv("WARPBOX_DATA_DIR", "/tmp/warpbox-test")
|
t.Setenv("WARPBOX_DATA_DIR", "/tmp/warpbox-test")
|
||||||
t.Setenv("WARPBOX_GUEST_UPLOADS_ENABLED", "false")
|
t.Setenv("WARPBOX_GUEST_UPLOADS_ENABLED", "false")
|
||||||
t.Setenv("WARPBOX_API_ENABLED", "false")
|
t.Setenv("WARPBOX_API_ENABLED", "false")
|
||||||
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_GB", "0.5")
|
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES", "100")
|
||||||
t.Setenv("WARPBOX_BOX_POLL_INTERVAL_MS", "2000")
|
t.Setenv("WARPBOX_BOX_POLL_INTERVAL_MS", "2000")
|
||||||
t.Setenv("WARPBOX_ADMIN_USERNAME", "root")
|
t.Setenv("WARPBOX_ADMIN_USERNAME", "root")
|
||||||
t.Setenv("WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE", "true")
|
t.Setenv("WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE", "true")
|
||||||
t.Setenv("WARPBOX_SECURITY_ENABLED", "false")
|
t.Setenv("WARPBOX_BOX_OWNER_MAX_REFRESH_COUNT", "5")
|
||||||
|
t.Setenv("WARPBOX_BOX_OWNER_PASSWORD_EDIT_ENABLED", "false")
|
||||||
|
|
||||||
cfg, err := Load()
|
cfg, err := Load()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -55,7 +59,7 @@ func TestEnvironmentOverrides(t *testing.T) {
|
|||||||
if cfg.GuestUploadsEnabled || cfg.APIEnabled {
|
if cfg.GuestUploadsEnabled || cfg.APIEnabled {
|
||||||
t.Fatal("expected boolean environment overrides to be applied")
|
t.Fatal("expected boolean environment overrides to be applied")
|
||||||
}
|
}
|
||||||
if cfg.GlobalMaxFileSizeBytes != 512*1024*1024 {
|
if cfg.GlobalMaxFileSizeBytes != 100 {
|
||||||
t.Fatalf("unexpected global max file size: %d", cfg.GlobalMaxFileSizeBytes)
|
t.Fatalf("unexpected global max file size: %d", cfg.GlobalMaxFileSizeBytes)
|
||||||
}
|
}
|
||||||
if cfg.BoxPollIntervalMS != 2000 {
|
if cfg.BoxPollIntervalMS != 2000 {
|
||||||
@@ -67,8 +71,8 @@ func TestEnvironmentOverrides(t *testing.T) {
|
|||||||
if !cfg.OneTimeDownloadRetryOnFailure {
|
if !cfg.OneTimeDownloadRetryOnFailure {
|
||||||
t.Fatal("expected one-time retry-on-failure env override to be applied")
|
t.Fatal("expected one-time retry-on-failure env override to be applied")
|
||||||
}
|
}
|
||||||
if cfg.SecurityEnabled {
|
if cfg.BoxOwnerMaxRefreshCount != 5 || cfg.BoxOwnerPasswordEditEnabled {
|
||||||
t.Fatal("expected security features toggle from environment to be applied")
|
t.Fatal("expected box owner policy env overrides to be applied")
|
||||||
}
|
}
|
||||||
if cfg.Source(SettingAPIEnabled) != SourceEnv {
|
if cfg.Source(SettingAPIEnabled) != SourceEnv {
|
||||||
t.Fatalf("expected API setting source to be env, got %s", cfg.Source(SettingAPIEnabled))
|
t.Fatalf("expected API setting source to be env, got %s", cfg.Source(SettingAPIEnabled))
|
||||||
@@ -77,25 +81,25 @@ func TestEnvironmentOverrides(t *testing.T) {
|
|||||||
|
|
||||||
func TestMegabyteSizeEnvironmentOverrides(t *testing.T) {
|
func TestMegabyteSizeEnvironmentOverrides(t *testing.T) {
|
||||||
clearConfigEnv(t)
|
clearConfigEnv(t)
|
||||||
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_GB", "2")
|
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_MB", "2048")
|
||||||
t.Setenv("WARPBOX_GLOBAL_MAX_BOX_SIZE_GB", "4")
|
t.Setenv("WARPBOX_GLOBAL_MAX_BOX_SIZE_MB", "4096")
|
||||||
|
|
||||||
cfg, err := Load()
|
cfg, err := Load()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
t.Fatalf("Load returned error: %v", err)
|
t.Fatalf("Load returned error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if cfg.GlobalMaxFileSizeBytes != 2*1024*1024*1024 {
|
if cfg.GlobalMaxFileSizeBytes != 2048*1024*1024 {
|
||||||
t.Fatalf("unexpected global max file size: %d", cfg.GlobalMaxFileSizeBytes)
|
t.Fatalf("unexpected global max file size: %d", cfg.GlobalMaxFileSizeBytes)
|
||||||
}
|
}
|
||||||
if cfg.GlobalMaxBoxSizeBytes != 4*1024*1024*1024 {
|
if cfg.GlobalMaxBoxSizeBytes != 4096*1024*1024 {
|
||||||
t.Fatalf("unexpected global max box size: %d", cfg.GlobalMaxBoxSizeBytes)
|
t.Fatalf("unexpected global max box size: %d", cfg.GlobalMaxBoxSizeBytes)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func TestGBEnvironmentOverridesTakePrecedenceOverLegacySizeEnvNames(t *testing.T) {
|
func TestByteSizeEnvironmentOverridesTakePrecedence(t *testing.T) {
|
||||||
clearConfigEnv(t)
|
clearConfigEnv(t)
|
||||||
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_GB", "2")
|
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_MB", "2048")
|
||||||
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES", "100")
|
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES", "100")
|
||||||
|
|
||||||
cfg, err := Load()
|
cfg, err := Load()
|
||||||
@@ -103,7 +107,7 @@ func TestGBEnvironmentOverridesTakePrecedenceOverLegacySizeEnvNames(t *testing.T
|
|||||||
t.Fatalf("Load returned error: %v", err)
|
t.Fatalf("Load returned error: %v", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
if cfg.GlobalMaxFileSizeBytes != 2*1024*1024*1024 {
|
if cfg.GlobalMaxFileSizeBytes != 100 {
|
||||||
t.Fatalf("unexpected global max file size: %d", cfg.GlobalMaxFileSizeBytes)
|
t.Fatalf("unexpected global max file size: %d", cfg.GlobalMaxFileSizeBytes)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -152,14 +156,14 @@ func TestSettingsOverrideValidation(t *testing.T) {
|
|||||||
if err := cfg.ApplyOverride(SettingDefaultGuestExpirySecs, "-1"); err == nil {
|
if err := cfg.ApplyOverride(SettingDefaultGuestExpirySecs, "-1"); err == nil {
|
||||||
t.Fatal("expected negative expiry override to fail")
|
t.Fatal("expected negative expiry override to fail")
|
||||||
}
|
}
|
||||||
if err := cfg.ApplyOverride(SettingGlobalMaxFileSizeBytes, "0.5"); err != nil {
|
if err := cfg.ApplyOverride(SettingGlobalMaxFileSizeBytes, "1"); err == nil {
|
||||||
t.Fatalf("expected global max file size override to succeed, got %v", err)
|
t.Fatal("expected hard limit override to fail")
|
||||||
}
|
}
|
||||||
if cfg.GlobalMaxFileSizeBytes != 512*1024*1024 {
|
if err := cfg.ApplyOverride(SettingBoxOwnerMaxRefreshCount, "2"); err != nil {
|
||||||
t.Fatalf("expected global max file size override to apply, got %d", cfg.GlobalMaxFileSizeBytes)
|
t.Fatalf("expected box owner policy override to pass: %v", err)
|
||||||
}
|
}
|
||||||
if err := cfg.ApplyOverride(SettingDataDir, "/tmp/elsewhere"); err == nil {
|
if cfg.BoxOwnerMaxRefreshCount != 2 {
|
||||||
t.Fatal("expected data_dir override to remain locked")
|
t.Fatalf("expected box owner policy override to apply, got %d", cfg.BoxOwnerMaxRefreshCount)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -182,24 +186,24 @@ func clearConfigEnv(t *testing.T) {
|
|||||||
"WARPBOX_RENEW_ON_DOWNLOAD_ENABLED",
|
"WARPBOX_RENEW_ON_DOWNLOAD_ENABLED",
|
||||||
"WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS",
|
"WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS",
|
||||||
"WARPBOX_MAX_GUEST_EXPIRY_SECONDS",
|
"WARPBOX_MAX_GUEST_EXPIRY_SECONDS",
|
||||||
"WARPBOX_GLOBAL_MAX_FILE_SIZE_GB",
|
|
||||||
"WARPBOX_GLOBAL_MAX_FILE_SIZE_MB",
|
"WARPBOX_GLOBAL_MAX_FILE_SIZE_MB",
|
||||||
"WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES",
|
"WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES",
|
||||||
"WARPBOX_GLOBAL_MAX_BOX_SIZE_GB",
|
|
||||||
"WARPBOX_GLOBAL_MAX_BOX_SIZE_MB",
|
"WARPBOX_GLOBAL_MAX_BOX_SIZE_MB",
|
||||||
"WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES",
|
"WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES",
|
||||||
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_GB",
|
|
||||||
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_MB",
|
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_MB",
|
||||||
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES",
|
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES",
|
||||||
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_GB",
|
|
||||||
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_MB",
|
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_MB",
|
||||||
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES",
|
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES",
|
||||||
"WARPBOX_SESSION_TTL_SECONDS",
|
"WARPBOX_SESSION_TTL_SECONDS",
|
||||||
"WARPBOX_BOX_POLL_INTERVAL_MS",
|
"WARPBOX_BOX_POLL_INTERVAL_MS",
|
||||||
"WARPBOX_THUMBNAIL_BATCH_SIZE",
|
"WARPBOX_THUMBNAIL_BATCH_SIZE",
|
||||||
"WARPBOX_THUMBNAIL_INTERVAL_SECONDS",
|
"WARPBOX_THUMBNAIL_INTERVAL_SECONDS",
|
||||||
"WARPBOX_SECURITY_ENABLED",
|
"WARPBOX_BOX_OWNER_EDIT_ENABLED",
|
||||||
"WARPBOX_EXPIRED_CLEANUP_INTERVAL_SECONDS",
|
"WARPBOX_BOX_OWNER_REFRESH_ENABLED",
|
||||||
|
"WARPBOX_BOX_OWNER_MAX_REFRESH_COUNT",
|
||||||
|
"WARPBOX_BOX_OWNER_MAX_REFRESH_AMOUNT_SECONDS",
|
||||||
|
"WARPBOX_BOX_OWNER_MAX_TOTAL_EXPIRY_SECONDS",
|
||||||
|
"WARPBOX_BOX_OWNER_PASSWORD_EDIT_ENABLED",
|
||||||
} {
|
} {
|
||||||
t.Setenv(name, "")
|
t.Setenv(name, "")
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -12,28 +12,20 @@ var Definitions = []SettingDefinition{
|
|||||||
{Key: SettingRenewOnDownloadEnabled, EnvName: "WARPBOX_RENEW_ON_DOWNLOAD_ENABLED", Label: "Renew on download enabled", Type: SettingTypeBool, Editable: true},
|
{Key: SettingRenewOnDownloadEnabled, EnvName: "WARPBOX_RENEW_ON_DOWNLOAD_ENABLED", Label: "Renew on download enabled", Type: SettingTypeBool, Editable: true},
|
||||||
{Key: SettingDefaultGuestExpirySecs, EnvName: "WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS", Label: "Default guest expiry seconds", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
{Key: SettingDefaultGuestExpirySecs, EnvName: "WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS", Label: "Default guest expiry seconds", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
||||||
{Key: SettingMaxGuestExpirySecs, EnvName: "WARPBOX_MAX_GUEST_EXPIRY_SECONDS", Label: "Max guest expiry seconds", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
{Key: SettingMaxGuestExpirySecs, EnvName: "WARPBOX_MAX_GUEST_EXPIRY_SECONDS", Label: "Max guest expiry seconds", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
||||||
{Key: SettingGlobalMaxFileSizeBytes, EnvName: "WARPBOX_GLOBAL_MAX_FILE_SIZE_GB", Label: "Global max file size GB", Type: SettingTypeSizeGB, Editable: true, Minimum: 0},
|
{Key: SettingGlobalMaxFileSizeBytes, EnvName: "WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES", Label: "Global max file size bytes", Type: SettingTypeInt64, Editable: false, HardLimit: true, Minimum: 0},
|
||||||
{Key: SettingGlobalMaxBoxSizeBytes, EnvName: "WARPBOX_GLOBAL_MAX_BOX_SIZE_GB", Label: "Global max box size GB", Type: SettingTypeSizeGB, Editable: true, Minimum: 0},
|
{Key: SettingGlobalMaxBoxSizeBytes, EnvName: "WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES", Label: "Global max box size bytes", Type: SettingTypeInt64, Editable: false, HardLimit: true, Minimum: 0},
|
||||||
{Key: SettingDefaultUserMaxFileBytes, EnvName: "WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_GB", Label: "Default user max file size GB", Type: SettingTypeSizeGB, Editable: true, Minimum: 0},
|
{Key: SettingDefaultUserMaxFileBytes, EnvName: "WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES", Label: "Default user max file size bytes", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
||||||
{Key: SettingDefaultUserMaxBoxBytes, EnvName: "WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_GB", Label: "Default user max box size GB", Type: SettingTypeSizeGB, Editable: true, Minimum: 0},
|
{Key: SettingDefaultUserMaxBoxBytes, EnvName: "WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES", Label: "Default user max box size bytes", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
||||||
{Key: SettingSessionTTLSeconds, EnvName: "WARPBOX_SESSION_TTL_SECONDS", Label: "Session TTL seconds", Type: SettingTypeInt64, Editable: true, Minimum: 60},
|
{Key: SettingSessionTTLSeconds, EnvName: "WARPBOX_SESSION_TTL_SECONDS", Label: "Session TTL seconds", Type: SettingTypeInt64, Editable: true, Minimum: 60},
|
||||||
{Key: SettingBoxPollIntervalMS, EnvName: "WARPBOX_BOX_POLL_INTERVAL_MS", Label: "Box poll interval milliseconds", Type: SettingTypeInt, Editable: true, Minimum: 1000},
|
{Key: SettingBoxPollIntervalMS, EnvName: "WARPBOX_BOX_POLL_INTERVAL_MS", Label: "Box poll interval milliseconds", Type: SettingTypeInt, Editable: true, Minimum: 1000},
|
||||||
{Key: SettingThumbnailBatchSize, EnvName: "WARPBOX_THUMBNAIL_BATCH_SIZE", Label: "Thumbnail batch size", Type: SettingTypeInt, Editable: true, Minimum: 1},
|
{Key: SettingThumbnailBatchSize, EnvName: "WARPBOX_THUMBNAIL_BATCH_SIZE", Label: "Thumbnail batch size", Type: SettingTypeInt, Editable: true, Minimum: 1},
|
||||||
{Key: SettingThumbnailIntervalSeconds, EnvName: "WARPBOX_THUMBNAIL_INTERVAL_SECONDS", Label: "Thumbnail interval seconds", Type: SettingTypeInt, Editable: true, Minimum: 1},
|
{Key: SettingThumbnailIntervalSeconds, EnvName: "WARPBOX_THUMBNAIL_INTERVAL_SECONDS", Label: "Thumbnail interval seconds", Type: SettingTypeInt, Editable: true, Minimum: 1},
|
||||||
{Key: SettingActivityRetentionSeconds, EnvName: "WARPBOX_ACTIVITY_RETENTION_SECONDS", Label: "Activity retention seconds", Type: SettingTypeInt64, Editable: true, Minimum: 60},
|
{Key: SettingBoxOwnerEditEnabled, EnvName: "WARPBOX_BOX_OWNER_EDIT_ENABLED", Label: "Box owner edit enabled", Type: SettingTypeBool, Editable: true},
|
||||||
{Key: SettingSecurityEnabled, EnvName: "WARPBOX_SECURITY_ENABLED", Label: "Security features enabled", Type: SettingTypeBool, Editable: true},
|
{Key: SettingBoxOwnerRefreshEnabled, EnvName: "WARPBOX_BOX_OWNER_REFRESH_ENABLED", Label: "Box owner refresh enabled", Type: SettingTypeBool, Editable: true},
|
||||||
{Key: SettingSecurityIPWhitelist, EnvName: "WARPBOX_SECURITY_IP_WHITELIST", Label: "Security IP whitelist", Type: SettingTypeText, Editable: true},
|
{Key: SettingBoxOwnerMaxRefreshCount, EnvName: "WARPBOX_BOX_OWNER_MAX_REFRESH_COUNT", Label: "Box owner max refresh count", Type: SettingTypeInt, Editable: true, Minimum: 0},
|
||||||
{Key: SettingSecurityAdminIPWhitelist, EnvName: "WARPBOX_SECURITY_ADMIN_IP_WHITELIST", Label: "Security admin IP whitelist", Type: SettingTypeText, Editable: true},
|
{Key: SettingBoxOwnerMaxRefreshAmount, EnvName: "WARPBOX_BOX_OWNER_MAX_REFRESH_AMOUNT_SECONDS", Label: "Box owner max refresh amount seconds", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
||||||
{Key: SettingTrustedProxyCIDRs, EnvName: "WARPBOX_TRUSTED_PROXY_CIDRS", Label: "Trusted proxy CIDRs", Type: SettingTypeText, Editable: true},
|
{Key: SettingBoxOwnerMaxTotalExpiry, EnvName: "WARPBOX_BOX_OWNER_MAX_TOTAL_EXPIRY_SECONDS", Label: "Box owner max total expiry seconds", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
||||||
{Key: SettingSecurityLoginWindowSecs, EnvName: "WARPBOX_SECURITY_LOGIN_WINDOW_SECONDS", Label: "Login attempt window seconds", Type: SettingTypeInt64, Editable: true, Minimum: 10},
|
{Key: SettingBoxOwnerPasswordEdit, EnvName: "WARPBOX_BOX_OWNER_PASSWORD_EDIT_ENABLED", Label: "Box owner password edit enabled", Type: SettingTypeBool, Editable: true},
|
||||||
{Key: SettingSecurityLoginMaxAttempts, EnvName: "WARPBOX_SECURITY_LOGIN_MAX_ATTEMPTS", Label: "Login max attempts per window", Type: SettingTypeInt, Editable: true, Minimum: 1},
|
|
||||||
{Key: SettingSecurityBanSeconds, EnvName: "WARPBOX_SECURITY_BAN_SECONDS", Label: "Security ban seconds", Type: SettingTypeInt64, Editable: true, Minimum: 10},
|
|
||||||
{Key: SettingSecurityScanWindowSecs, EnvName: "WARPBOX_SECURITY_SCAN_WINDOW_SECONDS", Label: "Malicious path window seconds", Type: SettingTypeInt64, Editable: true, Minimum: 10},
|
|
||||||
{Key: SettingSecurityScanMaxAttempts, EnvName: "WARPBOX_SECURITY_SCAN_MAX_ATTEMPTS", Label: "Malicious path max attempts", Type: SettingTypeInt, Editable: true, Minimum: 1},
|
|
||||||
{Key: SettingSecurityUploadWindowSecs, EnvName: "WARPBOX_SECURITY_UPLOAD_WINDOW_SECONDS", Label: "Upload limit window seconds", Type: SettingTypeInt64, Editable: true, Minimum: 10},
|
|
||||||
{Key: SettingSecurityUploadMaxRequests, EnvName: "WARPBOX_SECURITY_UPLOAD_MAX_REQUESTS", Label: "Upload max requests per window", Type: SettingTypeInt, Editable: true, Minimum: 1},
|
|
||||||
{Key: SettingSecurityUploadMaxGB, EnvName: "WARPBOX_SECURITY_UPLOAD_MAX_GB", Label: "Upload max total GB per window", Type: SettingTypeSizeGB, Editable: true, Minimum: 0},
|
|
||||||
{Key: SettingExpiredCleanupIntervalSecs, EnvName: "WARPBOX_EXPIRED_CLEANUP_INTERVAL_SECONDS", Label: "Expired boxes cleanup interval seconds", Type: SettingTypeInt64, Editable: true, Minimum: 0},
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cfg *Config) SettingRows() []SettingRow {
|
func (cfg *Config) SettingRows() []SettingRow {
|
||||||
@@ -52,6 +44,10 @@ func (cfg *Config) Source(key string) Source {
|
|||||||
return cfg.sourceFor(key)
|
return cfg.sourceFor(key)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (cfg *Config) SettingValue(key string) string {
|
||||||
|
return cfg.values[key]
|
||||||
|
}
|
||||||
|
|
||||||
func (cfg *Config) AdminLoginEnabled(hasAdminUser bool) bool {
|
func (cfg *Config) AdminLoginEnabled(hasAdminUser bool) bool {
|
||||||
switch cfg.AdminEnabled {
|
switch cfg.AdminEnabled {
|
||||||
case AdminEnabledFalse:
|
case AdminEnabledFalse:
|
||||||
@@ -64,7 +60,6 @@ func (cfg *Config) AdminLoginEnabled(hasAdminUser bool) bool {
|
|||||||
}
|
}
|
||||||
|
|
||||||
func Definition(key string) (SettingDefinition, bool) {
|
func Definition(key string) (SettingDefinition, bool) {
|
||||||
key = NormalizeLegacySettingKey(key)
|
|
||||||
for _, def := range Definitions {
|
for _, def := range Definitions {
|
||||||
if def.Key == key {
|
if def.Key == key {
|
||||||
return def, true
|
return def, true
|
||||||
@@ -73,35 +68,6 @@ func Definition(key string) (SettingDefinition, bool) {
|
|||||||
return SettingDefinition{}, false
|
return SettingDefinition{}, false
|
||||||
}
|
}
|
||||||
|
|
||||||
func NormalizeLegacySettingKey(key string) string {
|
|
||||||
switch key {
|
|
||||||
case "global_max_file_size_bytes":
|
|
||||||
return SettingGlobalMaxFileSizeBytes
|
|
||||||
case "global_max_box_size_bytes":
|
|
||||||
return SettingGlobalMaxBoxSizeBytes
|
|
||||||
case "default_user_max_file_size_bytes":
|
|
||||||
return SettingDefaultUserMaxFileBytes
|
|
||||||
case "default_user_max_box_size_bytes":
|
|
||||||
return SettingDefaultUserMaxBoxBytes
|
|
||||||
default:
|
|
||||||
return key
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func NormalizeOverrideInput(key string, value string) (string, string, error) {
|
|
||||||
normalizedKey := NormalizeLegacySettingKey(key)
|
|
||||||
switch key {
|
|
||||||
case "global_max_file_size_bytes", "global_max_box_size_bytes", "default_user_max_file_size_bytes", "default_user_max_box_size_bytes":
|
|
||||||
parsed, err := parseInt64(value, 0)
|
|
||||||
if err != nil {
|
|
||||||
return normalizedKey, "", err
|
|
||||||
}
|
|
||||||
return normalizedKey, formatGigabytesFromBytes(parsed), nil
|
|
||||||
default:
|
|
||||||
return normalizedKey, value, nil
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func EditableDefinitions() []SettingDefinition {
|
func EditableDefinitions() []SettingDefinition {
|
||||||
defs := make([]SettingDefinition, 0, len(Definitions))
|
defs := make([]SettingDefinition, 0, len(Definitions))
|
||||||
for _, def := range Definitions {
|
for _, def := range Definitions {
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package config
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
|
"math"
|
||||||
"os"
|
"os"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strconv"
|
"strconv"
|
||||||
@@ -26,20 +27,14 @@ func Load() (*Config, error) {
|
|||||||
BoxPollIntervalMS: 5000,
|
BoxPollIntervalMS: 5000,
|
||||||
ThumbnailBatchSize: 10,
|
ThumbnailBatchSize: 10,
|
||||||
ThumbnailIntervalSeconds: 30,
|
ThumbnailIntervalSeconds: 30,
|
||||||
ActivityRetentionSeconds: 7 * 24 * 60 * 60,
|
BoxOwnerEditEnabled: true,
|
||||||
SecurityEnabled: true,
|
BoxOwnerRefreshEnabled: true,
|
||||||
SecurityLoginWindowSeconds: 10 * 60,
|
BoxOwnerMaxRefreshCount: 3,
|
||||||
SecurityLoginMaxAttempts: 8,
|
BoxOwnerMaxRefreshAmountSeconds: 24 * 60 * 60,
|
||||||
SecurityBanSeconds: 30 * 60,
|
BoxOwnerMaxTotalExpirySeconds: 7 * 24 * 60 * 60,
|
||||||
SecurityScanWindowSeconds: 5 * 60,
|
BoxOwnerPasswordEditEnabled: true,
|
||||||
SecurityScanMaxAttempts: 12,
|
|
||||||
SecurityUploadWindowSeconds: 60,
|
|
||||||
SecurityUploadMaxRequests: 20,
|
|
||||||
SecurityUploadMaxBytes: 10 * 1024 * 1024 * 1024,
|
|
||||||
ExpiredCleanupIntervalSeconds: 300,
|
|
||||||
sources: make(map[string]Source),
|
sources: make(map[string]Source),
|
||||||
values: make(map[string]string),
|
values: make(map[string]string),
|
||||||
defaults: make(map[string]string),
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Config precedence: defaults -> env -> overrides.
|
// Config precedence: defaults -> env -> overrides.
|
||||||
@@ -58,15 +53,6 @@ func Load() (*Config, error) {
|
|||||||
if err := cfg.applyStringEnv("", "WARPBOX_ADMIN_EMAIL", &cfg.AdminEmail); err != nil {
|
if err := cfg.applyStringEnv("", "WARPBOX_ADMIN_EMAIL", &cfg.AdminEmail); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
if err := cfg.applyStringEnv(SettingSecurityIPWhitelist, "WARPBOX_SECURITY_IP_WHITELIST", &cfg.SecurityIPWhitelist); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err := cfg.applyStringEnv(SettingSecurityAdminIPWhitelist, "WARPBOX_SECURITY_ADMIN_IP_WHITELIST", &cfg.SecurityAdminIPWhitelist); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err := cfg.applyStringEnv(SettingTrustedProxyCIDRs, "WARPBOX_TRUSTED_PROXY_CIDRS", &cfg.TrustedProxyCIDRs); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if raw := strings.TrimSpace(os.Getenv("WARPBOX_ADMIN_ENABLED")); raw != "" {
|
if raw := strings.TrimSpace(os.Getenv("WARPBOX_ADMIN_ENABLED")); raw != "" {
|
||||||
mode := AdminEnabledMode(strings.ToLower(raw))
|
mode := AdminEnabledMode(strings.ToLower(raw))
|
||||||
if mode != AdminEnabledAuto && mode != AdminEnabledTrue && mode != AdminEnabledFalse {
|
if mode != AdminEnabledAuto && mode != AdminEnabledTrue && mode != AdminEnabledFalse {
|
||||||
@@ -93,7 +79,9 @@ func Load() (*Config, error) {
|
|||||||
{SettingOneTimeDownloadRetryFail, "WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE", &cfg.OneTimeDownloadRetryOnFailure},
|
{SettingOneTimeDownloadRetryFail, "WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE", &cfg.OneTimeDownloadRetryOnFailure},
|
||||||
{SettingRenewOnAccessEnabled, "WARPBOX_RENEW_ON_ACCESS_ENABLED", &cfg.RenewOnAccessEnabled},
|
{SettingRenewOnAccessEnabled, "WARPBOX_RENEW_ON_ACCESS_ENABLED", &cfg.RenewOnAccessEnabled},
|
||||||
{SettingRenewOnDownloadEnabled, "WARPBOX_RENEW_ON_DOWNLOAD_ENABLED", &cfg.RenewOnDownloadEnabled},
|
{SettingRenewOnDownloadEnabled, "WARPBOX_RENEW_ON_DOWNLOAD_ENABLED", &cfg.RenewOnDownloadEnabled},
|
||||||
{SettingSecurityEnabled, "WARPBOX_SECURITY_ENABLED", &cfg.SecurityEnabled},
|
{SettingBoxOwnerEditEnabled, "WARPBOX_BOX_OWNER_EDIT_ENABLED", &cfg.BoxOwnerEditEnabled},
|
||||||
|
{SettingBoxOwnerRefreshEnabled, "WARPBOX_BOX_OWNER_REFRESH_ENABLED", &cfg.BoxOwnerRefreshEnabled},
|
||||||
|
{SettingBoxOwnerPasswordEdit, "WARPBOX_BOX_OWNER_PASSWORD_EDIT_ENABLED", &cfg.BoxOwnerPasswordEditEnabled},
|
||||||
}
|
}
|
||||||
for _, item := range envBools {
|
for _, item := range envBools {
|
||||||
if err := cfg.applyBoolEnv(item.key, item.name, item.target); err != nil {
|
if err := cfg.applyBoolEnv(item.key, item.name, item.target); err != nil {
|
||||||
@@ -111,12 +99,8 @@ func Load() (*Config, error) {
|
|||||||
{SettingMaxGuestExpirySecs, "WARPBOX_MAX_GUEST_EXPIRY_SECONDS", 0, &cfg.MaxGuestExpirySeconds},
|
{SettingMaxGuestExpirySecs, "WARPBOX_MAX_GUEST_EXPIRY_SECONDS", 0, &cfg.MaxGuestExpirySeconds},
|
||||||
{SettingOneTimeDownloadExpirySecs, "WARPBOX_ONE_TIME_DOWNLOAD_EXPIRY_SECONDS", 0, &cfg.OneTimeDownloadExpirySeconds},
|
{SettingOneTimeDownloadExpirySecs, "WARPBOX_ONE_TIME_DOWNLOAD_EXPIRY_SECONDS", 0, &cfg.OneTimeDownloadExpirySeconds},
|
||||||
{SettingSessionTTLSeconds, "WARPBOX_SESSION_TTL_SECONDS", 60, &cfg.SessionTTLSeconds},
|
{SettingSessionTTLSeconds, "WARPBOX_SESSION_TTL_SECONDS", 60, &cfg.SessionTTLSeconds},
|
||||||
{SettingActivityRetentionSeconds, "WARPBOX_ACTIVITY_RETENTION_SECONDS", 60, &cfg.ActivityRetentionSeconds},
|
{SettingBoxOwnerMaxRefreshAmount, "WARPBOX_BOX_OWNER_MAX_REFRESH_AMOUNT_SECONDS", 0, &cfg.BoxOwnerMaxRefreshAmountSeconds},
|
||||||
{SettingSecurityLoginWindowSecs, "WARPBOX_SECURITY_LOGIN_WINDOW_SECONDS", 10, &cfg.SecurityLoginWindowSeconds},
|
{SettingBoxOwnerMaxTotalExpiry, "WARPBOX_BOX_OWNER_MAX_TOTAL_EXPIRY_SECONDS", 0, &cfg.BoxOwnerMaxTotalExpirySeconds},
|
||||||
{SettingSecurityBanSeconds, "WARPBOX_SECURITY_BAN_SECONDS", 10, &cfg.SecurityBanSeconds},
|
|
||||||
{SettingSecurityScanWindowSecs, "WARPBOX_SECURITY_SCAN_WINDOW_SECONDS", 10, &cfg.SecurityScanWindowSeconds},
|
|
||||||
{SettingSecurityUploadWindowSecs, "WARPBOX_SECURITY_UPLOAD_WINDOW_SECONDS", 10, &cfg.SecurityUploadWindowSeconds},
|
|
||||||
{SettingExpiredCleanupIntervalSecs, "WARPBOX_EXPIRED_CLEANUP_INTERVAL_SECONDS", 0, &cfg.ExpiredCleanupIntervalSeconds},
|
|
||||||
}
|
}
|
||||||
for _, item := range envInt64s {
|
for _, item := range envInt64s {
|
||||||
if err := cfg.applyInt64Env(item.key, item.name, item.min, item.target); err != nil {
|
if err := cfg.applyInt64Env(item.key, item.name, item.min, item.target); err != nil {
|
||||||
@@ -125,19 +109,17 @@ func Load() (*Config, error) {
|
|||||||
}
|
}
|
||||||
sizeEnvVars := []struct {
|
sizeEnvVars := []struct {
|
||||||
key string
|
key string
|
||||||
gbName string
|
|
||||||
mbName string
|
mbName string
|
||||||
bytesName string
|
bytesName string
|
||||||
target *int64
|
target *int64
|
||||||
}{
|
}{
|
||||||
{SettingGlobalMaxFileSizeBytes, "WARPBOX_GLOBAL_MAX_FILE_SIZE_GB", "WARPBOX_GLOBAL_MAX_FILE_SIZE_MB", "WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES", &cfg.GlobalMaxFileSizeBytes},
|
{SettingGlobalMaxFileSizeBytes, "WARPBOX_GLOBAL_MAX_FILE_SIZE_MB", "WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES", &cfg.GlobalMaxFileSizeBytes},
|
||||||
{SettingGlobalMaxBoxSizeBytes, "WARPBOX_GLOBAL_MAX_BOX_SIZE_GB", "WARPBOX_GLOBAL_MAX_BOX_SIZE_MB", "WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES", &cfg.GlobalMaxBoxSizeBytes},
|
{SettingGlobalMaxBoxSizeBytes, "WARPBOX_GLOBAL_MAX_BOX_SIZE_MB", "WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES", &cfg.GlobalMaxBoxSizeBytes},
|
||||||
{SettingDefaultUserMaxFileBytes, "WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_GB", "WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_MB", "WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES", &cfg.DefaultUserMaxFileSizeBytes},
|
{SettingDefaultUserMaxFileBytes, "WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_MB", "WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES", &cfg.DefaultUserMaxFileSizeBytes},
|
||||||
{SettingDefaultUserMaxBoxBytes, "WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_GB", "WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_MB", "WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES", &cfg.DefaultUserMaxBoxSizeBytes},
|
{SettingDefaultUserMaxBoxBytes, "WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_MB", "WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES", &cfg.DefaultUserMaxBoxSizeBytes},
|
||||||
{SettingSecurityUploadMaxGB, "WARPBOX_SECURITY_UPLOAD_MAX_GB", "WARPBOX_SECURITY_UPLOAD_MAX_MB", "WARPBOX_SECURITY_UPLOAD_MAX_BYTES", &cfg.SecurityUploadMaxBytes},
|
|
||||||
}
|
}
|
||||||
for _, item := range sizeEnvVars {
|
for _, item := range sizeEnvVars {
|
||||||
if err := cfg.applySizeEnv(item.key, item.gbName, item.mbName, item.bytesName, 0, item.target); err != nil {
|
if err := cfg.applyMegabytesOrBytesEnv(item.key, item.mbName, item.bytesName, 0, item.target); err != nil {
|
||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -151,9 +133,7 @@ func Load() (*Config, error) {
|
|||||||
{SettingBoxPollIntervalMS, "WARPBOX_BOX_POLL_INTERVAL_MS", 1000, &cfg.BoxPollIntervalMS},
|
{SettingBoxPollIntervalMS, "WARPBOX_BOX_POLL_INTERVAL_MS", 1000, &cfg.BoxPollIntervalMS},
|
||||||
{SettingThumbnailBatchSize, "WARPBOX_THUMBNAIL_BATCH_SIZE", 1, &cfg.ThumbnailBatchSize},
|
{SettingThumbnailBatchSize, "WARPBOX_THUMBNAIL_BATCH_SIZE", 1, &cfg.ThumbnailBatchSize},
|
||||||
{SettingThumbnailIntervalSeconds, "WARPBOX_THUMBNAIL_INTERVAL_SECONDS", 1, &cfg.ThumbnailIntervalSeconds},
|
{SettingThumbnailIntervalSeconds, "WARPBOX_THUMBNAIL_INTERVAL_SECONDS", 1, &cfg.ThumbnailIntervalSeconds},
|
||||||
{SettingSecurityLoginMaxAttempts, "WARPBOX_SECURITY_LOGIN_MAX_ATTEMPTS", 1, &cfg.SecurityLoginMaxAttempts},
|
{SettingBoxOwnerMaxRefreshCount, "WARPBOX_BOX_OWNER_MAX_REFRESH_COUNT", 0, &cfg.BoxOwnerMaxRefreshCount},
|
||||||
{SettingSecurityScanMaxAttempts, "WARPBOX_SECURITY_SCAN_MAX_ATTEMPTS", 1, &cfg.SecurityScanMaxAttempts},
|
|
||||||
{SettingSecurityUploadMaxRequests, "WARPBOX_SECURITY_UPLOAD_MAX_REQUESTS", 1, &cfg.SecurityUploadMaxRequests},
|
|
||||||
}
|
}
|
||||||
for _, item := range envInts {
|
for _, item := range envInts {
|
||||||
if err := cfg.applyIntEnv(item.key, item.name, item.min, item.target); err != nil {
|
if err := cfg.applyIntEnv(item.key, item.name, item.min, item.target); err != nil {
|
||||||
@@ -169,15 +149,6 @@ func Load() (*Config, error) {
|
|||||||
return nil, fmt.Errorf("WARPBOX_ADMIN_USERNAME cannot be empty")
|
return nil, fmt.Errorf("WARPBOX_ADMIN_USERNAME cannot be empty")
|
||||||
}
|
}
|
||||||
cfg.AdminEmail = strings.TrimSpace(cfg.AdminEmail)
|
cfg.AdminEmail = strings.TrimSpace(cfg.AdminEmail)
|
||||||
if err := validateSecurityTextSetting(SettingSecurityIPWhitelist, cfg.SecurityIPWhitelist); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err := validateSecurityTextSetting(SettingSecurityAdminIPWhitelist, cfg.SecurityAdminIPWhitelist); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if err := validateSecurityTextSetting(SettingTrustedProxyCIDRs, cfg.TrustedProxyCIDRs); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
cfg.UploadsDir = filepath.Join(cfg.DataDir, "uploads")
|
cfg.UploadsDir = filepath.Join(cfg.DataDir, "uploads")
|
||||||
cfg.DBDir = filepath.Join(cfg.DataDir, "db")
|
cfg.DBDir = filepath.Join(cfg.DataDir, "db")
|
||||||
cfg.setValue(SettingDataDir, cfg.DataDir, cfg.sourceFor(SettingDataDir))
|
cfg.setValue(SettingDataDir, cfg.DataDir, cfg.sourceFor(SettingDataDir))
|
||||||
@@ -193,46 +164,31 @@ func (cfg *Config) EnsureDirectories() error {
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
func (cfg *Config) captureDefaults() {
|
func (cfg *Config) captureDefaults() {
|
||||||
cfg.captureDefaultValue(SettingDataDir, cfg.DataDir)
|
cfg.setValue(SettingDataDir, cfg.DataDir, SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingGuestUploadsEnabled, formatBool(cfg.GuestUploadsEnabled))
|
cfg.setValue(SettingGuestUploadsEnabled, formatBool(cfg.GuestUploadsEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingAPIEnabled, formatBool(cfg.APIEnabled))
|
cfg.setValue(SettingAPIEnabled, formatBool(cfg.APIEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingZipDownloadsEnabled, formatBool(cfg.ZipDownloadsEnabled))
|
cfg.setValue(SettingZipDownloadsEnabled, formatBool(cfg.ZipDownloadsEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingOneTimeDownloadsEnabled, formatBool(cfg.OneTimeDownloadsEnabled))
|
cfg.setValue(SettingOneTimeDownloadsEnabled, formatBool(cfg.OneTimeDownloadsEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingOneTimeDownloadExpirySecs, strconv.FormatInt(cfg.OneTimeDownloadExpirySeconds, 10))
|
cfg.setValue(SettingOneTimeDownloadExpirySecs, strconv.FormatInt(cfg.OneTimeDownloadExpirySeconds, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingOneTimeDownloadRetryFail, formatBool(cfg.OneTimeDownloadRetryOnFailure))
|
cfg.setValue(SettingOneTimeDownloadRetryFail, formatBool(cfg.OneTimeDownloadRetryOnFailure), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingRenewOnAccessEnabled, formatBool(cfg.RenewOnAccessEnabled))
|
cfg.setValue(SettingRenewOnAccessEnabled, formatBool(cfg.RenewOnAccessEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingRenewOnDownloadEnabled, formatBool(cfg.RenewOnDownloadEnabled))
|
cfg.setValue(SettingRenewOnDownloadEnabled, formatBool(cfg.RenewOnDownloadEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingDefaultGuestExpirySecs, strconv.FormatInt(cfg.DefaultGuestExpirySeconds, 10))
|
cfg.setValue(SettingDefaultGuestExpirySecs, strconv.FormatInt(cfg.DefaultGuestExpirySeconds, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingMaxGuestExpirySecs, strconv.FormatInt(cfg.MaxGuestExpirySeconds, 10))
|
cfg.setValue(SettingMaxGuestExpirySecs, strconv.FormatInt(cfg.MaxGuestExpirySeconds, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingGlobalMaxFileSizeBytes, formatGigabytesFromBytes(cfg.GlobalMaxFileSizeBytes))
|
cfg.setValue(SettingGlobalMaxFileSizeBytes, strconv.FormatInt(cfg.GlobalMaxFileSizeBytes, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingGlobalMaxBoxSizeBytes, formatGigabytesFromBytes(cfg.GlobalMaxBoxSizeBytes))
|
cfg.setValue(SettingGlobalMaxBoxSizeBytes, strconv.FormatInt(cfg.GlobalMaxBoxSizeBytes, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingDefaultUserMaxFileBytes, formatGigabytesFromBytes(cfg.DefaultUserMaxFileSizeBytes))
|
cfg.setValue(SettingDefaultUserMaxFileBytes, strconv.FormatInt(cfg.DefaultUserMaxFileSizeBytes, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingDefaultUserMaxBoxBytes, formatGigabytesFromBytes(cfg.DefaultUserMaxBoxSizeBytes))
|
cfg.setValue(SettingDefaultUserMaxBoxBytes, strconv.FormatInt(cfg.DefaultUserMaxBoxSizeBytes, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingSessionTTLSeconds, strconv.FormatInt(cfg.SessionTTLSeconds, 10))
|
cfg.setValue(SettingSessionTTLSeconds, strconv.FormatInt(cfg.SessionTTLSeconds, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingBoxPollIntervalMS, strconv.Itoa(cfg.BoxPollIntervalMS))
|
cfg.setValue(SettingBoxPollIntervalMS, strconv.Itoa(cfg.BoxPollIntervalMS), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingThumbnailBatchSize, strconv.Itoa(cfg.ThumbnailBatchSize))
|
cfg.setValue(SettingThumbnailBatchSize, strconv.Itoa(cfg.ThumbnailBatchSize), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingThumbnailIntervalSeconds, strconv.Itoa(cfg.ThumbnailIntervalSeconds))
|
cfg.setValue(SettingThumbnailIntervalSeconds, strconv.Itoa(cfg.ThumbnailIntervalSeconds), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingActivityRetentionSeconds, strconv.FormatInt(cfg.ActivityRetentionSeconds, 10))
|
cfg.setValue(SettingBoxOwnerEditEnabled, formatBool(cfg.BoxOwnerEditEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingSecurityEnabled, formatBool(cfg.SecurityEnabled))
|
cfg.setValue(SettingBoxOwnerRefreshEnabled, formatBool(cfg.BoxOwnerRefreshEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingSecurityIPWhitelist, cfg.SecurityIPWhitelist)
|
cfg.setValue(SettingBoxOwnerMaxRefreshCount, strconv.Itoa(cfg.BoxOwnerMaxRefreshCount), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingSecurityAdminIPWhitelist, cfg.SecurityAdminIPWhitelist)
|
cfg.setValue(SettingBoxOwnerMaxRefreshAmount, strconv.FormatInt(cfg.BoxOwnerMaxRefreshAmountSeconds, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingTrustedProxyCIDRs, cfg.TrustedProxyCIDRs)
|
cfg.setValue(SettingBoxOwnerMaxTotalExpiry, strconv.FormatInt(cfg.BoxOwnerMaxTotalExpirySeconds, 10), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingSecurityLoginWindowSecs, strconv.FormatInt(cfg.SecurityLoginWindowSeconds, 10))
|
cfg.setValue(SettingBoxOwnerPasswordEdit, formatBool(cfg.BoxOwnerPasswordEditEnabled), SourceDefault)
|
||||||
cfg.captureDefaultValue(SettingSecurityLoginMaxAttempts, strconv.Itoa(cfg.SecurityLoginMaxAttempts))
|
|
||||||
cfg.captureDefaultValue(SettingSecurityBanSeconds, strconv.FormatInt(cfg.SecurityBanSeconds, 10))
|
|
||||||
cfg.captureDefaultValue(SettingSecurityScanWindowSecs, strconv.FormatInt(cfg.SecurityScanWindowSeconds, 10))
|
|
||||||
cfg.captureDefaultValue(SettingSecurityScanMaxAttempts, strconv.Itoa(cfg.SecurityScanMaxAttempts))
|
|
||||||
cfg.captureDefaultValue(SettingSecurityUploadWindowSecs, strconv.FormatInt(cfg.SecurityUploadWindowSeconds, 10))
|
|
||||||
cfg.captureDefaultValue(SettingSecurityUploadMaxRequests, strconv.Itoa(cfg.SecurityUploadMaxRequests))
|
|
||||||
cfg.captureDefaultValue(SettingSecurityUploadMaxGB, formatGigabytesFromBytes(cfg.SecurityUploadMaxBytes))
|
|
||||||
cfg.captureDefaultValue(SettingExpiredCleanupIntervalSecs, strconv.FormatInt(cfg.ExpiredCleanupIntervalSeconds, 10))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cfg *Config) captureDefaultValue(key string, value string) {
|
|
||||||
cfg.setValue(key, value, SourceDefault)
|
|
||||||
if cfg.defaults != nil {
|
|
||||||
cfg.defaults[key] = value
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cfg *Config) applyStringEnv(key string, name string, target *string) error {
|
func (cfg *Config) applyStringEnv(key string, name string, target *string) error {
|
||||||
@@ -279,23 +235,14 @@ func (cfg *Config) applyInt64Env(key string, name string, min int64, target *int
|
|||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cfg *Config) applySizeEnv(key string, gbName string, mbName string, bytesName string, min int64, target *int64) error {
|
func (cfg *Config) applyMegabytesOrBytesEnv(key string, mbName string, bytesName string, min int64, target *int64) error {
|
||||||
if rawGB := strings.TrimSpace(os.Getenv(gbName)); rawGB != "" {
|
|
||||||
parsed, err := parseGigabytes(rawGB, float64(min))
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("%s: %w", gbName, err)
|
|
||||||
}
|
|
||||||
*target = parsed
|
|
||||||
cfg.setValue(key, formatGigabytesFromBytes(parsed), SourceEnv)
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if rawBytes := strings.TrimSpace(os.Getenv(bytesName)); rawBytes != "" {
|
if rawBytes := strings.TrimSpace(os.Getenv(bytesName)); rawBytes != "" {
|
||||||
parsed, err := parseInt64(rawBytes, min)
|
parsed, err := parseInt64(rawBytes, min)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%s: %w", bytesName, err)
|
return fmt.Errorf("%s: %w", bytesName, err)
|
||||||
}
|
}
|
||||||
*target = parsed
|
*target = parsed
|
||||||
cfg.setValue(key, formatGigabytesFromBytes(parsed), SourceEnv)
|
cfg.setValue(key, strconv.FormatInt(parsed, 10), SourceEnv)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -307,9 +254,12 @@ func (cfg *Config) applySizeEnv(key string, gbName string, mbName string, bytesN
|
|||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%s: %w", mbName, err)
|
return fmt.Errorf("%s: %w", mbName, err)
|
||||||
}
|
}
|
||||||
parsedBytes := parsedMB * 1000 * 1000
|
if parsedMB > math.MaxInt64/(1024*1024) {
|
||||||
|
return fmt.Errorf("%s: is too large", mbName)
|
||||||
|
}
|
||||||
|
parsedBytes := parsedMB * 1024 * 1024
|
||||||
*target = parsedBytes
|
*target = parsedBytes
|
||||||
cfg.setValue(key, formatGigabytesFromBytes(parsedBytes), SourceEnv)
|
cfg.setValue(key, strconv.FormatInt(parsedBytes, 10), SourceEnv)
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -27,29 +27,21 @@ const (
|
|||||||
SettingRenewOnDownloadEnabled = "renew_on_download_enabled"
|
SettingRenewOnDownloadEnabled = "renew_on_download_enabled"
|
||||||
SettingDefaultGuestExpirySecs = "default_guest_expiry_seconds"
|
SettingDefaultGuestExpirySecs = "default_guest_expiry_seconds"
|
||||||
SettingMaxGuestExpirySecs = "max_guest_expiry_seconds"
|
SettingMaxGuestExpirySecs = "max_guest_expiry_seconds"
|
||||||
SettingGlobalMaxFileSizeBytes = "global_max_file_size_gb"
|
SettingGlobalMaxFileSizeBytes = "global_max_file_size_bytes"
|
||||||
SettingGlobalMaxBoxSizeBytes = "global_max_box_size_gb"
|
SettingGlobalMaxBoxSizeBytes = "global_max_box_size_bytes"
|
||||||
SettingDefaultUserMaxFileBytes = "default_user_max_file_size_gb"
|
SettingDefaultUserMaxFileBytes = "default_user_max_file_size_bytes"
|
||||||
SettingDefaultUserMaxBoxBytes = "default_user_max_box_size_gb"
|
SettingDefaultUserMaxBoxBytes = "default_user_max_box_size_bytes"
|
||||||
SettingSessionTTLSeconds = "session_ttl_seconds"
|
SettingSessionTTLSeconds = "session_ttl_seconds"
|
||||||
SettingBoxPollIntervalMS = "box_poll_interval_ms"
|
SettingBoxPollIntervalMS = "box_poll_interval_ms"
|
||||||
SettingThumbnailBatchSize = "thumbnail_batch_size"
|
SettingThumbnailBatchSize = "thumbnail_batch_size"
|
||||||
SettingThumbnailIntervalSeconds = "thumbnail_interval_seconds"
|
SettingThumbnailIntervalSeconds = "thumbnail_interval_seconds"
|
||||||
SettingDataDir = "data_dir"
|
SettingDataDir = "data_dir"
|
||||||
SettingActivityRetentionSeconds = "activity_retention_seconds"
|
SettingBoxOwnerEditEnabled = "box_owner_edit_enabled"
|
||||||
SettingSecurityEnabled = "security_enabled"
|
SettingBoxOwnerRefreshEnabled = "box_owner_refresh_enabled"
|
||||||
SettingSecurityIPWhitelist = "security_ip_whitelist"
|
SettingBoxOwnerMaxRefreshCount = "box_owner_max_refresh_count"
|
||||||
SettingSecurityAdminIPWhitelist = "security_admin_ip_whitelist"
|
SettingBoxOwnerMaxRefreshAmount = "box_owner_max_refresh_amount_seconds"
|
||||||
SettingTrustedProxyCIDRs = "trusted_proxy_cidrs"
|
SettingBoxOwnerMaxTotalExpiry = "box_owner_max_total_expiry_seconds"
|
||||||
SettingSecurityLoginWindowSecs = "security_login_window_seconds"
|
SettingBoxOwnerPasswordEdit = "box_owner_password_edit_enabled"
|
||||||
SettingSecurityLoginMaxAttempts = "security_login_max_attempts"
|
|
||||||
SettingSecurityBanSeconds = "security_ban_seconds"
|
|
||||||
SettingSecurityScanWindowSecs = "security_scan_window_seconds"
|
|
||||||
SettingSecurityScanMaxAttempts = "security_scan_max_attempts"
|
|
||||||
SettingSecurityUploadWindowSecs = "security_upload_window_seconds"
|
|
||||||
SettingSecurityUploadMaxRequests = "security_upload_max_requests"
|
|
||||||
SettingSecurityUploadMaxGB = "security_upload_max_gb"
|
|
||||||
SettingExpiredCleanupIntervalSecs = "expired_cleanup_interval_seconds"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type SettingType string
|
type SettingType string
|
||||||
@@ -59,7 +51,6 @@ const (
|
|||||||
SettingTypeInt64 SettingType = "int64"
|
SettingTypeInt64 SettingType = "int64"
|
||||||
SettingTypeInt SettingType = "int"
|
SettingTypeInt SettingType = "int"
|
||||||
SettingTypeText SettingType = "text"
|
SettingTypeText SettingType = "text"
|
||||||
SettingTypeSizeGB SettingType = "size_gb"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type SettingDefinition struct {
|
type SettingDefinition struct {
|
||||||
@@ -109,22 +100,13 @@ type Config struct {
|
|||||||
BoxPollIntervalMS int
|
BoxPollIntervalMS int
|
||||||
ThumbnailBatchSize int
|
ThumbnailBatchSize int
|
||||||
ThumbnailIntervalSeconds int
|
ThumbnailIntervalSeconds int
|
||||||
ActivityRetentionSeconds int64
|
BoxOwnerEditEnabled bool
|
||||||
SecurityEnabled bool
|
BoxOwnerRefreshEnabled bool
|
||||||
SecurityIPWhitelist string
|
BoxOwnerMaxRefreshCount int
|
||||||
SecurityAdminIPWhitelist string
|
BoxOwnerMaxRefreshAmountSeconds int64
|
||||||
TrustedProxyCIDRs string
|
BoxOwnerMaxTotalExpirySeconds int64
|
||||||
SecurityLoginWindowSeconds int64
|
BoxOwnerPasswordEditEnabled bool
|
||||||
SecurityLoginMaxAttempts int
|
|
||||||
SecurityBanSeconds int64
|
|
||||||
SecurityScanWindowSeconds int64
|
|
||||||
SecurityScanMaxAttempts int
|
|
||||||
SecurityUploadWindowSeconds int64
|
|
||||||
SecurityUploadMaxRequests int
|
|
||||||
SecurityUploadMaxBytes int64
|
|
||||||
ExpiredCleanupIntervalSeconds int64
|
|
||||||
|
|
||||||
sources map[string]Source
|
sources map[string]Source
|
||||||
values map[string]string
|
values map[string]string
|
||||||
defaults map[string]string
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,76 +0,0 @@
|
|||||||
package config
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"sort"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
const AdminSettingsOverrideFilename = "admin_settings_overrides.json"
|
|
||||||
|
|
||||||
type adminSettingsOverrideFile struct {
|
|
||||||
Format string `json:"format"`
|
|
||||||
SavedAt string `json:"saved_at"`
|
|
||||||
Overrides map[string]string `json:"overrides"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func ReadAdminSettingsOverrides(path string) (map[string]string, error) {
|
|
||||||
data, err := os.ReadFile(path)
|
|
||||||
if err != nil {
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
return map[string]string{}, nil
|
|
||||||
}
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
var payload adminSettingsOverrideFile
|
|
||||||
if err := json.Unmarshal(data, &payload); err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
if payload.Overrides == nil {
|
|
||||||
return map[string]string{}, nil
|
|
||||||
}
|
|
||||||
normalized := make(map[string]string, len(payload.Overrides))
|
|
||||||
for key, value := range payload.Overrides {
|
|
||||||
nextKey, nextValue, err := NormalizeOverrideInput(key, value)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
normalized[nextKey] = nextValue
|
|
||||||
}
|
|
||||||
return normalized, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func WriteAdminSettingsOverrides(path string, overrides map[string]string) error {
|
|
||||||
if overrides == nil {
|
|
||||||
overrides = map[string]string{}
|
|
||||||
}
|
|
||||||
|
|
||||||
keys := make([]string, 0, len(overrides))
|
|
||||||
for key := range overrides {
|
|
||||||
keys = append(keys, key)
|
|
||||||
}
|
|
||||||
sort.Strings(keys)
|
|
||||||
|
|
||||||
normalized := make(map[string]string, len(overrides))
|
|
||||||
for _, key := range keys {
|
|
||||||
normalized[key] = overrides[key]
|
|
||||||
}
|
|
||||||
|
|
||||||
payload := adminSettingsOverrideFile{
|
|
||||||
Format: "warpbox.admin.settings.overrides.v1",
|
|
||||||
SavedAt: time.Now().UTC().Format(time.RFC3339),
|
|
||||||
Overrides: normalized,
|
|
||||||
}
|
|
||||||
data, err := json.MarshalIndent(payload, "", " ")
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
if err := os.MkdirAll(filepath.Dir(path), 0755); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return os.WriteFile(path, data, 0644)
|
|
||||||
}
|
|
||||||
@@ -3,9 +3,6 @@ package config
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
|
||||||
|
|
||||||
"warpbox/lib/security"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
func (cfg *Config) ApplyOverrides(overrides map[string]string) error {
|
func (cfg *Config) ApplyOverrides(overrides map[string]string) error {
|
||||||
@@ -29,11 +26,6 @@ func (cfg *Config) ApplyOverride(key string, value string) error {
|
|||||||
return fmt.Errorf("setting %q cannot be changed from the admin UI", key)
|
return fmt.Errorf("setting %q cannot be changed from the admin UI", key)
|
||||||
}
|
}
|
||||||
|
|
||||||
value = strings.TrimSpace(value)
|
|
||||||
if err := validateSecurityTextSetting(key, value); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
switch def.Type {
|
switch def.Type {
|
||||||
case SettingTypeBool:
|
case SettingTypeBool:
|
||||||
parsed, err := parseBool(value)
|
parsed, err := parseBool(value)
|
||||||
@@ -47,40 +39,17 @@ func (cfg *Config) ApplyOverride(key string, value string) error {
|
|||||||
return fmt.Errorf("%s: %w", key, err)
|
return fmt.Errorf("%s: %w", key, err)
|
||||||
}
|
}
|
||||||
cfg.assignInt64(key, parsed, SourceDB)
|
cfg.assignInt64(key, parsed, SourceDB)
|
||||||
case SettingTypeSizeGB:
|
|
||||||
parsed, err := parseGigabytes(value, float64(def.Minimum))
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("%s: %w", key, err)
|
|
||||||
}
|
|
||||||
cfg.assignInt64(key, parsed, SourceDB)
|
|
||||||
case SettingTypeInt:
|
case SettingTypeInt:
|
||||||
parsed64, err := parseInt64(value, def.Minimum)
|
parsed64, err := parseInt64(value, def.Minimum)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return fmt.Errorf("%s: %w", key, err)
|
return fmt.Errorf("%s: %w", key, err)
|
||||||
}
|
}
|
||||||
cfg.assignInt(key, int(parsed64), SourceDB)
|
cfg.assignInt(key, int(parsed64), SourceDB)
|
||||||
case SettingTypeText:
|
|
||||||
cfg.assignText(key, value, SourceDB)
|
|
||||||
default:
|
default:
|
||||||
return fmt.Errorf("setting %q is not runtime editable", key)
|
return fmt.Errorf("setting %q is not runtime editable", key)
|
||||||
}
|
}
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func validateSecurityTextSetting(key string, value string) error {
|
|
||||||
switch key {
|
|
||||||
case SettingSecurityIPWhitelist, SettingSecurityAdminIPWhitelist:
|
|
||||||
if _, err := security.ParseIPMatchers(value, true); err != nil {
|
|
||||||
return fmt.Errorf("%s: %w", key, err)
|
|
||||||
}
|
|
||||||
case SettingTrustedProxyCIDRs:
|
|
||||||
if _, err := security.ParseCIDRList(value); err != nil {
|
|
||||||
return fmt.Errorf("%s: %w", key, err)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cfg *Config) assignBool(key string, value bool, source Source) {
|
func (cfg *Config) assignBool(key string, value bool, source Source) {
|
||||||
switch key {
|
switch key {
|
||||||
case SettingGuestUploadsEnabled:
|
case SettingGuestUploadsEnabled:
|
||||||
@@ -95,8 +64,12 @@ func (cfg *Config) assignBool(key string, value bool, source Source) {
|
|||||||
cfg.RenewOnAccessEnabled = value
|
cfg.RenewOnAccessEnabled = value
|
||||||
case SettingRenewOnDownloadEnabled:
|
case SettingRenewOnDownloadEnabled:
|
||||||
cfg.RenewOnDownloadEnabled = value
|
cfg.RenewOnDownloadEnabled = value
|
||||||
case SettingSecurityEnabled:
|
case SettingBoxOwnerEditEnabled:
|
||||||
cfg.SecurityEnabled = value
|
cfg.BoxOwnerEditEnabled = value
|
||||||
|
case SettingBoxOwnerRefreshEnabled:
|
||||||
|
cfg.BoxOwnerRefreshEnabled = value
|
||||||
|
case SettingBoxOwnerPasswordEdit:
|
||||||
|
cfg.BoxOwnerPasswordEditEnabled = value
|
||||||
}
|
}
|
||||||
cfg.setValue(key, formatBool(value), source)
|
cfg.setValue(key, formatBool(value), source)
|
||||||
}
|
}
|
||||||
@@ -109,34 +82,16 @@ func (cfg *Config) assignInt64(key string, value int64, source Source) {
|
|||||||
cfg.MaxGuestExpirySeconds = value
|
cfg.MaxGuestExpirySeconds = value
|
||||||
case SettingOneTimeDownloadExpirySecs:
|
case SettingOneTimeDownloadExpirySecs:
|
||||||
cfg.OneTimeDownloadExpirySeconds = value
|
cfg.OneTimeDownloadExpirySeconds = value
|
||||||
case SettingGlobalMaxFileSizeBytes:
|
|
||||||
cfg.GlobalMaxFileSizeBytes = value
|
|
||||||
case SettingGlobalMaxBoxSizeBytes:
|
|
||||||
cfg.GlobalMaxBoxSizeBytes = value
|
|
||||||
case SettingDefaultUserMaxFileBytes:
|
case SettingDefaultUserMaxFileBytes:
|
||||||
cfg.DefaultUserMaxFileSizeBytes = value
|
cfg.DefaultUserMaxFileSizeBytes = value
|
||||||
case SettingDefaultUserMaxBoxBytes:
|
case SettingDefaultUserMaxBoxBytes:
|
||||||
cfg.DefaultUserMaxBoxSizeBytes = value
|
cfg.DefaultUserMaxBoxSizeBytes = value
|
||||||
case SettingSessionTTLSeconds:
|
case SettingSessionTTLSeconds:
|
||||||
cfg.SessionTTLSeconds = value
|
cfg.SessionTTLSeconds = value
|
||||||
case SettingActivityRetentionSeconds:
|
case SettingBoxOwnerMaxRefreshAmount:
|
||||||
cfg.ActivityRetentionSeconds = value
|
cfg.BoxOwnerMaxRefreshAmountSeconds = value
|
||||||
case SettingSecurityLoginWindowSecs:
|
case SettingBoxOwnerMaxTotalExpiry:
|
||||||
cfg.SecurityLoginWindowSeconds = value
|
cfg.BoxOwnerMaxTotalExpirySeconds = value
|
||||||
case SettingSecurityBanSeconds:
|
|
||||||
cfg.SecurityBanSeconds = value
|
|
||||||
case SettingSecurityScanWindowSecs:
|
|
||||||
cfg.SecurityScanWindowSeconds = value
|
|
||||||
case SettingSecurityUploadWindowSecs:
|
|
||||||
cfg.SecurityUploadWindowSeconds = value
|
|
||||||
case SettingSecurityUploadMaxGB:
|
|
||||||
cfg.SecurityUploadMaxBytes = value
|
|
||||||
case SettingExpiredCleanupIntervalSecs:
|
|
||||||
cfg.ExpiredCleanupIntervalSeconds = value
|
|
||||||
}
|
|
||||||
if key == SettingGlobalMaxFileSizeBytes || key == SettingGlobalMaxBoxSizeBytes || key == SettingDefaultUserMaxFileBytes || key == SettingDefaultUserMaxBoxBytes || key == SettingSecurityUploadMaxGB {
|
|
||||||
cfg.setValue(key, formatGigabytesFromBytes(value), source)
|
|
||||||
return
|
|
||||||
}
|
}
|
||||||
cfg.setValue(key, strconv.FormatInt(value, 10), source)
|
cfg.setValue(key, strconv.FormatInt(value, 10), source)
|
||||||
}
|
}
|
||||||
@@ -149,28 +104,12 @@ func (cfg *Config) assignInt(key string, value int, source Source) {
|
|||||||
cfg.ThumbnailBatchSize = value
|
cfg.ThumbnailBatchSize = value
|
||||||
case SettingThumbnailIntervalSeconds:
|
case SettingThumbnailIntervalSeconds:
|
||||||
cfg.ThumbnailIntervalSeconds = value
|
cfg.ThumbnailIntervalSeconds = value
|
||||||
case SettingSecurityLoginMaxAttempts:
|
case SettingBoxOwnerMaxRefreshCount:
|
||||||
cfg.SecurityLoginMaxAttempts = value
|
cfg.BoxOwnerMaxRefreshCount = value
|
||||||
case SettingSecurityScanMaxAttempts:
|
|
||||||
cfg.SecurityScanMaxAttempts = value
|
|
||||||
case SettingSecurityUploadMaxRequests:
|
|
||||||
cfg.SecurityUploadMaxRequests = value
|
|
||||||
}
|
}
|
||||||
cfg.setValue(key, strconv.Itoa(value), source)
|
cfg.setValue(key, strconv.Itoa(value), source)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cfg *Config) assignText(key string, value string, source Source) {
|
|
||||||
switch key {
|
|
||||||
case SettingSecurityIPWhitelist:
|
|
||||||
cfg.SecurityIPWhitelist = value
|
|
||||||
case SettingSecurityAdminIPWhitelist:
|
|
||||||
cfg.SecurityAdminIPWhitelist = value
|
|
||||||
case SettingTrustedProxyCIDRs:
|
|
||||||
cfg.TrustedProxyCIDRs = value
|
|
||||||
}
|
|
||||||
cfg.setValue(key, value, source)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (cfg *Config) setValue(key string, value string, source Source) {
|
func (cfg *Config) setValue(key string, value string, source Source) {
|
||||||
if key == "" {
|
if key == "" {
|
||||||
return
|
return
|
||||||
@@ -186,10 +125,3 @@ func (cfg *Config) sourceFor(key string) Source {
|
|||||||
}
|
}
|
||||||
return source
|
return source
|
||||||
}
|
}
|
||||||
|
|
||||||
func (cfg *Config) DefaultValue(key string) string {
|
|
||||||
if cfg.defaults == nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
return cfg.defaults[key]
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -2,7 +2,6 @@ package config
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"math"
|
|
||||||
"strconv"
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
@@ -40,46 +39,6 @@ func parseInt(value string, min int) (int, error) {
|
|||||||
return int(parsed64), nil
|
return int(parsed64), nil
|
||||||
}
|
}
|
||||||
|
|
||||||
const bytesPerGigabyte = 1024 * 1024 * 1024
|
|
||||||
|
|
||||||
func parseGigabytes(value string, min float64) (int64, error) {
|
|
||||||
raw := strings.TrimSpace(value)
|
|
||||||
lower := strings.ToLower(raw)
|
|
||||||
if strings.HasSuffix(lower, "gb") {
|
|
||||||
raw = strings.TrimSpace(raw[:len(raw)-2])
|
|
||||||
}
|
|
||||||
parsed, err := strconv.ParseFloat(raw, 64)
|
|
||||||
if err != nil {
|
|
||||||
return 0, fmt.Errorf("must be a number of GB")
|
|
||||||
}
|
|
||||||
if parsed < min {
|
|
||||||
return 0, fmt.Errorf("must be at least %s", trimTrailingZeros(min))
|
|
||||||
}
|
|
||||||
bytes := parsed * bytesPerGigabyte
|
|
||||||
if bytes > math.MaxInt64 {
|
|
||||||
return 0, fmt.Errorf("is too large")
|
|
||||||
}
|
|
||||||
return int64(math.Round(bytes)), nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func formatGigabytesFromBytes(bytes int64) string {
|
|
||||||
if bytes <= 0 {
|
|
||||||
return "0"
|
|
||||||
}
|
|
||||||
value := float64(bytes) / bytesPerGigabyte
|
|
||||||
return trimTrailingZeros(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
func trimTrailingZeros(value float64) string {
|
|
||||||
text := strconv.FormatFloat(value, 'f', 3, 64)
|
|
||||||
text = strings.TrimRight(text, "0")
|
|
||||||
text = strings.TrimRight(text, ".")
|
|
||||||
if text == "" {
|
|
||||||
return "0"
|
|
||||||
}
|
|
||||||
return text
|
|
||||||
}
|
|
||||||
|
|
||||||
func formatBool(value bool) string {
|
func formatBool(value bool) string {
|
||||||
if value {
|
if value {
|
||||||
return "true"
|
return "true"
|
||||||
|
|||||||
247
lib/metastore/alerts.go
Normal file
247
lib/metastore/alerts.go
Normal file
@@ -0,0 +1,247 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/dgraph-io/badger/v4"
|
||||||
|
|
||||||
|
"warpbox/lib/helpers"
|
||||||
|
)
|
||||||
|
|
||||||
|
const (
|
||||||
|
AlertSeverityLow = "low"
|
||||||
|
AlertSeverityMedium = "medium"
|
||||||
|
AlertSeverityHigh = "high"
|
||||||
|
|
||||||
|
AlertStatusOpen = "open"
|
||||||
|
AlertStatusAcknowledged = "acknowledged"
|
||||||
|
AlertStatusClosed = "closed"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (store *Store) CreateAlert(input AlertInput) (Alert, error) {
|
||||||
|
alert, err := normalizeAlertInput(input)
|
||||||
|
if err != nil {
|
||||||
|
return Alert{}, err
|
||||||
|
}
|
||||||
|
id, err := helpers.RandomHexID(16)
|
||||||
|
if err != nil {
|
||||||
|
return Alert{}, err
|
||||||
|
}
|
||||||
|
now := time.Now().UTC()
|
||||||
|
alert.ID = id
|
||||||
|
alert.Status = AlertStatusOpen
|
||||||
|
alert.CreatedAt = now
|
||||||
|
alert.UpdatedAt = now
|
||||||
|
|
||||||
|
err = store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
return putJSON(txn, alertKey(alert.ID), alert)
|
||||||
|
})
|
||||||
|
return alert, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) ListAlerts(filters AlertFilters) ([]Alert, error) {
|
||||||
|
alerts := []Alert{}
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
opts := badger.DefaultIteratorOptions
|
||||||
|
opts.Prefix = []byte("alert/")
|
||||||
|
it := txn.NewIterator(opts)
|
||||||
|
defer it.Close()
|
||||||
|
|
||||||
|
for it.Rewind(); it.Valid(); it.Next() {
|
||||||
|
var alert Alert
|
||||||
|
if err := it.Item().Value(func(data []byte) error {
|
||||||
|
return json.Unmarshal(data, &alert)
|
||||||
|
}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if alertMatchesFilters(alert, filters) {
|
||||||
|
alerts = append(alerts, alert)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
sortAlerts(alerts, filters.Sort)
|
||||||
|
return alerts, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetAlert(id string) (Alert, bool, error) {
|
||||||
|
id = strings.TrimSpace(id)
|
||||||
|
if id == "" {
|
||||||
|
return Alert{}, false, nil
|
||||||
|
}
|
||||||
|
var alert Alert
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
return getJSON(txn, alertKey(id), &alert)
|
||||||
|
})
|
||||||
|
if errors.Is(err, ErrNotFound) {
|
||||||
|
return Alert{}, false, nil
|
||||||
|
}
|
||||||
|
return alert, err == nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) AcknowledgeAlert(id string) error {
|
||||||
|
return store.updateAlertStatus(id, AlertStatusAcknowledged)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) CloseAlert(id string) error {
|
||||||
|
return store.updateAlertStatus(id, AlertStatusClosed)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) updateAlertStatus(id string, status string) error {
|
||||||
|
id = strings.TrimSpace(id)
|
||||||
|
if id == "" {
|
||||||
|
return fmt.Errorf("%w: alert id cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
status, err := normalizeAlertStatus(status)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
now := time.Now().UTC()
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
var alert Alert
|
||||||
|
if err := getJSON(txn, alertKey(id), &alert); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
alert.Status = status
|
||||||
|
alert.UpdatedAt = now
|
||||||
|
switch status {
|
||||||
|
case AlertStatusAcknowledged:
|
||||||
|
alert.AcknowledgedAt = &now
|
||||||
|
case AlertStatusClosed:
|
||||||
|
alert.ClosedAt = &now
|
||||||
|
}
|
||||||
|
return putJSON(txn, alertKey(id), alert)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func normalizeAlertInput(input AlertInput) (Alert, error) {
|
||||||
|
title := strings.TrimSpace(input.Title)
|
||||||
|
description := strings.TrimSpace(input.Description)
|
||||||
|
code := strings.TrimSpace(input.Code)
|
||||||
|
trace := strings.TrimSpace(input.Trace)
|
||||||
|
severity, err := normalizeAlertSeverity(input.Severity)
|
||||||
|
if err != nil {
|
||||||
|
return Alert{}, err
|
||||||
|
}
|
||||||
|
if title == "" {
|
||||||
|
return Alert{}, fmt.Errorf("%w: alert title cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
if code == "" {
|
||||||
|
return Alert{}, fmt.Errorf("%w: alert code cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
if trace == "" {
|
||||||
|
return Alert{}, fmt.Errorf("%w: alert trace cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
metadata := input.Metadata
|
||||||
|
if len(metadata) == 0 {
|
||||||
|
metadata = json.RawMessage(`{}`)
|
||||||
|
}
|
||||||
|
var object map[string]any
|
||||||
|
if err := json.Unmarshal(metadata, &object); err != nil {
|
||||||
|
return Alert{}, fmt.Errorf("%w: alert metadata must be a JSON object", ErrInvalid)
|
||||||
|
}
|
||||||
|
normalizedMetadata, err := json.Marshal(object)
|
||||||
|
if err != nil {
|
||||||
|
return Alert{}, err
|
||||||
|
}
|
||||||
|
return Alert{
|
||||||
|
Title: title,
|
||||||
|
Description: description,
|
||||||
|
Severity: severity,
|
||||||
|
Code: code,
|
||||||
|
Trace: trace,
|
||||||
|
Metadata: normalizedMetadata,
|
||||||
|
CreatedBy: strings.TrimSpace(input.CreatedBy),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func normalizeAlertSeverity(value string) (string, error) {
|
||||||
|
switch strings.ToLower(strings.TrimSpace(value)) {
|
||||||
|
case AlertSeverityLow, AlertSeverityMedium, AlertSeverityHigh:
|
||||||
|
return strings.ToLower(strings.TrimSpace(value)), nil
|
||||||
|
default:
|
||||||
|
return "", fmt.Errorf("%w: invalid alert severity", ErrInvalid)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func normalizeAlertStatus(value string) (string, error) {
|
||||||
|
switch strings.ToLower(strings.TrimSpace(value)) {
|
||||||
|
case AlertStatusOpen, AlertStatusAcknowledged, AlertStatusClosed:
|
||||||
|
return strings.ToLower(strings.TrimSpace(value)), nil
|
||||||
|
default:
|
||||||
|
return "", fmt.Errorf("%w: invalid alert status", ErrInvalid)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func alertMatchesFilters(alert Alert, filters AlertFilters) bool {
|
||||||
|
query := strings.ToLower(strings.TrimSpace(filters.Query))
|
||||||
|
if query != "" {
|
||||||
|
haystack := strings.ToLower(strings.Join([]string{alert.Title, alert.Description, alert.Code, alert.Trace}, " "))
|
||||||
|
if !strings.Contains(haystack, query) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if severity := strings.ToLower(strings.TrimSpace(filters.Severity)); severity != "" && severity != "all" && alert.Severity != severity {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if status := strings.ToLower(strings.TrimSpace(filters.Status)); status != "" && status != "all" && alert.Status != status {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
if group := strings.ToLower(strings.TrimSpace(filters.Group)); group != "" && group != "all" && alertGroup(alert.Trace) != group {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func sortAlerts(alerts []Alert, sortKey string) {
|
||||||
|
switch strings.ToLower(strings.TrimSpace(sortKey)) {
|
||||||
|
case "oldest":
|
||||||
|
sort.Slice(alerts, func(i int, j int) bool { return alerts[i].CreatedAt.Before(alerts[j].CreatedAt) })
|
||||||
|
case "severity":
|
||||||
|
sort.Slice(alerts, func(i int, j int) bool {
|
||||||
|
left := alertSeverityRank(alerts[i].Severity)
|
||||||
|
right := alertSeverityRank(alerts[j].Severity)
|
||||||
|
if left == right {
|
||||||
|
return alerts[i].CreatedAt.After(alerts[j].CreatedAt)
|
||||||
|
}
|
||||||
|
return left > right
|
||||||
|
})
|
||||||
|
default:
|
||||||
|
sort.Slice(alerts, func(i int, j int) bool { return alerts[i].CreatedAt.After(alerts[j].CreatedAt) })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func alertSeverityRank(severity string) int {
|
||||||
|
switch severity {
|
||||||
|
case AlertSeverityHigh:
|
||||||
|
return 3
|
||||||
|
case AlertSeverityMedium:
|
||||||
|
return 2
|
||||||
|
default:
|
||||||
|
return 1
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func alertGroup(trace string) string {
|
||||||
|
trace = strings.TrimSpace(trace)
|
||||||
|
if trace == "" {
|
||||||
|
return "system"
|
||||||
|
}
|
||||||
|
before, _, found := strings.Cut(trace, ".")
|
||||||
|
if !found || before == "" {
|
||||||
|
return "system"
|
||||||
|
}
|
||||||
|
return strings.ToLower(before)
|
||||||
|
}
|
||||||
|
|
||||||
|
func alertKey(id string) []byte {
|
||||||
|
return []byte("alert/" + strings.TrimSpace(id))
|
||||||
|
}
|
||||||
89
lib/metastore/alerts_test.go
Normal file
89
lib/metastore/alerts_test.go
Normal file
@@ -0,0 +1,89 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"testing"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestAlertCreateListFilterLifecycle(t *testing.T) {
|
||||||
|
store, err := Open(t.TempDir())
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Open returned error: %v", err)
|
||||||
|
}
|
||||||
|
defer store.Close()
|
||||||
|
|
||||||
|
alert, err := store.CreateAlert(AlertInput{
|
||||||
|
Title: "Thumbnail failed",
|
||||||
|
Description: "Could not generate preview.",
|
||||||
|
Severity: AlertSeverityMedium,
|
||||||
|
Code: "601",
|
||||||
|
Trace: "thumbnail.generate.failed",
|
||||||
|
Metadata: json.RawMessage(`{"box":"box-1","file":"photo.jpg"}`),
|
||||||
|
CreatedBy: "system",
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateAlert returned error: %v", err)
|
||||||
|
}
|
||||||
|
if alert.ID == "" || alert.Status != AlertStatusOpen {
|
||||||
|
t.Fatalf("unexpected alert: %#v", alert)
|
||||||
|
}
|
||||||
|
|
||||||
|
alerts, err := store.ListAlerts(AlertFilters{Severity: AlertSeverityMedium, Status: AlertStatusOpen})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("ListAlerts returned error: %v", err)
|
||||||
|
}
|
||||||
|
if len(alerts) != 1 || alerts[0].Trace != "thumbnail.generate.failed" {
|
||||||
|
t.Fatalf("unexpected filtered alerts: %#v", alerts)
|
||||||
|
}
|
||||||
|
|
||||||
|
if !json.Valid(alerts[0].Metadata) {
|
||||||
|
t.Fatalf("expected valid metadata JSON: %s", string(alerts[0].Metadata))
|
||||||
|
}
|
||||||
|
var metadata map[string]string
|
||||||
|
if err := json.Unmarshal(alerts[0].Metadata, &metadata); err != nil {
|
||||||
|
t.Fatalf("Unmarshal metadata returned error: %v", err)
|
||||||
|
}
|
||||||
|
if metadata["file"] != "photo.jpg" {
|
||||||
|
t.Fatalf("metadata did not survive round trip: %#v", metadata)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := store.AcknowledgeAlert(alert.ID); err != nil {
|
||||||
|
t.Fatalf("AcknowledgeAlert returned error: %v", err)
|
||||||
|
}
|
||||||
|
acknowledged, ok, err := store.GetAlert(alert.ID)
|
||||||
|
if err != nil || !ok {
|
||||||
|
t.Fatalf("GetAlert returned ok=%v err=%v", ok, err)
|
||||||
|
}
|
||||||
|
if acknowledged.Status != AlertStatusAcknowledged || acknowledged.AcknowledgedAt == nil {
|
||||||
|
t.Fatalf("expected acknowledged alert, got %#v", acknowledged)
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := store.CloseAlert(alert.ID); err != nil {
|
||||||
|
t.Fatalf("CloseAlert returned error: %v", err)
|
||||||
|
}
|
||||||
|
closed, ok, err := store.GetAlert(alert.ID)
|
||||||
|
if err != nil || !ok {
|
||||||
|
t.Fatalf("GetAlert returned ok=%v err=%v", ok, err)
|
||||||
|
}
|
||||||
|
if closed.Status != AlertStatusClosed || closed.ClosedAt == nil {
|
||||||
|
t.Fatalf("expected closed alert, got %#v", closed)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAlertRejectsInvalidMetadata(t *testing.T) {
|
||||||
|
store, err := Open(t.TempDir())
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Open returned error: %v", err)
|
||||||
|
}
|
||||||
|
defer store.Close()
|
||||||
|
|
||||||
|
if _, err := store.CreateAlert(AlertInput{
|
||||||
|
Title: "Bad alert",
|
||||||
|
Severity: AlertSeverityLow,
|
||||||
|
Code: "999",
|
||||||
|
Trace: "test.bad",
|
||||||
|
Metadata: json.RawMessage(`[]`),
|
||||||
|
}); err == nil {
|
||||||
|
t.Fatal("expected non-object metadata to be rejected")
|
||||||
|
}
|
||||||
|
}
|
||||||
71
lib/metastore/bootstrap.go
Normal file
71
lib/metastore/bootstrap.go
Normal file
@@ -0,0 +1,71 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"warpbox/lib/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
func BootstrapAdmin(cfg *config.Config, store *Store) (BootstrapResult, error) {
|
||||||
|
adminTag, err := store.EnsureAdminTag()
|
||||||
|
if err != nil {
|
||||||
|
return BootstrapResult{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
var adminUser *User
|
||||||
|
user, ok, err := store.GetUserByUsername(cfg.AdminUsername)
|
||||||
|
if err != nil {
|
||||||
|
return BootstrapResult{}, err
|
||||||
|
}
|
||||||
|
if ok {
|
||||||
|
if !hasString(user.TagIDs, adminTag.ID) {
|
||||||
|
user.TagIDs = append(user.TagIDs, adminTag.ID)
|
||||||
|
if err := store.UpdateUser(user); err != nil {
|
||||||
|
return BootstrapResult{}, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
adminUser = &user
|
||||||
|
} else if strings.TrimSpace(cfg.AdminPassword) != "" {
|
||||||
|
created, err := store.CreateUserWithPassword(cfg.AdminUsername, cfg.AdminEmail, cfg.AdminPassword, []string{adminTag.ID})
|
||||||
|
if err != nil {
|
||||||
|
return BootstrapResult{}, err
|
||||||
|
}
|
||||||
|
adminUser = &created
|
||||||
|
}
|
||||||
|
|
||||||
|
hasAdminUser, err := store.HasAdminUser(adminTag.ID)
|
||||||
|
if err != nil {
|
||||||
|
return BootstrapResult{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
return BootstrapResult{
|
||||||
|
AdminTag: adminTag,
|
||||||
|
AdminUser: adminUser,
|
||||||
|
AdminLoginEnabled: cfg.AdminLoginEnabled(hasAdminUser),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) HasAdminUser(adminTagID string) (bool, error) {
|
||||||
|
users, err := store.ListUsers()
|
||||||
|
if err != nil {
|
||||||
|
return false, err
|
||||||
|
}
|
||||||
|
for _, user := range users {
|
||||||
|
if user.Disabled {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if hasString(user.TagIDs, adminTagID) {
|
||||||
|
return true, nil
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func hasString(values []string, target string) bool {
|
||||||
|
for _, value := range values {
|
||||||
|
if value == target {
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false
|
||||||
|
}
|
||||||
188
lib/metastore/boxes.go
Normal file
188
lib/metastore/boxes.go
Normal file
@@ -0,0 +1,188 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/dgraph-io/badger/v4"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (store *Store) UpsertBoxRecord(record BoxRecord) error {
|
||||||
|
record.ID = strings.TrimSpace(record.ID)
|
||||||
|
if record.ID == "" {
|
||||||
|
return errors.New("box id cannot be empty")
|
||||||
|
}
|
||||||
|
record.OwnerID = strings.TrimSpace(record.OwnerID)
|
||||||
|
record.OwnerUsername = strings.TrimSpace(record.OwnerUsername)
|
||||||
|
record.FileNames = uniqueStrings(record.FileNames)
|
||||||
|
record.UpdatedAt = time.Now().UTC()
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
return putJSON(txn, boxRecordKey(record.ID), record)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetBoxRecord(id string) (BoxRecord, bool, error) {
|
||||||
|
var record BoxRecord
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
return getJSON(txn, boxRecordKey(id), &record)
|
||||||
|
})
|
||||||
|
if errors.Is(err, ErrNotFound) {
|
||||||
|
return BoxRecord{}, false, nil
|
||||||
|
}
|
||||||
|
return record, err == nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) DeleteBoxRecord(id string) error {
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
err := txn.Delete(boxRecordKey(id))
|
||||||
|
if errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) ListBoxRecords(filters BoxFilters, page BoxPageRequest) (BoxRecordPage, error) {
|
||||||
|
if page.Page < 1 {
|
||||||
|
page.Page = 1
|
||||||
|
}
|
||||||
|
switch page.PageSize {
|
||||||
|
case 25, 50, 100:
|
||||||
|
default:
|
||||||
|
page.PageSize = 25
|
||||||
|
}
|
||||||
|
|
||||||
|
rows := []BoxRecord{}
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
opts := badger.DefaultIteratorOptions
|
||||||
|
opts.Prefix = []byte("box_record/")
|
||||||
|
it := txn.NewIterator(opts)
|
||||||
|
defer it.Close()
|
||||||
|
|
||||||
|
for it.Rewind(); it.Valid(); it.Next() {
|
||||||
|
var record BoxRecord
|
||||||
|
if err := it.Item().Value(func(data []byte) error {
|
||||||
|
return json.Unmarshal(data, &record)
|
||||||
|
}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if boxRecordMatches(record, filters) {
|
||||||
|
rows = append(rows, record)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
return BoxRecordPage{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
sortBoxRecords(rows, filters.Sort)
|
||||||
|
total := len(rows)
|
||||||
|
start := (page.Page - 1) * page.PageSize
|
||||||
|
if start > total {
|
||||||
|
start = total
|
||||||
|
}
|
||||||
|
end := start + page.PageSize
|
||||||
|
if end > total {
|
||||||
|
end = total
|
||||||
|
}
|
||||||
|
totalPages := 1
|
||||||
|
if total > 0 {
|
||||||
|
totalPages = (total + page.PageSize - 1) / page.PageSize
|
||||||
|
}
|
||||||
|
return BoxRecordPage{
|
||||||
|
Rows: rows[start:end],
|
||||||
|
Page: page.Page,
|
||||||
|
PageSize: page.PageSize,
|
||||||
|
Total: total,
|
||||||
|
HasPrev: page.Page > 1,
|
||||||
|
HasNext: end < total,
|
||||||
|
PrevPage: maxInt(page.Page-1, 1),
|
||||||
|
NextPage: page.Page + 1,
|
||||||
|
TotalPages: totalPages,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxRecordMatches(record BoxRecord, filters BoxFilters) bool {
|
||||||
|
query := strings.ToLower(strings.TrimSpace(filters.Query))
|
||||||
|
if query != "" {
|
||||||
|
haystack := strings.ToLower(record.ID + " " + record.OwnerUsername + " " + strings.Join(record.FileNames, " "))
|
||||||
|
if !strings.Contains(haystack, query) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
owner := strings.ToLower(strings.TrimSpace(filters.Owner))
|
||||||
|
if owner != "" && owner != "all" && strings.ToLower(record.OwnerUsername) != owner && strings.ToLower(record.OwnerID) != owner {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
status := strings.ToLower(strings.TrimSpace(filters.Status))
|
||||||
|
if status != "" && status != "all" && boxRecordStatus(record) != status {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
switch strings.ToLower(strings.TrimSpace(filters.Flag)) {
|
||||||
|
case "", "all":
|
||||||
|
return true
|
||||||
|
case "password":
|
||||||
|
return record.PasswordProtected
|
||||||
|
case "one-time":
|
||||||
|
return record.OneTimeDownload
|
||||||
|
case "zip-disabled":
|
||||||
|
return record.DisableZip
|
||||||
|
case "expired":
|
||||||
|
return boxRecordExpired(record)
|
||||||
|
case "refreshable":
|
||||||
|
return !record.OneTimeDownload && !boxRecordExpired(record)
|
||||||
|
default:
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func sortBoxRecords(rows []BoxRecord, sortKey string) {
|
||||||
|
switch strings.ToLower(strings.TrimSpace(sortKey)) {
|
||||||
|
case "oldest":
|
||||||
|
sort.Slice(rows, func(i int, j int) bool { return rows[i].CreatedAt.Before(rows[j].CreatedAt) })
|
||||||
|
case "largest":
|
||||||
|
sort.Slice(rows, func(i int, j int) bool { return rows[i].TotalSize > rows[j].TotalSize })
|
||||||
|
case "expires":
|
||||||
|
sort.Slice(rows, func(i int, j int) bool { return rows[i].ExpiresAt.Before(rows[j].ExpiresAt) })
|
||||||
|
case "expired":
|
||||||
|
sort.Slice(rows, func(i int, j int) bool {
|
||||||
|
left := boxRecordExpired(rows[i])
|
||||||
|
right := boxRecordExpired(rows[j])
|
||||||
|
if left == right {
|
||||||
|
return rows[i].CreatedAt.After(rows[j].CreatedAt)
|
||||||
|
}
|
||||||
|
return left
|
||||||
|
})
|
||||||
|
default:
|
||||||
|
sort.Slice(rows, func(i int, j int) bool { return rows[i].CreatedAt.After(rows[j].CreatedAt) })
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxRecordStatus(record BoxRecord) string {
|
||||||
|
if boxRecordExpired(record) {
|
||||||
|
return "expired"
|
||||||
|
}
|
||||||
|
if record.ExpiresAt.IsZero() {
|
||||||
|
return "pending"
|
||||||
|
}
|
||||||
|
return "active"
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxRecordExpired(record BoxRecord) bool {
|
||||||
|
return !record.ExpiresAt.IsZero() && time.Now().UTC().After(record.ExpiresAt)
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxRecordKey(id string) []byte {
|
||||||
|
return []byte("box_record/" + strings.TrimSpace(id))
|
||||||
|
}
|
||||||
|
|
||||||
|
func maxInt(a int, b int) int {
|
||||||
|
if a > b {
|
||||||
|
return a
|
||||||
|
}
|
||||||
|
return b
|
||||||
|
}
|
||||||
222
lib/metastore/metastore_test.go
Normal file
222
lib/metastore/metastore_test.go
Normal file
@@ -0,0 +1,222 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"warpbox/lib/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestOpenClose(t *testing.T) {
|
||||||
|
store, err := Open(t.TempDir())
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Open returned error: %v", err)
|
||||||
|
}
|
||||||
|
if err := store.Close(); err != nil {
|
||||||
|
t.Fatalf("Close returned error: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBootstrapAdminFromPassword(t *testing.T) {
|
||||||
|
clearMetastoreConfigEnv(t)
|
||||||
|
t.Setenv("WARPBOX_ADMIN_PASSWORD", "secret-pass")
|
||||||
|
t.Setenv("WARPBOX_ADMIN_EMAIL", "admin@example.test")
|
||||||
|
|
||||||
|
cfg, err := config.Load()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Load returned error: %v", err)
|
||||||
|
}
|
||||||
|
store := openTestStore(t)
|
||||||
|
|
||||||
|
result, err := BootstrapAdmin(cfg, store)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("BootstrapAdmin returned error: %v", err)
|
||||||
|
}
|
||||||
|
if !result.AdminLoginEnabled {
|
||||||
|
t.Fatal("expected admin login to be enabled")
|
||||||
|
}
|
||||||
|
if !result.AdminTag.Protected {
|
||||||
|
t.Fatal("expected admin tag to be protected")
|
||||||
|
}
|
||||||
|
if result.AdminUser == nil {
|
||||||
|
t.Fatal("expected bootstrap admin user")
|
||||||
|
}
|
||||||
|
if !hasString(result.AdminUser.TagIDs, result.AdminTag.ID) {
|
||||||
|
t.Fatal("expected bootstrap admin to have admin tag")
|
||||||
|
}
|
||||||
|
if !VerifyPassword(result.AdminUser.PasswordHash, "secret-pass") {
|
||||||
|
t.Fatal("expected bootstrap admin password to verify")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBootstrapAdminDisabledWithoutPassword(t *testing.T) {
|
||||||
|
clearMetastoreConfigEnv(t)
|
||||||
|
|
||||||
|
cfg, err := config.Load()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Load returned error: %v", err)
|
||||||
|
}
|
||||||
|
store := openTestStore(t)
|
||||||
|
|
||||||
|
result, err := BootstrapAdmin(cfg, store)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("BootstrapAdmin returned error: %v", err)
|
||||||
|
}
|
||||||
|
if result.AdminLoginEnabled {
|
||||||
|
t.Fatal("expected admin login to be disabled without password or existing admin")
|
||||||
|
}
|
||||||
|
if !result.AdminTag.Protected {
|
||||||
|
t.Fatal("expected admin tag to still be created")
|
||||||
|
}
|
||||||
|
users, err := store.ListUsers()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("ListUsers returned error: %v", err)
|
||||||
|
}
|
||||||
|
if len(users) != 0 {
|
||||||
|
t.Fatalf("expected no users, got %d", len(users))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDuplicateUsersAndTags(t *testing.T) {
|
||||||
|
store := openTestStore(t)
|
||||||
|
|
||||||
|
if _, err := store.CreateUserWithPassword("alex", "alex@example.test", "secret", nil); err != nil {
|
||||||
|
t.Fatalf("CreateUserWithPassword returned error: %v", err)
|
||||||
|
}
|
||||||
|
if _, err := store.CreateUserWithPassword("Alex", "other@example.test", "secret", nil); !errors.Is(err, ErrDuplicate) {
|
||||||
|
t.Fatalf("expected duplicate username error, got %v", err)
|
||||||
|
}
|
||||||
|
if _, err := store.CreateUserWithPassword("other", "alex@example.test", "secret", nil); !errors.Is(err, ErrDuplicate) {
|
||||||
|
t.Fatalf("expected duplicate email error, got %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
tag := Tag{Name: "staff"}
|
||||||
|
if err := store.CreateTag(&tag); err != nil {
|
||||||
|
t.Fatalf("CreateTag returned error: %v", err)
|
||||||
|
}
|
||||||
|
duplicate := Tag{Name: "Staff"}
|
||||||
|
if err := store.CreateTag(&duplicate); !errors.Is(err, ErrDuplicate) {
|
||||||
|
t.Fatalf("expected duplicate tag error, got %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestPermissionResolutionAndGlobalCaps(t *testing.T) {
|
||||||
|
clearMetastoreConfigEnv(t)
|
||||||
|
t.Setenv("WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES", "50")
|
||||||
|
t.Setenv("WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES", "100")
|
||||||
|
t.Setenv("WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES", "1000")
|
||||||
|
|
||||||
|
cfg, err := config.Load()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Load returned error: %v", err)
|
||||||
|
}
|
||||||
|
tagFileLimit := int64(80)
|
||||||
|
tagBoxLimit := int64(2000)
|
||||||
|
userFileLimit := int64(60)
|
||||||
|
user := User{MaxFileSizeBytes: &userFileLimit}
|
||||||
|
tags := []Tag{
|
||||||
|
{
|
||||||
|
Permissions: TagPermissions{
|
||||||
|
UploadAllowed: true,
|
||||||
|
AllowedExpirySeconds: []int64{3600, 600},
|
||||||
|
MaxFileSizeBytes: &tagFileLimit,
|
||||||
|
MaxBoxSizeBytes: &tagBoxLimit,
|
||||||
|
ZipDownloadAllowed: true,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
perms := ResolveUserPermissions(cfg, user, tags)
|
||||||
|
if !perms.UploadAllowed || !perms.ZipDownloadAllowed {
|
||||||
|
t.Fatal("expected tag booleans to grant permissions")
|
||||||
|
}
|
||||||
|
if perms.MaxFileSizeBytes != 80 {
|
||||||
|
t.Fatalf("expected tag limit to beat user/default limit, got %d", perms.MaxFileSizeBytes)
|
||||||
|
}
|
||||||
|
if perms.MaxBoxSizeBytes != 1000 {
|
||||||
|
t.Fatalf("expected global max box cap, got %d", perms.MaxBoxSizeBytes)
|
||||||
|
}
|
||||||
|
if len(perms.AllowedExpirySeconds) != 2 || perms.AllowedExpirySeconds[0] != 600 || perms.AllowedExpirySeconds[1] != 3600 {
|
||||||
|
t.Fatalf("unexpected expiry durations: %#v", perms.AllowedExpirySeconds)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSettingsStorageAndPrecedence(t *testing.T) {
|
||||||
|
clearMetastoreConfigEnv(t)
|
||||||
|
t.Setenv("WARPBOX_API_ENABLED", "true")
|
||||||
|
|
||||||
|
store := openTestStore(t)
|
||||||
|
if err := store.SetSetting(config.SettingAPIEnabled, "false"); err != nil {
|
||||||
|
t.Fatalf("SetSetting returned error: %v", err)
|
||||||
|
}
|
||||||
|
overrides, err := store.ListSettings()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("ListSettings returned error: %v", err)
|
||||||
|
}
|
||||||
|
cfg, err := config.Load()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Load returned error: %v", err)
|
||||||
|
}
|
||||||
|
if err := cfg.ApplyOverrides(overrides); err != nil {
|
||||||
|
t.Fatalf("ApplyOverrides returned error: %v", err)
|
||||||
|
}
|
||||||
|
if cfg.APIEnabled {
|
||||||
|
t.Fatal("expected stored DB override to beat env")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestSessionExpiry(t *testing.T) {
|
||||||
|
store := openTestStore(t)
|
||||||
|
session, err := store.CreateSession("user-id", time.Millisecond)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateSession returned error: %v", err)
|
||||||
|
}
|
||||||
|
time.Sleep(2 * time.Millisecond)
|
||||||
|
if _, ok, err := store.GetSession(session.Token); err != nil || ok {
|
||||||
|
t.Fatalf("expected expired session to be invalid, ok=%v err=%v", ok, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func openTestStore(t *testing.T) *Store {
|
||||||
|
t.Helper()
|
||||||
|
store, err := Open(t.TempDir())
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Open returned error: %v", err)
|
||||||
|
}
|
||||||
|
t.Cleanup(func() {
|
||||||
|
_ = store.Close()
|
||||||
|
})
|
||||||
|
return store
|
||||||
|
}
|
||||||
|
|
||||||
|
func clearMetastoreConfigEnv(t *testing.T) {
|
||||||
|
t.Helper()
|
||||||
|
for _, name := range []string{
|
||||||
|
"WARPBOX_DATA_DIR",
|
||||||
|
"WARPBOX_ADMIN_PASSWORD",
|
||||||
|
"WARPBOX_ADMIN_USERNAME",
|
||||||
|
"WARPBOX_ADMIN_EMAIL",
|
||||||
|
"WARPBOX_ADMIN_ENABLED",
|
||||||
|
"WARPBOX_ALLOW_ADMIN_SETTINGS_OVERRIDE",
|
||||||
|
"WARPBOX_ADMIN_COOKIE_SECURE",
|
||||||
|
"WARPBOX_GUEST_UPLOADS_ENABLED",
|
||||||
|
"WARPBOX_API_ENABLED",
|
||||||
|
"WARPBOX_ZIP_DOWNLOADS_ENABLED",
|
||||||
|
"WARPBOX_ONE_TIME_DOWNLOADS_ENABLED",
|
||||||
|
"WARPBOX_RENEW_ON_ACCESS_ENABLED",
|
||||||
|
"WARPBOX_RENEW_ON_DOWNLOAD_ENABLED",
|
||||||
|
"WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS",
|
||||||
|
"WARPBOX_MAX_GUEST_EXPIRY_SECONDS",
|
||||||
|
"WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES",
|
||||||
|
"WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES",
|
||||||
|
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES",
|
||||||
|
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES",
|
||||||
|
"WARPBOX_SESSION_TTL_SECONDS",
|
||||||
|
"WARPBOX_BOX_POLL_INTERVAL_MS",
|
||||||
|
"WARPBOX_THUMBNAIL_BATCH_SIZE",
|
||||||
|
"WARPBOX_THUMBNAIL_INTERVAL_SECONDS",
|
||||||
|
} {
|
||||||
|
t.Setenv(name, "")
|
||||||
|
}
|
||||||
|
}
|
||||||
154
lib/metastore/models.go
Normal file
154
lib/metastore/models.go
Normal file
@@ -0,0 +1,154 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
const AdminTagName = "admin"
|
||||||
|
|
||||||
|
type User struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Username string `json:"username"`
|
||||||
|
Email string `json:"email,omitempty"`
|
||||||
|
PasswordHash string `json:"password_hash"`
|
||||||
|
TagIDs []string `json:"tag_ids"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
Disabled bool `json:"disabled"`
|
||||||
|
MaxFileSizeBytes *int64 `json:"max_file_size_bytes,omitempty"`
|
||||||
|
MaxBoxSizeBytes *int64 `json:"max_box_size_bytes,omitempty"`
|
||||||
|
MaxExpirySeconds *int64 `json:"max_expiry_seconds,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Tag struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Name string `json:"name"`
|
||||||
|
Description string `json:"description,omitempty"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
Protected bool `json:"protected"`
|
||||||
|
Permissions TagPermissions `json:"permissions"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type TagPermissions struct {
|
||||||
|
UploadAllowed bool `json:"upload_allowed"`
|
||||||
|
AllowedExpirySeconds []int64 `json:"allowed_expiry_seconds,omitempty"`
|
||||||
|
MaxFileSizeBytes *int64 `json:"max_file_size_bytes,omitempty"`
|
||||||
|
MaxBoxSizeBytes *int64 `json:"max_box_size_bytes,omitempty"`
|
||||||
|
OneTimeDownloadAllowed bool `json:"one_time_download_allowed"`
|
||||||
|
ZipDownloadAllowed bool `json:"zip_download_allowed"`
|
||||||
|
RenewableAllowed bool `json:"renewable_allowed"`
|
||||||
|
RenewOnAccessSeconds int64 `json:"renew_on_access_seconds,omitempty"`
|
||||||
|
RenewOnDownloadSeconds int64 `json:"renew_on_download_seconds,omitempty"`
|
||||||
|
AdminAccess bool `json:"admin_access"`
|
||||||
|
AdminUsersManage bool `json:"admin_users_manage"`
|
||||||
|
AdminSettingsManage bool `json:"admin_settings_manage"`
|
||||||
|
AdminBoxesView bool `json:"admin_boxes_view"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type Session struct {
|
||||||
|
Token string `json:"token"`
|
||||||
|
CSRFToken string `json:"csrf_token"`
|
||||||
|
UserID string `json:"user_id"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
ExpiresAt time.Time `json:"expires_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type EffectivePermissions struct {
|
||||||
|
UploadAllowed bool
|
||||||
|
AllowedExpirySeconds []int64
|
||||||
|
MaxFileSizeBytes int64
|
||||||
|
MaxBoxSizeBytes int64
|
||||||
|
MaxExpirySeconds int64
|
||||||
|
OneTimeDownloadAllowed bool
|
||||||
|
ZipDownloadAllowed bool
|
||||||
|
RenewableAllowed bool
|
||||||
|
RenewOnAccessSeconds int64
|
||||||
|
RenewOnDownloadSeconds int64
|
||||||
|
AdminAccess bool
|
||||||
|
AdminUsersManage bool
|
||||||
|
AdminSettingsManage bool
|
||||||
|
AdminBoxesView bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type BootstrapResult struct {
|
||||||
|
AdminTag Tag
|
||||||
|
AdminUser *User
|
||||||
|
AdminLoginEnabled bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type Alert struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
Title string `json:"title"`
|
||||||
|
Description string `json:"description"`
|
||||||
|
Severity string `json:"severity"`
|
||||||
|
Status string `json:"status"`
|
||||||
|
Code string `json:"code"`
|
||||||
|
Trace string `json:"trace"`
|
||||||
|
Metadata json.RawMessage `json:"metadata,omitempty"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
AcknowledgedAt *time.Time `json:"acknowledged_at,omitempty"`
|
||||||
|
ClosedAt *time.Time `json:"closed_at,omitempty"`
|
||||||
|
CreatedBy string `json:"created_by"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type AlertInput struct {
|
||||||
|
Title string
|
||||||
|
Description string
|
||||||
|
Severity string
|
||||||
|
Code string
|
||||||
|
Trace string
|
||||||
|
Metadata json.RawMessage
|
||||||
|
CreatedBy string
|
||||||
|
}
|
||||||
|
|
||||||
|
type AlertFilters struct {
|
||||||
|
Query string
|
||||||
|
Severity string
|
||||||
|
Status string
|
||||||
|
Group string
|
||||||
|
Sort string
|
||||||
|
}
|
||||||
|
|
||||||
|
type BoxRecord struct {
|
||||||
|
ID string `json:"id"`
|
||||||
|
OwnerID string `json:"owner_id,omitempty"`
|
||||||
|
OwnerUsername string `json:"owner_username,omitempty"`
|
||||||
|
FileNames []string `json:"file_names,omitempty"`
|
||||||
|
FileCount int `json:"file_count"`
|
||||||
|
TotalSize int64 `json:"total_size"`
|
||||||
|
CreatedAt time.Time `json:"created_at"`
|
||||||
|
ExpiresAt time.Time `json:"expires_at"`
|
||||||
|
PasswordProtected bool `json:"password_protected"`
|
||||||
|
OneTimeDownload bool `json:"one_time_download"`
|
||||||
|
DisableZip bool `json:"disable_zip"`
|
||||||
|
RefreshCount int `json:"refresh_count"`
|
||||||
|
UpdatedAt time.Time `json:"updated_at"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type BoxFilters struct {
|
||||||
|
Query string
|
||||||
|
Owner string
|
||||||
|
Status string
|
||||||
|
Flag string
|
||||||
|
Sort string
|
||||||
|
}
|
||||||
|
|
||||||
|
type BoxPageRequest struct {
|
||||||
|
Page int
|
||||||
|
PageSize int
|
||||||
|
}
|
||||||
|
|
||||||
|
type BoxRecordPage struct {
|
||||||
|
Rows []BoxRecord
|
||||||
|
Page int
|
||||||
|
PageSize int
|
||||||
|
Total int
|
||||||
|
HasPrev bool
|
||||||
|
HasNext bool
|
||||||
|
PrevPage int
|
||||||
|
NextPage int
|
||||||
|
TotalPages int
|
||||||
|
}
|
||||||
141
lib/metastore/permissions.go
Normal file
141
lib/metastore/permissions.go
Normal file
@@ -0,0 +1,141 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"sort"
|
||||||
|
|
||||||
|
"warpbox/lib/config"
|
||||||
|
)
|
||||||
|
|
||||||
|
func ResolveUserPermissions(cfg *config.Config, user User, tags []Tag) EffectivePermissions {
|
||||||
|
perms := EffectivePermissions{
|
||||||
|
MaxFileSizeBytes: cfg.DefaultUserMaxFileSizeBytes,
|
||||||
|
MaxBoxSizeBytes: cfg.DefaultUserMaxBoxSizeBytes,
|
||||||
|
ZipDownloadAllowed: cfg.ZipDownloadsEnabled,
|
||||||
|
OneTimeDownloadAllowed: cfg.OneTimeDownloadsEnabled,
|
||||||
|
}
|
||||||
|
|
||||||
|
expirySet := make(map[int64]bool)
|
||||||
|
for _, tag := range tags {
|
||||||
|
tagPerms := tag.Permissions
|
||||||
|
perms.UploadAllowed = perms.UploadAllowed || tagPerms.UploadAllowed
|
||||||
|
perms.OneTimeDownloadAllowed = perms.OneTimeDownloadAllowed || tagPerms.OneTimeDownloadAllowed
|
||||||
|
perms.ZipDownloadAllowed = perms.ZipDownloadAllowed || tagPerms.ZipDownloadAllowed
|
||||||
|
perms.RenewableAllowed = perms.RenewableAllowed || tagPerms.RenewableAllowed
|
||||||
|
perms.AdminAccess = perms.AdminAccess || tagPerms.AdminAccess
|
||||||
|
perms.AdminUsersManage = perms.AdminUsersManage || tagPerms.AdminUsersManage
|
||||||
|
perms.AdminSettingsManage = perms.AdminSettingsManage || tagPerms.AdminSettingsManage
|
||||||
|
perms.AdminBoxesView = perms.AdminBoxesView || tagPerms.AdminBoxesView
|
||||||
|
perms.RenewOnAccessSeconds = maxInt64(perms.RenewOnAccessSeconds, tagPerms.RenewOnAccessSeconds)
|
||||||
|
perms.RenewOnDownloadSeconds = maxInt64(perms.RenewOnDownloadSeconds, tagPerms.RenewOnDownloadSeconds)
|
||||||
|
if tagPerms.MaxFileSizeBytes != nil {
|
||||||
|
perms.MaxFileSizeBytes = morePermissiveLimit(perms.MaxFileSizeBytes, *tagPerms.MaxFileSizeBytes)
|
||||||
|
}
|
||||||
|
if tagPerms.MaxBoxSizeBytes != nil {
|
||||||
|
perms.MaxBoxSizeBytes = morePermissiveLimit(perms.MaxBoxSizeBytes, *tagPerms.MaxBoxSizeBytes)
|
||||||
|
}
|
||||||
|
for _, seconds := range tagPerms.AllowedExpirySeconds {
|
||||||
|
if seconds >= 0 {
|
||||||
|
expirySet[seconds] = true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if user.MaxFileSizeBytes != nil {
|
||||||
|
perms.MaxFileSizeBytes = morePermissiveLimit(perms.MaxFileSizeBytes, *user.MaxFileSizeBytes)
|
||||||
|
}
|
||||||
|
if user.MaxBoxSizeBytes != nil {
|
||||||
|
perms.MaxBoxSizeBytes = morePermissiveLimit(perms.MaxBoxSizeBytes, *user.MaxBoxSizeBytes)
|
||||||
|
}
|
||||||
|
if user.MaxExpirySeconds != nil {
|
||||||
|
perms.MaxExpirySeconds = *user.MaxExpirySeconds
|
||||||
|
}
|
||||||
|
|
||||||
|
perms.MaxFileSizeBytes = capLimit(perms.MaxFileSizeBytes, cfg.GlobalMaxFileSizeBytes)
|
||||||
|
perms.MaxBoxSizeBytes = capLimit(perms.MaxBoxSizeBytes, cfg.GlobalMaxBoxSizeBytes)
|
||||||
|
perms.AllowedExpirySeconds = sortedExpirySet(expirySet)
|
||||||
|
if !cfg.ZipDownloadsEnabled {
|
||||||
|
perms.ZipDownloadAllowed = false
|
||||||
|
}
|
||||||
|
if !cfg.OneTimeDownloadsEnabled {
|
||||||
|
perms.OneTimeDownloadAllowed = false
|
||||||
|
}
|
||||||
|
return perms
|
||||||
|
}
|
||||||
|
|
||||||
|
func ResolveGuestPermissions(cfg *config.Config) EffectivePermissions {
|
||||||
|
return EffectivePermissions{
|
||||||
|
UploadAllowed: cfg.GuestUploadsEnabled,
|
||||||
|
AllowedExpirySeconds: guestExpirySeconds(cfg),
|
||||||
|
MaxFileSizeBytes: cfg.GlobalMaxFileSizeBytes,
|
||||||
|
MaxBoxSizeBytes: cfg.GlobalMaxBoxSizeBytes,
|
||||||
|
MaxExpirySeconds: cfg.MaxGuestExpirySeconds,
|
||||||
|
OneTimeDownloadAllowed: cfg.OneTimeDownloadsEnabled,
|
||||||
|
ZipDownloadAllowed: cfg.ZipDownloadsEnabled,
|
||||||
|
RenewableAllowed: cfg.RenewOnAccessEnabled || cfg.RenewOnDownloadEnabled,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func morePermissiveLimit(current int64, candidate int64) int64 {
|
||||||
|
if current == 0 || candidate == 0 {
|
||||||
|
return 0
|
||||||
|
}
|
||||||
|
if candidate > current {
|
||||||
|
return candidate
|
||||||
|
}
|
||||||
|
return current
|
||||||
|
}
|
||||||
|
|
||||||
|
func capLimit(value int64, globalMax int64) int64 {
|
||||||
|
if globalMax == 0 {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
if value == 0 || value > globalMax {
|
||||||
|
return globalMax
|
||||||
|
}
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
||||||
|
func sortedExpirySet(expirySet map[int64]bool) []int64 {
|
||||||
|
values := make([]int64, 0, len(expirySet))
|
||||||
|
for value := range expirySet {
|
||||||
|
values = append(values, value)
|
||||||
|
}
|
||||||
|
sort.Slice(values, func(i int, j int) bool {
|
||||||
|
return values[i] < values[j]
|
||||||
|
})
|
||||||
|
return values
|
||||||
|
}
|
||||||
|
|
||||||
|
func guestExpirySeconds(cfg *config.Config) []int64 {
|
||||||
|
values := []int64{}
|
||||||
|
if cfg.DefaultGuestExpirySeconds >= 0 {
|
||||||
|
values = append(values, cfg.DefaultGuestExpirySeconds)
|
||||||
|
}
|
||||||
|
if cfg.MaxGuestExpirySeconds > 0 && cfg.MaxGuestExpirySeconds != cfg.DefaultGuestExpirySeconds {
|
||||||
|
values = append(values, cfg.MaxGuestExpirySeconds)
|
||||||
|
}
|
||||||
|
return uniqueInt64s(values)
|
||||||
|
}
|
||||||
|
|
||||||
|
func uniqueInt64s(values []int64) []int64 {
|
||||||
|
seen := make(map[int64]bool, len(values))
|
||||||
|
out := make([]int64, 0, len(values))
|
||||||
|
for _, value := range values {
|
||||||
|
if seen[value] {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
seen[value] = true
|
||||||
|
out = append(out, value)
|
||||||
|
}
|
||||||
|
sort.Slice(out, func(i int, j int) bool {
|
||||||
|
return out[i] < out[j]
|
||||||
|
})
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func maxInt64(a int64, b int64) int64 {
|
||||||
|
if b > a {
|
||||||
|
return b
|
||||||
|
}
|
||||||
|
return a
|
||||||
|
}
|
||||||
79
lib/metastore/sessions.go
Normal file
79
lib/metastore/sessions.go
Normal file
@@ -0,0 +1,79 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/dgraph-io/badger/v4"
|
||||||
|
|
||||||
|
"warpbox/lib/helpers"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (store *Store) CreateSession(userID string, ttl time.Duration) (Session, error) {
|
||||||
|
userID = strings.TrimSpace(userID)
|
||||||
|
if userID == "" {
|
||||||
|
return Session{}, fmt.Errorf("%w: user id cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
if ttl <= 0 {
|
||||||
|
return Session{}, fmt.Errorf("%w: session ttl must be positive", ErrInvalid)
|
||||||
|
}
|
||||||
|
token, err := helpers.RandomHexID(32)
|
||||||
|
if err != nil {
|
||||||
|
return Session{}, err
|
||||||
|
}
|
||||||
|
csrfToken, err := helpers.RandomHexID(32)
|
||||||
|
if err != nil {
|
||||||
|
return Session{}, err
|
||||||
|
}
|
||||||
|
now := time.Now().UTC()
|
||||||
|
session := Session{
|
||||||
|
Token: token,
|
||||||
|
CSRFToken: csrfToken,
|
||||||
|
UserID: userID,
|
||||||
|
CreatedAt: now,
|
||||||
|
ExpiresAt: now.Add(ttl),
|
||||||
|
}
|
||||||
|
err = store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
return putJSON(txn, sessionKey(token), session)
|
||||||
|
})
|
||||||
|
return session, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetSession(token string) (Session, bool, error) {
|
||||||
|
token = strings.TrimSpace(token)
|
||||||
|
if token == "" {
|
||||||
|
return Session{}, false, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
var session Session
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
return getJSON(txn, sessionKey(token), &session)
|
||||||
|
})
|
||||||
|
if errors.Is(err, ErrNotFound) {
|
||||||
|
return Session{}, false, nil
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return Session{}, false, err
|
||||||
|
}
|
||||||
|
if time.Now().UTC().After(session.ExpiresAt) {
|
||||||
|
_ = store.DeleteSession(token)
|
||||||
|
return Session{}, false, nil
|
||||||
|
}
|
||||||
|
return session, true, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) DeleteSession(token string) error {
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
err := txn.Delete(sessionKey(token))
|
||||||
|
if errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func sessionKey(token string) []byte {
|
||||||
|
return []byte("session/" + strings.TrimSpace(token))
|
||||||
|
}
|
||||||
379
lib/metastore/store.go
Normal file
379
lib/metastore/store.go
Normal file
@@ -0,0 +1,379 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/dgraph-io/badger/v4"
|
||||||
|
"golang.org/x/crypto/bcrypt"
|
||||||
|
|
||||||
|
"warpbox/lib/helpers"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
ErrNotFound = errors.New("not found")
|
||||||
|
ErrDuplicate = errors.New("duplicate")
|
||||||
|
ErrInvalid = errors.New("invalid")
|
||||||
|
)
|
||||||
|
|
||||||
|
type Store struct {
|
||||||
|
db *badger.DB
|
||||||
|
}
|
||||||
|
|
||||||
|
func Open(path string) (*Store, error) {
|
||||||
|
opts := badger.DefaultOptions(path).WithLogger(nil)
|
||||||
|
db, err := badger.Open(opts)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &Store{db: db}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) Close() error {
|
||||||
|
if store == nil || store.db == nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
return store.db.Close()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) SetSetting(name string, value string) error {
|
||||||
|
name = strings.TrimSpace(name)
|
||||||
|
if name == "" {
|
||||||
|
return fmt.Errorf("%w: setting name cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
return txn.Set(settingKey(name), []byte(value))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) DeleteSetting(name string) error {
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
return txn.Delete(settingKey(name))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetSetting(name string) (string, bool, error) {
|
||||||
|
var value string
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
item, err := txn.Get(settingKey(name))
|
||||||
|
if errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return ErrNotFound
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return item.Value(func(data []byte) error {
|
||||||
|
value = string(data)
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
})
|
||||||
|
if errors.Is(err, ErrNotFound) {
|
||||||
|
return "", false, nil
|
||||||
|
}
|
||||||
|
return value, err == nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) ListSettings() (map[string]string, error) {
|
||||||
|
settings := make(map[string]string)
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
opts := badger.DefaultIteratorOptions
|
||||||
|
opts.Prefix = []byte("setting/")
|
||||||
|
it := txn.NewIterator(opts)
|
||||||
|
defer it.Close()
|
||||||
|
|
||||||
|
for it.Rewind(); it.Valid(); it.Next() {
|
||||||
|
item := it.Item()
|
||||||
|
name := strings.TrimPrefix(string(item.Key()), "setting/")
|
||||||
|
if err := item.Value(func(data []byte) error {
|
||||||
|
settings[name] = string(data)
|
||||||
|
return nil
|
||||||
|
}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
return settings, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func HashPassword(password string) (string, error) {
|
||||||
|
if strings.TrimSpace(password) == "" {
|
||||||
|
return "", fmt.Errorf("%w: password cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
hash, err := bcrypt.GenerateFromPassword([]byte(password), bcrypt.DefaultCost)
|
||||||
|
if err != nil {
|
||||||
|
return "", err
|
||||||
|
}
|
||||||
|
return string(hash), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func VerifyPassword(hash string, password string) bool {
|
||||||
|
if hash == "" || password == "" {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return bcrypt.CompareHashAndPassword([]byte(hash), []byte(password)) == nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) CreateUserWithPassword(username string, email string, password string, tagIDs []string) (User, error) {
|
||||||
|
hash, err := HashPassword(password)
|
||||||
|
if err != nil {
|
||||||
|
return User{}, err
|
||||||
|
}
|
||||||
|
user := User{
|
||||||
|
Username: username,
|
||||||
|
Email: email,
|
||||||
|
PasswordHash: hash,
|
||||||
|
TagIDs: uniqueStrings(tagIDs),
|
||||||
|
}
|
||||||
|
if err := store.CreateUser(&user); err != nil {
|
||||||
|
return User{}, err
|
||||||
|
}
|
||||||
|
return user, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) CreateUser(user *User) error {
|
||||||
|
if user == nil {
|
||||||
|
return fmt.Errorf("%w: user cannot be nil", ErrInvalid)
|
||||||
|
}
|
||||||
|
username := strings.TrimSpace(user.Username)
|
||||||
|
if username == "" {
|
||||||
|
return fmt.Errorf("%w: username cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
email := strings.TrimSpace(user.Email)
|
||||||
|
if user.PasswordHash == "" {
|
||||||
|
return fmt.Errorf("%w: password hash cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
|
||||||
|
now := time.Now().UTC()
|
||||||
|
if user.ID == "" {
|
||||||
|
id, err := helpers.RandomHexID(16)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
user.ID = id
|
||||||
|
}
|
||||||
|
user.Username = username
|
||||||
|
user.Email = email
|
||||||
|
user.TagIDs = uniqueStrings(user.TagIDs)
|
||||||
|
user.CreatedAt = now
|
||||||
|
user.UpdatedAt = now
|
||||||
|
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
if exists, err := keyExists(txn, usernameKey(username)); err != nil || exists {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return fmt.Errorf("%w: username already exists", ErrDuplicate)
|
||||||
|
}
|
||||||
|
if email != "" {
|
||||||
|
if exists, err := keyExists(txn, emailKey(email)); err != nil || exists {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return fmt.Errorf("%w: email already exists", ErrDuplicate)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err := putJSON(txn, userKey(user.ID), user); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := txn.Set(usernameKey(username), []byte(user.ID)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if email != "" {
|
||||||
|
return txn.Set(emailKey(email), []byte(user.ID))
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) UpdateUser(user User) error {
|
||||||
|
if strings.TrimSpace(user.ID) == "" {
|
||||||
|
return fmt.Errorf("%w: user id cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
user.Username = strings.TrimSpace(user.Username)
|
||||||
|
user.Email = strings.TrimSpace(user.Email)
|
||||||
|
if user.Username == "" {
|
||||||
|
return fmt.Errorf("%w: username cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
user.TagIDs = uniqueStrings(user.TagIDs)
|
||||||
|
user.UpdatedAt = time.Now().UTC()
|
||||||
|
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
var existing User
|
||||||
|
if err := getJSON(txn, userKey(user.ID), &existing); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
oldUsername := normalizeIndex(existing.Username)
|
||||||
|
newUsername := normalizeIndex(user.Username)
|
||||||
|
if oldUsername != newUsername {
|
||||||
|
if exists, err := keyExists(txn, usernameKey(user.Username)); err != nil || exists {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return fmt.Errorf("%w: username already exists", ErrDuplicate)
|
||||||
|
}
|
||||||
|
if err := txn.Delete(usernameKey(existing.Username)); err != nil && !errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := txn.Set(usernameKey(user.Username), []byte(user.ID)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
oldEmail := normalizeIndex(existing.Email)
|
||||||
|
newEmail := normalizeIndex(user.Email)
|
||||||
|
if oldEmail != newEmail {
|
||||||
|
if newEmail != "" {
|
||||||
|
if exists, err := keyExists(txn, emailKey(user.Email)); err != nil || exists {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return fmt.Errorf("%w: email already exists", ErrDuplicate)
|
||||||
|
}
|
||||||
|
if err := txn.Set(emailKey(user.Email), []byte(user.ID)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if oldEmail != "" {
|
||||||
|
if err := txn.Delete(emailKey(existing.Email)); err != nil && !errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return putJSON(txn, userKey(user.ID), user)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetUser(id string) (User, bool, error) {
|
||||||
|
var user User
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
return getJSON(txn, userKey(id), &user)
|
||||||
|
})
|
||||||
|
if errors.Is(err, ErrNotFound) {
|
||||||
|
return User{}, false, nil
|
||||||
|
}
|
||||||
|
return user, err == nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetUserByUsername(username string) (User, bool, error) {
|
||||||
|
return store.getUserByIndex(usernameKey(username))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetUserByEmail(email string) (User, bool, error) {
|
||||||
|
return store.getUserByIndex(emailKey(email))
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) ListUsers() ([]User, error) {
|
||||||
|
users := []User{}
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
opts := badger.DefaultIteratorOptions
|
||||||
|
opts.Prefix = []byte("user/")
|
||||||
|
it := txn.NewIterator(opts)
|
||||||
|
defer it.Close()
|
||||||
|
|
||||||
|
for it.Rewind(); it.Valid(); it.Next() {
|
||||||
|
var user User
|
||||||
|
if err := it.Item().Value(func(data []byte) error {
|
||||||
|
return json.Unmarshal(data, &user)
|
||||||
|
}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
users = append(users, user)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
return users, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) getUserByIndex(key []byte) (User, bool, error) {
|
||||||
|
var id string
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
item, err := txn.Get(key)
|
||||||
|
if errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return ErrNotFound
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return item.Value(func(data []byte) error {
|
||||||
|
id = string(data)
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
})
|
||||||
|
if errors.Is(err, ErrNotFound) {
|
||||||
|
return User{}, false, nil
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return User{}, false, err
|
||||||
|
}
|
||||||
|
return store.GetUser(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
func putJSON(txn *badger.Txn, key []byte, value any) error {
|
||||||
|
data, err := json.Marshal(value)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return txn.Set(key, data)
|
||||||
|
}
|
||||||
|
|
||||||
|
func getJSON(txn *badger.Txn, key []byte, value any) error {
|
||||||
|
item, err := txn.Get(key)
|
||||||
|
if errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return ErrNotFound
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return item.Value(func(data []byte) error {
|
||||||
|
return json.Unmarshal(data, value)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func keyExists(txn *badger.Txn, key []byte) (bool, error) {
|
||||||
|
_, err := txn.Get(key)
|
||||||
|
if errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return false, nil
|
||||||
|
}
|
||||||
|
return err == nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func settingKey(name string) []byte {
|
||||||
|
return []byte("setting/" + strings.TrimSpace(name))
|
||||||
|
}
|
||||||
|
|
||||||
|
func userKey(id string) []byte {
|
||||||
|
return []byte("user/" + strings.TrimSpace(id))
|
||||||
|
}
|
||||||
|
|
||||||
|
func usernameKey(username string) []byte {
|
||||||
|
return []byte("user_by_name/" + normalizeIndex(username))
|
||||||
|
}
|
||||||
|
|
||||||
|
func emailKey(email string) []byte {
|
||||||
|
return []byte("user_by_email/" + normalizeIndex(email))
|
||||||
|
}
|
||||||
|
|
||||||
|
func normalizeIndex(value string) string {
|
||||||
|
return strings.ToLower(strings.TrimSpace(value))
|
||||||
|
}
|
||||||
|
|
||||||
|
func uniqueStrings(values []string) []string {
|
||||||
|
seen := make(map[string]bool, len(values))
|
||||||
|
out := make([]string, 0, len(values))
|
||||||
|
for _, value := range values {
|
||||||
|
value = strings.TrimSpace(value)
|
||||||
|
if value == "" || seen[value] {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
seen[value] = true
|
||||||
|
out = append(out, value)
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
220
lib/metastore/tags.go
Normal file
220
lib/metastore/tags.go
Normal file
@@ -0,0 +1,220 @@
|
|||||||
|
package metastore
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/dgraph-io/badger/v4"
|
||||||
|
|
||||||
|
"warpbox/lib/helpers"
|
||||||
|
)
|
||||||
|
|
||||||
|
func AdminPermissions() TagPermissions {
|
||||||
|
unlimited := int64(0)
|
||||||
|
return TagPermissions{
|
||||||
|
UploadAllowed: true,
|
||||||
|
MaxFileSizeBytes: &unlimited,
|
||||||
|
MaxBoxSizeBytes: &unlimited,
|
||||||
|
OneTimeDownloadAllowed: true,
|
||||||
|
ZipDownloadAllowed: true,
|
||||||
|
RenewableAllowed: true,
|
||||||
|
AdminAccess: true,
|
||||||
|
AdminUsersManage: true,
|
||||||
|
AdminSettingsManage: true,
|
||||||
|
AdminBoxesView: true,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) EnsureAdminTag() (Tag, error) {
|
||||||
|
tag, ok, err := store.GetTagByName(AdminTagName)
|
||||||
|
if err != nil {
|
||||||
|
return Tag{}, err
|
||||||
|
}
|
||||||
|
if ok {
|
||||||
|
tag.Protected = true
|
||||||
|
tag.Permissions = AdminPermissions()
|
||||||
|
tag.Description = "Built-in administrator permissions"
|
||||||
|
if err := store.UpdateTag(tag); err != nil {
|
||||||
|
return Tag{}, err
|
||||||
|
}
|
||||||
|
return tag, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
tag = Tag{
|
||||||
|
Name: AdminTagName,
|
||||||
|
Description: "Built-in administrator permissions",
|
||||||
|
Protected: true,
|
||||||
|
Permissions: AdminPermissions(),
|
||||||
|
}
|
||||||
|
if err := store.CreateTag(&tag); err != nil {
|
||||||
|
return Tag{}, err
|
||||||
|
}
|
||||||
|
return tag, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) CreateTag(tag *Tag) error {
|
||||||
|
if tag == nil {
|
||||||
|
return fmt.Errorf("%w: tag cannot be nil", ErrInvalid)
|
||||||
|
}
|
||||||
|
tag.Name = strings.TrimSpace(tag.Name)
|
||||||
|
tag.Description = strings.TrimSpace(tag.Description)
|
||||||
|
if tag.Name == "" {
|
||||||
|
return fmt.Errorf("%w: tag name cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
|
||||||
|
now := time.Now().UTC()
|
||||||
|
if tag.ID == "" {
|
||||||
|
id, err := helpers.RandomHexID(16)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
tag.ID = id
|
||||||
|
}
|
||||||
|
tag.CreatedAt = now
|
||||||
|
tag.UpdatedAt = now
|
||||||
|
normalizeTagPermissions(&tag.Permissions)
|
||||||
|
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
if exists, err := keyExists(txn, tagNameKey(tag.Name)); err != nil || exists {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return fmt.Errorf("%w: tag name already exists", ErrDuplicate)
|
||||||
|
}
|
||||||
|
if err := putJSON(txn, tagKey(tag.ID), tag); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return txn.Set(tagNameKey(tag.Name), []byte(tag.ID))
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) UpdateTag(tag Tag) error {
|
||||||
|
tag.Name = strings.TrimSpace(tag.Name)
|
||||||
|
tag.Description = strings.TrimSpace(tag.Description)
|
||||||
|
if tag.ID == "" {
|
||||||
|
return fmt.Errorf("%w: tag id cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
if tag.Name == "" {
|
||||||
|
return fmt.Errorf("%w: tag name cannot be empty", ErrInvalid)
|
||||||
|
}
|
||||||
|
tag.UpdatedAt = time.Now().UTC()
|
||||||
|
normalizeTagPermissions(&tag.Permissions)
|
||||||
|
|
||||||
|
return store.db.Update(func(txn *badger.Txn) error {
|
||||||
|
var existing Tag
|
||||||
|
if err := getJSON(txn, tagKey(tag.ID), &existing); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if normalizeIndex(existing.Name) != normalizeIndex(tag.Name) {
|
||||||
|
if exists, err := keyExists(txn, tagNameKey(tag.Name)); err != nil || exists {
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return fmt.Errorf("%w: tag name already exists", ErrDuplicate)
|
||||||
|
}
|
||||||
|
if err := txn.Delete(tagNameKey(existing.Name)); err != nil && !errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := txn.Set(tagNameKey(tag.Name), []byte(tag.ID)); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if existing.Protected {
|
||||||
|
tag.Protected = true
|
||||||
|
}
|
||||||
|
if tag.Name == AdminTagName {
|
||||||
|
tag.Protected = true
|
||||||
|
tag.Permissions = AdminPermissions()
|
||||||
|
}
|
||||||
|
return putJSON(txn, tagKey(tag.ID), tag)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetTag(id string) (Tag, bool, error) {
|
||||||
|
var tag Tag
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
return getJSON(txn, tagKey(id), &tag)
|
||||||
|
})
|
||||||
|
if errors.Is(err, ErrNotFound) {
|
||||||
|
return Tag{}, false, nil
|
||||||
|
}
|
||||||
|
return tag, err == nil, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) GetTagByName(name string) (Tag, bool, error) {
|
||||||
|
var id string
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
item, err := txn.Get(tagNameKey(name))
|
||||||
|
if errors.Is(err, badger.ErrKeyNotFound) {
|
||||||
|
return ErrNotFound
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return item.Value(func(data []byte) error {
|
||||||
|
id = string(data)
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
})
|
||||||
|
if errors.Is(err, ErrNotFound) {
|
||||||
|
return Tag{}, false, nil
|
||||||
|
}
|
||||||
|
if err != nil {
|
||||||
|
return Tag{}, false, err
|
||||||
|
}
|
||||||
|
return store.GetTag(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) ListTags() ([]Tag, error) {
|
||||||
|
tags := []Tag{}
|
||||||
|
err := store.db.View(func(txn *badger.Txn) error {
|
||||||
|
opts := badger.DefaultIteratorOptions
|
||||||
|
opts.Prefix = []byte("tag/")
|
||||||
|
it := txn.NewIterator(opts)
|
||||||
|
defer it.Close()
|
||||||
|
|
||||||
|
for it.Rewind(); it.Valid(); it.Next() {
|
||||||
|
var tag Tag
|
||||||
|
if err := it.Item().Value(func(data []byte) error {
|
||||||
|
return json.Unmarshal(data, &tag)
|
||||||
|
}); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
tags = append(tags, tag)
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
})
|
||||||
|
return tags, err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (store *Store) TagsByID(ids []string) ([]Tag, error) {
|
||||||
|
tags := make([]Tag, 0, len(ids))
|
||||||
|
for _, id := range ids {
|
||||||
|
tag, ok, err := store.GetTag(id)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if ok {
|
||||||
|
tags = append(tags, tag)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return tags, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func normalizeTagPermissions(perms *TagPermissions) {
|
||||||
|
if perms == nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
perms.AllowedExpirySeconds = uniqueInt64s(perms.AllowedExpirySeconds)
|
||||||
|
}
|
||||||
|
|
||||||
|
func tagKey(id string) []byte {
|
||||||
|
return []byte("tag/" + strings.TrimSpace(id))
|
||||||
|
}
|
||||||
|
|
||||||
|
func tagNameKey(name string) []byte {
|
||||||
|
return []byte("tag_by_name/" + normalizeIndex(name))
|
||||||
|
}
|
||||||
@@ -42,6 +42,8 @@ type BoxFile struct {
|
|||||||
|
|
||||||
type BoxManifest struct {
|
type BoxManifest struct {
|
||||||
Files []BoxFile `json:"files"`
|
Files []BoxFile `json:"files"`
|
||||||
|
OwnerID string `json:"owner_id,omitempty"`
|
||||||
|
OwnerUsername string `json:"owner_username,omitempty"`
|
||||||
CreatedAt time.Time `json:"created_at"`
|
CreatedAt time.Time `json:"created_at"`
|
||||||
ExpiresAt time.Time `json:"expires_at"`
|
ExpiresAt time.Time `json:"expires_at"`
|
||||||
RetentionKey string `json:"retention_key"`
|
RetentionKey string `json:"retention_key"`
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ package routing
|
|||||||
import "github.com/gin-gonic/gin"
|
import "github.com/gin-gonic/gin"
|
||||||
|
|
||||||
type Handlers struct {
|
type Handlers struct {
|
||||||
Health gin.HandlerFunc
|
|
||||||
Index gin.HandlerFunc
|
Index gin.HandlerFunc
|
||||||
ShowBox gin.HandlerFunc
|
ShowBox gin.HandlerFunc
|
||||||
BoxLogin gin.HandlerFunc
|
BoxLogin gin.HandlerFunc
|
||||||
@@ -17,29 +16,9 @@ type Handlers struct {
|
|||||||
FileStatusUpdate gin.HandlerFunc
|
FileStatusUpdate gin.HandlerFunc
|
||||||
DirectBoxUpload gin.HandlerFunc
|
DirectBoxUpload gin.HandlerFunc
|
||||||
LegacyUpload gin.HandlerFunc
|
LegacyUpload gin.HandlerFunc
|
||||||
|
|
||||||
AdminLogin gin.HandlerFunc
|
|
||||||
AdminLoginPost gin.HandlerFunc
|
|
||||||
AdminLogout gin.HandlerFunc
|
|
||||||
AdminDashboard gin.HandlerFunc
|
|
||||||
AdminAlerts gin.HandlerFunc
|
|
||||||
AdminBoxes gin.HandlerFunc
|
|
||||||
AdminBoxesAction gin.HandlerFunc
|
|
||||||
AdminUsers gin.HandlerFunc
|
|
||||||
AdminActivity gin.HandlerFunc
|
|
||||||
AdminSecurity gin.HandlerFunc
|
|
||||||
AdminAlertsAction gin.HandlerFunc
|
|
||||||
AdminSecurityAction gin.HandlerFunc
|
|
||||||
AdminSettings gin.HandlerFunc
|
|
||||||
AdminSettingsExport gin.HandlerFunc
|
|
||||||
AdminSettingsSave gin.HandlerFunc
|
|
||||||
AdminSettingsImport gin.HandlerFunc
|
|
||||||
AdminSettingsReset gin.HandlerFunc
|
|
||||||
AdminAuth gin.HandlerFunc
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func Register(router *gin.Engine, handlers Handlers) {
|
func Register(router *gin.Engine, handlers Handlers) {
|
||||||
router.GET("/health", handlers.Health)
|
|
||||||
router.GET("/", handlers.Index)
|
router.GET("/", handlers.Index)
|
||||||
|
|
||||||
router.GET("/box/:id", handlers.ShowBox)
|
router.GET("/box/:id", handlers.ShowBox)
|
||||||
@@ -57,25 +36,4 @@ func Register(router *gin.Engine, handlers Handlers) {
|
|||||||
// Legacy upload routes are kept for compatibility with older clients.
|
// Legacy upload routes are kept for compatibility with older clients.
|
||||||
router.POST("/box/:id/upload", handlers.DirectBoxUpload)
|
router.POST("/box/:id/upload", handlers.DirectBoxUpload)
|
||||||
router.POST("/upload", handlers.LegacyUpload)
|
router.POST("/upload", handlers.LegacyUpload)
|
||||||
|
|
||||||
admin := router.Group("/admin")
|
|
||||||
admin.GET("/login", handlers.AdminLogin)
|
|
||||||
admin.POST("/login", handlers.AdminLoginPost)
|
|
||||||
admin.GET("/logout", handlers.AdminLogout)
|
|
||||||
|
|
||||||
protected := router.Group("/admin", handlers.AdminAuth)
|
|
||||||
protected.GET("/dashboard", handlers.AdminDashboard)
|
|
||||||
protected.GET("/alerts", handlers.AdminAlerts)
|
|
||||||
protected.POST("/alerts/actions", handlers.AdminAlertsAction)
|
|
||||||
protected.GET("/boxes", handlers.AdminBoxes)
|
|
||||||
protected.POST("/boxes/actions", handlers.AdminBoxesAction)
|
|
||||||
protected.GET("/users", handlers.AdminUsers)
|
|
||||||
protected.GET("/activity", handlers.AdminActivity)
|
|
||||||
protected.GET("/security", handlers.AdminSecurity)
|
|
||||||
protected.POST("/security/actions", handlers.AdminSecurityAction)
|
|
||||||
protected.GET("/settings", handlers.AdminSettings)
|
|
||||||
protected.GET("/settings/export", handlers.AdminSettingsExport)
|
|
||||||
protected.POST("/settings/save", handlers.AdminSettingsSave)
|
|
||||||
protected.POST("/settings/import", handlers.AdminSettingsImport)
|
|
||||||
protected.POST("/settings/reset", handlers.AdminSettingsReset)
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,426 +0,0 @@
|
|||||||
package security
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/binary"
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
"os"
|
|
||||||
"sort"
|
|
||||||
"strings"
|
|
||||||
"sync"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/dgraph-io/badger/v4"
|
|
||||||
)
|
|
||||||
|
|
||||||
type Config struct {
|
|
||||||
IPWhitelist string
|
|
||||||
AdminIPWhitelist string
|
|
||||||
LoginWindowSeconds int64
|
|
||||||
LoginMaxAttempts int
|
|
||||||
BanSeconds int64
|
|
||||||
ScanWindowSeconds int64
|
|
||||||
ScanMaxAttempts int
|
|
||||||
UploadWindowSeconds int64
|
|
||||||
UploadMaxRequests int
|
|
||||||
UploadMaxBytes int64
|
|
||||||
}
|
|
||||||
|
|
||||||
type Guard struct {
|
|
||||||
mu sync.Mutex
|
|
||||||
failedLogins map[string][]time.Time
|
|
||||||
scanAttempts map[string][]time.Time
|
|
||||||
uploadEvents map[string][]uploadEvent
|
|
||||||
bannedUntil map[string]time.Time
|
|
||||||
ipWhitelist []ipMatcher
|
|
||||||
adminWhitelist []ipMatcher
|
|
||||||
banDB *badger.DB
|
|
||||||
}
|
|
||||||
|
|
||||||
type ipMatcher struct {
|
|
||||||
exact net.IP
|
|
||||||
prefix *net.IPNet
|
|
||||||
}
|
|
||||||
|
|
||||||
type uploadEvent struct {
|
|
||||||
at time.Time
|
|
||||||
bytes int64
|
|
||||||
}
|
|
||||||
|
|
||||||
type BanEntry struct {
|
|
||||||
IP string `json:"ip"`
|
|
||||||
Until time.Time `json:"until"`
|
|
||||||
}
|
|
||||||
|
|
||||||
const banKeyPrefix = "ban:"
|
|
||||||
|
|
||||||
func NewGuard() *Guard {
|
|
||||||
return &Guard{
|
|
||||||
failedLogins: map[string][]time.Time{},
|
|
||||||
scanAttempts: map[string][]time.Time{},
|
|
||||||
uploadEvents: map[string][]uploadEvent{},
|
|
||||||
bannedUntil: map[string]time.Time{},
|
|
||||||
ipWhitelist: []ipMatcher{},
|
|
||||||
adminWhitelist: []ipMatcher{},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) Close() error {
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
if g.banDB == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
err := g.banDB.Close()
|
|
||||||
g.banDB = nil
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) EnableBanPersistence(path string) error {
|
|
||||||
if strings.TrimSpace(path) == "" {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
|
|
||||||
if g.banDB != nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
opts := badger.DefaultOptions(path)
|
|
||||||
opts.Logger = nil
|
|
||||||
db, err := badger.Open(opts)
|
|
||||||
if err != nil {
|
|
||||||
// Corruption-safe fallback: quarantine badger files and start fresh.
|
|
||||||
_ = os.Rename(path, path+".corrupt."+time.Now().UTC().Format("20060102T150405"))
|
|
||||||
db, err = badger.Open(opts)
|
|
||||||
}
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
g.banDB = db
|
|
||||||
|
|
||||||
if err := g.loadBansLocked(); err != nil {
|
|
||||||
_ = g.banDB.Close()
|
|
||||||
g.banDB = nil
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) Reload(cfg Config) error {
|
|
||||||
ipWhitelist, err := ParseIPMatchers(cfg.IPWhitelist, true)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("ip whitelist: %w", err)
|
|
||||||
}
|
|
||||||
adminWhitelist, err := ParseIPMatchers(cfg.AdminIPWhitelist, true)
|
|
||||||
if err != nil {
|
|
||||||
return fmt.Errorf("admin ip whitelist: %w", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
g.ipWhitelist = ipWhitelist
|
|
||||||
g.adminWhitelist = adminWhitelist
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) IsWhitelisted(ip string) bool {
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
return matchIP(g.ipWhitelist, ip)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) IsAdminWhitelisted(ip string) bool {
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
return matchIP(g.adminWhitelist, ip) || matchIP(g.ipWhitelist, ip)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) IsBanned(ip string) bool {
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
until, ok := g.bannedUntil[ip]
|
|
||||||
if !ok {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
if time.Now().UTC().After(until) {
|
|
||||||
delete(g.bannedUntil, ip)
|
|
||||||
g.deleteBanLocked(ip)
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) Ban(ip string, seconds int64) {
|
|
||||||
if seconds <= 0 || ip == "" {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
until := time.Now().UTC().Add(time.Duration(seconds) * time.Second)
|
|
||||||
g.bannedUntil[ip] = until
|
|
||||||
g.saveBanLocked(ip, until)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) BanUntil(ip string, until time.Time) {
|
|
||||||
if ip == "" || until.IsZero() {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
until = until.UTC()
|
|
||||||
g.bannedUntil[ip] = until
|
|
||||||
g.saveBanLocked(ip, until)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) Unban(ip string) {
|
|
||||||
if ip == "" {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
delete(g.bannedUntil, ip)
|
|
||||||
g.deleteBanLocked(ip)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) BanList() []BanEntry {
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
now := time.Now().UTC()
|
|
||||||
out := make([]BanEntry, 0, len(g.bannedUntil))
|
|
||||||
for ip, until := range g.bannedUntil {
|
|
||||||
if now.After(until) {
|
|
||||||
delete(g.bannedUntil, ip)
|
|
||||||
g.deleteBanLocked(ip)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
out = append(out, BanEntry{IP: ip, Until: until})
|
|
||||||
}
|
|
||||||
sort.Slice(out, func(i, j int) bool {
|
|
||||||
return out[i].Until.Before(out[j].Until)
|
|
||||||
})
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) RegisterFailedLogin(ip string, windowSeconds int64, maxAttempts int, banSeconds int64) (bool, int) {
|
|
||||||
if ip == "" || maxAttempts <= 0 || windowSeconds <= 0 {
|
|
||||||
return false, 0
|
|
||||||
}
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
now := time.Now().UTC()
|
|
||||||
cutoff := now.Add(-time.Duration(windowSeconds) * time.Second)
|
|
||||||
attempts := pruneTimes(g.failedLogins[ip], cutoff)
|
|
||||||
attempts = append(attempts, now)
|
|
||||||
g.failedLogins[ip] = attempts
|
|
||||||
if len(attempts) >= maxAttempts {
|
|
||||||
until := now.Add(time.Duration(banSeconds) * time.Second)
|
|
||||||
g.bannedUntil[ip] = until
|
|
||||||
g.saveBanLocked(ip, until)
|
|
||||||
return true, len(attempts)
|
|
||||||
}
|
|
||||||
return false, len(attempts)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) RegisterScanAttempt(ip string, windowSeconds int64, maxAttempts int, banSeconds int64) (bool, int) {
|
|
||||||
if ip == "" || maxAttempts <= 0 || windowSeconds <= 0 {
|
|
||||||
return false, 0
|
|
||||||
}
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
now := time.Now().UTC()
|
|
||||||
cutoff := now.Add(-time.Duration(windowSeconds) * time.Second)
|
|
||||||
attempts := pruneTimes(g.scanAttempts[ip], cutoff)
|
|
||||||
attempts = append(attempts, now)
|
|
||||||
g.scanAttempts[ip] = attempts
|
|
||||||
if len(attempts) >= maxAttempts {
|
|
||||||
until := now.Add(time.Duration(banSeconds) * time.Second)
|
|
||||||
g.bannedUntil[ip] = until
|
|
||||||
g.saveBanLocked(ip, until)
|
|
||||||
return true, len(attempts)
|
|
||||||
}
|
|
||||||
return false, len(attempts)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) AllowUpload(ip string, size int64, windowSeconds int64, maxRequests int, maxBytes int64) (bool, int, int64) {
|
|
||||||
if ip == "" || windowSeconds <= 0 || maxRequests <= 0 {
|
|
||||||
return true, 0, 0
|
|
||||||
}
|
|
||||||
g.mu.Lock()
|
|
||||||
defer g.mu.Unlock()
|
|
||||||
now := time.Now().UTC()
|
|
||||||
cutoff := now.Add(-time.Duration(windowSeconds) * time.Second)
|
|
||||||
events := g.uploadEvents[ip]
|
|
||||||
kept := make([]uploadEvent, 0, len(events)+1)
|
|
||||||
totalBytes := int64(0)
|
|
||||||
for _, event := range events {
|
|
||||||
if event.at.After(cutoff) {
|
|
||||||
kept = append(kept, event)
|
|
||||||
totalBytes += event.bytes
|
|
||||||
}
|
|
||||||
}
|
|
||||||
nextCount := len(kept) + 1
|
|
||||||
nextBytes := totalBytes + size
|
|
||||||
if nextCount > maxRequests {
|
|
||||||
return false, nextCount, nextBytes
|
|
||||||
}
|
|
||||||
if maxBytes > 0 && nextBytes > maxBytes {
|
|
||||||
return false, nextCount, nextBytes
|
|
||||||
}
|
|
||||||
kept = append(kept, uploadEvent{at: now, bytes: size})
|
|
||||||
g.uploadEvents[ip] = kept
|
|
||||||
return true, nextCount, nextBytes
|
|
||||||
}
|
|
||||||
|
|
||||||
func ParseIPMatchers(raw string, allowCIDR bool) ([]ipMatcher, error) {
|
|
||||||
entries := []ipMatcher{}
|
|
||||||
for _, chunk := range strings.Split(raw, ",") {
|
|
||||||
value := strings.TrimSpace(chunk)
|
|
||||||
if value == "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if strings.Contains(value, "/") {
|
|
||||||
if !allowCIDR {
|
|
||||||
return nil, fmt.Errorf("%q must be a CIDR", value)
|
|
||||||
}
|
|
||||||
_, network, err := net.ParseCIDR(value)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("invalid CIDR %q", value)
|
|
||||||
}
|
|
||||||
entries = append(entries, ipMatcher{prefix: network})
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
parsed := net.ParseIP(value)
|
|
||||||
if parsed == nil {
|
|
||||||
return nil, fmt.Errorf("invalid IP %q", value)
|
|
||||||
}
|
|
||||||
entries = append(entries, ipMatcher{exact: parsed})
|
|
||||||
}
|
|
||||||
return entries, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func ParseCIDRList(raw string) ([]net.IPNet, error) {
|
|
||||||
entries := []net.IPNet{}
|
|
||||||
for _, chunk := range strings.Split(raw, ",") {
|
|
||||||
value := strings.TrimSpace(chunk)
|
|
||||||
if value == "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
_, network, err := net.ParseCIDR(value)
|
|
||||||
if err != nil {
|
|
||||||
return nil, fmt.Errorf("invalid CIDR %q", value)
|
|
||||||
}
|
|
||||||
entries = append(entries, *network)
|
|
||||||
}
|
|
||||||
return entries, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func pruneTimes(values []time.Time, cutoff time.Time) []time.Time {
|
|
||||||
kept := make([]time.Time, 0, len(values))
|
|
||||||
for _, value := range values {
|
|
||||||
if value.After(cutoff) {
|
|
||||||
kept = append(kept, value)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return kept
|
|
||||||
}
|
|
||||||
|
|
||||||
func matchIP(rules []ipMatcher, value string) bool {
|
|
||||||
ip := net.ParseIP(strings.TrimSpace(value))
|
|
||||||
if ip == nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for _, rule := range rules {
|
|
||||||
if rule.exact != nil && rule.exact.Equal(ip) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
if rule.prefix != nil && rule.prefix.Contains(ip) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) saveBanLocked(ip string, until time.Time) {
|
|
||||||
if g.banDB == nil || ip == "" || until.IsZero() {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
seconds := int64(time.Until(until).Seconds())
|
|
||||||
if seconds <= 0 {
|
|
||||||
_ = g.banDB.Update(func(txn *badger.Txn) error {
|
|
||||||
return txn.Delete([]byte(banKeyPrefix + ip))
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
value := make([]byte, 8)
|
|
||||||
binary.BigEndian.PutUint64(value, uint64(until.Unix()))
|
|
||||||
_ = g.banDB.Update(func(txn *badger.Txn) error {
|
|
||||||
entry := badger.NewEntry([]byte(banKeyPrefix+ip), value).WithTTL(time.Duration(seconds) * time.Second)
|
|
||||||
return txn.SetEntry(entry)
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) deleteBanLocked(ip string) {
|
|
||||||
if g.banDB == nil || ip == "" {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
_ = g.banDB.Update(func(txn *badger.Txn) error {
|
|
||||||
return txn.Delete([]byte(banKeyPrefix + ip))
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (g *Guard) loadBansLocked() error {
|
|
||||||
if g.banDB == nil {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
now := time.Now().UTC()
|
|
||||||
loaded := map[string]time.Time{}
|
|
||||||
expired := [][]byte{}
|
|
||||||
|
|
||||||
err := g.banDB.View(func(txn *badger.Txn) error {
|
|
||||||
it := txn.NewIterator(badger.DefaultIteratorOptions)
|
|
||||||
defer it.Close()
|
|
||||||
for it.Seek([]byte(banKeyPrefix)); it.ValidForPrefix([]byte(banKeyPrefix)); it.Next() {
|
|
||||||
item := it.Item()
|
|
||||||
key := string(item.Key())
|
|
||||||
ip := strings.TrimPrefix(key, banKeyPrefix)
|
|
||||||
err := item.Value(func(val []byte) error {
|
|
||||||
if len(val) != 8 {
|
|
||||||
expired = append(expired, append([]byte(nil), item.Key()...))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
unix := int64(binary.BigEndian.Uint64(val))
|
|
||||||
until := time.Unix(unix, 0).UTC()
|
|
||||||
if now.After(until) {
|
|
||||||
expired = append(expired, append([]byte(nil), item.Key()...))
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
loaded[ip] = until
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
g.bannedUntil = loaded
|
|
||||||
if len(expired) == 0 {
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
return g.banDB.Update(func(txn *badger.Txn) error {
|
|
||||||
for _, key := range expired {
|
|
||||||
if err := txn.Delete(key); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
})
|
|
||||||
}
|
|
||||||
@@ -1,52 +0,0 @@
|
|||||||
package security
|
|
||||||
|
|
||||||
import (
|
|
||||||
"path/filepath"
|
|
||||||
"testing"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestGuardWhitelistSupportsIPAndCIDR(t *testing.T) {
|
|
||||||
g := NewGuard()
|
|
||||||
if err := g.Reload(Config{IPWhitelist: "203.0.113.10,10.0.0.0/8", AdminIPWhitelist: "192.168.1.0/24"}); err != nil {
|
|
||||||
t.Fatalf("Reload returned error: %v", err)
|
|
||||||
}
|
|
||||||
if !g.IsWhitelisted("203.0.113.10") || !g.IsWhitelisted("10.2.3.4") {
|
|
||||||
t.Fatal("expected IP and CIDR entries to match")
|
|
||||||
}
|
|
||||||
if !g.IsAdminWhitelisted("192.168.1.5") {
|
|
||||||
t.Fatal("expected admin CIDR whitelist match")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGuardBanPersistenceAcrossRestart(t *testing.T) {
|
|
||||||
dir := filepath.Join(t.TempDir(), "bans.badger")
|
|
||||||
g1 := NewGuard()
|
|
||||||
if err := g1.EnableBanPersistence(dir); err != nil {
|
|
||||||
t.Fatalf("EnableBanPersistence returned error: %v", err)
|
|
||||||
}
|
|
||||||
g1.Ban("198.51.100.4", 3600)
|
|
||||||
if err := g1.Close(); err != nil {
|
|
||||||
t.Fatalf("Close returned error: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
g2 := NewGuard()
|
|
||||||
if err := g2.EnableBanPersistence(dir); err != nil {
|
|
||||||
t.Fatalf("EnableBanPersistence returned error: %v", err)
|
|
||||||
}
|
|
||||||
defer g2.Close()
|
|
||||||
if !g2.IsBanned("198.51.100.4") {
|
|
||||||
t.Fatal("expected ban to persist across guard restart")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestGuardBanListPrunesExpired(t *testing.T) {
|
|
||||||
g := NewGuard()
|
|
||||||
g.BanUntil("198.51.100.7", time.Now().UTC().Add(-time.Minute))
|
|
||||||
if g.IsBanned("198.51.100.7") {
|
|
||||||
t.Fatal("expected expired ban to be treated as inactive")
|
|
||||||
}
|
|
||||||
if len(g.BanList()) != 0 {
|
|
||||||
t.Fatal("expected BanList to prune expired entries")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
386
lib/server/account_alerts.go
Normal file
386
lib/server/account_alerts.go
Normal file
@@ -0,0 +1,386 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"log"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AlertPageView struct {
|
||||||
|
PageTitle string
|
||||||
|
WindowTitle string
|
||||||
|
WindowIcon string
|
||||||
|
PageScripts []string
|
||||||
|
AccountNav AccountNavView
|
||||||
|
CSRFToken string
|
||||||
|
Filters AlertFiltersView
|
||||||
|
Stats AlertStatsView
|
||||||
|
Alerts []AlertRowView
|
||||||
|
SelectedAlert *AlertRowView
|
||||||
|
Groups []string
|
||||||
|
CanManageAlerts bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type AlertFiltersView struct {
|
||||||
|
Query string
|
||||||
|
Severity string
|
||||||
|
Status string
|
||||||
|
Group string
|
||||||
|
Sort string
|
||||||
|
}
|
||||||
|
|
||||||
|
type AlertStatsView struct {
|
||||||
|
Open int
|
||||||
|
Acknowledged int
|
||||||
|
Closed int
|
||||||
|
High int
|
||||||
|
Medium int
|
||||||
|
Low int
|
||||||
|
}
|
||||||
|
|
||||||
|
type AlertRowView struct {
|
||||||
|
ID string
|
||||||
|
Title string
|
||||||
|
Description string
|
||||||
|
Severity string
|
||||||
|
Status string
|
||||||
|
Code string
|
||||||
|
Trace string
|
||||||
|
Group string
|
||||||
|
MetadataPretty string
|
||||||
|
CreatedAt string
|
||||||
|
UpdatedAt string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountAlerts(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
page, err := app.ListAlerts(ctx, actor, accountAlertFiltersFromRequest(ctx))
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.HTML(http.StatusOK, "account_alerts.html", page)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountAlertAcknowledge(ctx *gin.Context) {
|
||||||
|
app.handleAccountAlertAction(ctx, func(actor metastore.User, id string) error {
|
||||||
|
return app.AcknowledgeAlert(ctx, actor, id)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountAlertClose(ctx *gin.Context) {
|
||||||
|
app.handleAccountAlertAction(ctx, func(actor metastore.User, id string) error {
|
||||||
|
return app.CloseAlert(ctx, actor, id)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountAlertBulkAcknowledge(ctx *gin.Context) {
|
||||||
|
app.handleAccountAlertBulkAction(ctx, func(actor metastore.User, ids []string) error {
|
||||||
|
return app.BulkAcknowledgeAlerts(ctx, actor, ids)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountAlertBulkClose(ctx *gin.Context) {
|
||||||
|
app.handleAccountAlertBulkAction(ctx, func(actor metastore.User, ids []string) error {
|
||||||
|
return app.BulkCloseAlerts(ctx, actor, ids)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountAlertsExport(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
page, err := app.ListAlerts(ctx, actor, accountAlertFiltersFromRequest(ctx))
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Header("Content-Disposition", `attachment; filename="warpbox-alerts.json"`)
|
||||||
|
ctx.JSON(http.StatusOK, gin.H{"alerts": page.Alerts, "filters": page.Filters, "stats": page.Stats})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountAlertAction(ctx *gin.Context, action func(metastore.User, string) error) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if err := action(actor, ctx.Param("id")); err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/alerts")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountAlertBulkAction(ctx *gin.Context, action func(metastore.User, []string) error) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if err := action(actor, ctx.PostFormArray("alert_ids")); err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/alerts")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) CreateAlert(ctx *gin.Context, actor metastore.User, input metastore.AlertInput) (metastore.Alert, error) {
|
||||||
|
if err := app.requireAlertManage(ctx); err != nil {
|
||||||
|
return metastore.Alert{}, err
|
||||||
|
}
|
||||||
|
if input.CreatedBy == "" {
|
||||||
|
input.CreatedBy = actor.Username
|
||||||
|
}
|
||||||
|
return app.store.CreateAlert(input)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) ListAlerts(ctx *gin.Context, actor metastore.User, filters metastore.AlertFilters) (AlertPageView, error) {
|
||||||
|
if err := app.requireAlertView(ctx); err != nil {
|
||||||
|
return AlertPageView{}, err
|
||||||
|
}
|
||||||
|
alerts, err := app.store.ListAlerts(filters)
|
||||||
|
if err != nil {
|
||||||
|
return AlertPageView{}, err
|
||||||
|
}
|
||||||
|
rows := make([]AlertRowView, 0, len(alerts))
|
||||||
|
stats := AlertStatsView{}
|
||||||
|
groupSet := map[string]bool{}
|
||||||
|
for _, alert := range alerts {
|
||||||
|
row := alertRowView(alert)
|
||||||
|
rows = append(rows, row)
|
||||||
|
groupSet[row.Group] = true
|
||||||
|
switch alert.Status {
|
||||||
|
case metastore.AlertStatusAcknowledged:
|
||||||
|
stats.Acknowledged++
|
||||||
|
case metastore.AlertStatusClosed:
|
||||||
|
stats.Closed++
|
||||||
|
default:
|
||||||
|
stats.Open++
|
||||||
|
}
|
||||||
|
switch alert.Severity {
|
||||||
|
case metastore.AlertSeverityHigh:
|
||||||
|
stats.High++
|
||||||
|
case metastore.AlertSeverityMedium:
|
||||||
|
stats.Medium++
|
||||||
|
default:
|
||||||
|
stats.Low++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
groups := make([]string, 0, len(groupSet))
|
||||||
|
for group := range groupSet {
|
||||||
|
groups = append(groups, group)
|
||||||
|
}
|
||||||
|
if len(groups) == 0 {
|
||||||
|
groups = []string{"system"}
|
||||||
|
}
|
||||||
|
|
||||||
|
nav := app.accountNavView(ctx, "alerts")
|
||||||
|
nav.AlertCount, nav.AlertSeverity = app.openAlertSummary()
|
||||||
|
|
||||||
|
var selected *AlertRowView
|
||||||
|
if len(rows) > 0 {
|
||||||
|
selected = &rows[0]
|
||||||
|
}
|
||||||
|
return AlertPageView{
|
||||||
|
PageTitle: "WarpBox Alerts",
|
||||||
|
WindowTitle: "WarpBox Alerts",
|
||||||
|
WindowIcon: "!",
|
||||||
|
PageScripts: []string{"/static/js/account-alerts.js"},
|
||||||
|
AccountNav: nav,
|
||||||
|
CSRFToken: app.currentCSRFToken(ctx),
|
||||||
|
Filters: AlertFiltersView{Query: filters.Query, Severity: filters.Severity, Status: filters.Status, Group: filters.Group, Sort: filters.Sort},
|
||||||
|
Stats: stats,
|
||||||
|
Alerts: rows,
|
||||||
|
SelectedAlert: selected,
|
||||||
|
Groups: groups,
|
||||||
|
CanManageAlerts: currentAccountPermissions(ctx).AdminAccess,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) AcknowledgeAlert(ctx *gin.Context, actor metastore.User, id string) error {
|
||||||
|
if err := app.requireAlertManage(ctx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return app.store.AcknowledgeAlert(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) CloseAlert(ctx *gin.Context, actor metastore.User, id string) error {
|
||||||
|
if err := app.requireAlertManage(ctx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return app.store.CloseAlert(id)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) BulkAcknowledgeAlerts(ctx *gin.Context, actor metastore.User, ids []string) error {
|
||||||
|
if err := app.requireAlertManage(ctx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, id := range uniqueNonEmpty(ids) {
|
||||||
|
if err := app.store.AcknowledgeAlert(id); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) BulkCloseAlerts(ctx *gin.Context, actor metastore.User, ids []string) error {
|
||||||
|
if err := app.requireAlertManage(ctx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, id := range uniqueNonEmpty(ids) {
|
||||||
|
if err := app.store.CloseAlert(id); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) EmitSystemAlert(code string, severity string, title string, description string, trace string, metadata any) error {
|
||||||
|
raw, err := json.Marshal(metadata)
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("alert metadata marshal failed: %v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
_, err = app.store.CreateAlert(metastore.AlertInput{
|
||||||
|
Title: title,
|
||||||
|
Description: description,
|
||||||
|
Severity: severity,
|
||||||
|
Code: code,
|
||||||
|
Trace: trace,
|
||||||
|
Metadata: raw,
|
||||||
|
CreatedBy: "system",
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
log.Printf("alert persistence failed: %v", err)
|
||||||
|
}
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) requireAlertView(ctx *gin.Context) error {
|
||||||
|
if !currentAccountPermissions(ctx).AdminAccess {
|
||||||
|
return fmt.Errorf("permission denied")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) requireAlertManage(ctx *gin.Context) error {
|
||||||
|
if !currentAccountPermissions(ctx).AdminAccess {
|
||||||
|
return fmt.Errorf("permission denied")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func accountAlertFiltersFromRequest(ctx *gin.Context) metastore.AlertFilters {
|
||||||
|
return metastore.AlertFilters{
|
||||||
|
Query: strings.TrimSpace(ctx.Query("q")),
|
||||||
|
Severity: emptyAsAll(ctx.Query("severity")),
|
||||||
|
Status: emptyAsAll(ctx.Query("status")),
|
||||||
|
Group: emptyAsAll(ctx.Query("group")),
|
||||||
|
Sort: emptyAsNewest(ctx.Query("sort")),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func emptyAsAll(value string) string {
|
||||||
|
value = strings.TrimSpace(value)
|
||||||
|
if value == "" {
|
||||||
|
return "all"
|
||||||
|
}
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
||||||
|
func emptyAsNewest(value string) string {
|
||||||
|
value = strings.TrimSpace(value)
|
||||||
|
if value == "" {
|
||||||
|
return "newest"
|
||||||
|
}
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
|
||||||
|
func alertRowView(alert metastore.Alert) AlertRowView {
|
||||||
|
return AlertRowView{
|
||||||
|
ID: alert.ID,
|
||||||
|
Title: alert.Title,
|
||||||
|
Description: alert.Description,
|
||||||
|
Severity: alert.Severity,
|
||||||
|
Status: alert.Status,
|
||||||
|
Code: alert.Code,
|
||||||
|
Trace: alert.Trace,
|
||||||
|
Group: alertGroupFromTrace(alert.Trace),
|
||||||
|
MetadataPretty: prettyAlertMetadata(alert.Metadata),
|
||||||
|
CreatedAt: formatAdminTime(alert.CreatedAt),
|
||||||
|
UpdatedAt: formatAdminTime(alert.UpdatedAt),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func prettyAlertMetadata(raw json.RawMessage) string {
|
||||||
|
if len(raw) == 0 {
|
||||||
|
return "{}"
|
||||||
|
}
|
||||||
|
var value any
|
||||||
|
if err := json.Unmarshal(raw, &value); err != nil {
|
||||||
|
return string(raw)
|
||||||
|
}
|
||||||
|
pretty, err := json.MarshalIndent(value, "", " ")
|
||||||
|
if err != nil {
|
||||||
|
return string(raw)
|
||||||
|
}
|
||||||
|
return string(pretty)
|
||||||
|
}
|
||||||
|
|
||||||
|
func alertGroupFromTrace(trace string) string {
|
||||||
|
trace = strings.TrimSpace(trace)
|
||||||
|
if trace == "" {
|
||||||
|
return "system"
|
||||||
|
}
|
||||||
|
before, _, found := strings.Cut(trace, ".")
|
||||||
|
if !found || before == "" {
|
||||||
|
return "system"
|
||||||
|
}
|
||||||
|
return strings.ToLower(before)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) openAlertSummary() (int, string) {
|
||||||
|
alerts, err := app.store.ListAlerts(metastore.AlertFilters{Status: metastore.AlertStatusOpen})
|
||||||
|
if err != nil {
|
||||||
|
return 0, "ok"
|
||||||
|
}
|
||||||
|
severity := "ok"
|
||||||
|
for _, alert := range alerts {
|
||||||
|
if alert.Severity == metastore.AlertSeverityHigh {
|
||||||
|
return len(alerts), "danger"
|
||||||
|
}
|
||||||
|
if alert.Severity == metastore.AlertSeverityMedium {
|
||||||
|
severity = "warning"
|
||||||
|
} else if severity == "ok" {
|
||||||
|
severity = "info"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return len(alerts), severity
|
||||||
|
}
|
||||||
|
|
||||||
|
func uniqueNonEmpty(values []string) []string {
|
||||||
|
seen := map[string]bool{}
|
||||||
|
out := []string{}
|
||||||
|
for _, value := range values {
|
||||||
|
value = strings.TrimSpace(value)
|
||||||
|
if value == "" || seen[value] {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
seen[value] = true
|
||||||
|
out = append(out, value)
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
155
lib/server/account_alerts_test.go
Normal file
155
lib/server/account_alerts_test.go
Normal file
@@ -0,0 +1,155 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestAccountAlertsPageListsAndFiltersAlerts(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
createTestAlert(t, app, "601", metastore.AlertSeverityMedium, "thumbnail.generate.failed")
|
||||||
|
createTestAlert(t, app, "701", metastore.AlertSeverityHigh, "storage.connector.health_failed")
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account/alerts?severity=high", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected alerts page, got %d body=%s", response.Code, response.Body.String())
|
||||||
|
}
|
||||||
|
body := response.Body.String()
|
||||||
|
if !strings.Contains(body, "storage.connector.health_failed") {
|
||||||
|
t.Fatal("expected high severity alert")
|
||||||
|
}
|
||||||
|
if strings.Contains(body, "thumbnail.generate.failed") {
|
||||||
|
t.Fatal("did not expect medium severity alert in high filter")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountAlertAcknowledgeAndClose(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
alert := createTestAlert(t, app, "601", metastore.AlertSeverityMedium, "thumbnail.generate.failed")
|
||||||
|
|
||||||
|
response := postAlertAction(router, session, "/account/alerts/"+alert.ID+"/acknowledge", nil)
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected acknowledge redirect, got %d", response.Code)
|
||||||
|
}
|
||||||
|
updated, ok, err := app.store.GetAlert(alert.ID)
|
||||||
|
if err != nil || !ok {
|
||||||
|
t.Fatalf("GetAlert returned ok=%v err=%v", ok, err)
|
||||||
|
}
|
||||||
|
if updated.Status != metastore.AlertStatusAcknowledged {
|
||||||
|
t.Fatalf("expected acknowledged alert, got %s", updated.Status)
|
||||||
|
}
|
||||||
|
|
||||||
|
response = postAlertAction(router, session, "/account/alerts/"+alert.ID+"/close", nil)
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected close redirect, got %d", response.Code)
|
||||||
|
}
|
||||||
|
updated, ok, err = app.store.GetAlert(alert.ID)
|
||||||
|
if err != nil || !ok {
|
||||||
|
t.Fatalf("GetAlert returned ok=%v err=%v", ok, err)
|
||||||
|
}
|
||||||
|
if updated.Status != metastore.AlertStatusClosed {
|
||||||
|
t.Fatalf("expected closed alert, got %s", updated.Status)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountAlertManagePermissionDenied(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
regular, err := app.store.CreateUserWithPassword("regular-alerts", "regular-alerts@example.test", "secret", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateUserWithPassword returned error: %v", err)
|
||||||
|
}
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, regular)
|
||||||
|
alert := createTestAlert(t, app, "601", metastore.AlertSeverityMedium, "thumbnail.generate.failed")
|
||||||
|
|
||||||
|
response := postAlertAction(router, session, "/account/alerts/"+alert.ID+"/acknowledge", nil)
|
||||||
|
if response.Code != http.StatusForbidden {
|
||||||
|
t.Fatalf("expected permission denied, got %d", response.Code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestDashboardUsesRealAlertCount(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
createTestAlert(t, app, "601", metastore.AlertSeverityMedium, "thumbnail.generate.failed")
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected dashboard, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if !strings.Contains(response.Body.String(), "1 alerts") {
|
||||||
|
t.Fatal("expected dashboard alert chip/count")
|
||||||
|
}
|
||||||
|
if !strings.Contains(response.Body.String(), "Thumbnail alert") {
|
||||||
|
t.Fatal("expected dashboard alert preview")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountAlertsExportJSON(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
createTestAlert(t, app, "601", metastore.AlertSeverityMedium, "thumbnail.generate.failed")
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account/alerts/export.json", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected export success, got %d", response.Code)
|
||||||
|
}
|
||||||
|
var payload map[string]any
|
||||||
|
if err := json.Unmarshal(response.Body.Bytes(), &payload); err != nil {
|
||||||
|
t.Fatalf("Unmarshal returned error: %v", err)
|
||||||
|
}
|
||||||
|
if _, ok := payload["alerts"]; !ok {
|
||||||
|
t.Fatal("expected alerts export shape")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func createTestAlert(t *testing.T, app *App, code string, severity string, trace string) metastore.Alert {
|
||||||
|
t.Helper()
|
||||||
|
alert, err := app.store.CreateAlert(metastore.AlertInput{
|
||||||
|
Title: "Thumbnail alert",
|
||||||
|
Description: "Alert test description.",
|
||||||
|
Severity: severity,
|
||||||
|
Code: code,
|
||||||
|
Trace: trace,
|
||||||
|
Metadata: json.RawMessage(`{"box":"box-1","file":"photo.jpg"}`),
|
||||||
|
CreatedBy: "system",
|
||||||
|
})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateAlert returned error: %v", err)
|
||||||
|
}
|
||||||
|
return alert
|
||||||
|
}
|
||||||
|
|
||||||
|
func postAlertAction(router http.Handler, session metastore.Session, path string, values url.Values) *httptest.ResponseRecorder {
|
||||||
|
if values == nil {
|
||||||
|
values = url.Values{}
|
||||||
|
}
|
||||||
|
values.Set("csrf_token", session.CSRFToken)
|
||||||
|
request := httptest.NewRequest(http.MethodPost, path, strings.NewReader(values.Encode()))
|
||||||
|
request.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
return response
|
||||||
|
}
|
||||||
174
lib/server/account_auth.go
Normal file
174
lib/server/account_auth.go
Normal file
@@ -0,0 +1,174 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
const accountSessionCookie = "warpbox_account_session"
|
||||||
|
|
||||||
|
func (app *App) registerAccountRoutes(router *gin.Engine) {
|
||||||
|
account := router.Group("/account")
|
||||||
|
account.Use(noStoreAdminHeaders)
|
||||||
|
account.GET("/login", app.handleAccountLogin)
|
||||||
|
account.POST("/login", app.handleAccountLoginPost)
|
||||||
|
|
||||||
|
protected := account.Group("")
|
||||||
|
protected.Use(app.requireAccountSession)
|
||||||
|
protected.GET("", app.handleAccountDashboard)
|
||||||
|
protected.GET("/", app.handleAccountDashboard)
|
||||||
|
protected.POST("/logout", app.handleAccountLogout)
|
||||||
|
protected.GET("/settings", app.handleAccountSettings)
|
||||||
|
protected.POST("/settings", app.handleAccountSettingsPost)
|
||||||
|
protected.POST("/settings/reset", app.handleAccountSettingsReset)
|
||||||
|
protected.GET("/settings/export.json", app.handleAccountSettingsExport)
|
||||||
|
protected.POST("/settings/import.json", app.handleAccountSettingsImport)
|
||||||
|
protected.GET("/alerts", app.handleAccountAlerts)
|
||||||
|
protected.GET("/alerts/export.json", app.handleAccountAlertsExport)
|
||||||
|
protected.POST("/alerts/bulk/acknowledge", app.handleAccountAlertBulkAcknowledge)
|
||||||
|
protected.POST("/alerts/bulk/close", app.handleAccountAlertBulkClose)
|
||||||
|
protected.POST("/alerts/:id/acknowledge", app.handleAccountAlertAcknowledge)
|
||||||
|
protected.POST("/alerts/:id/close", app.handleAccountAlertClose)
|
||||||
|
protected.GET("/boxes", app.handleAccountBoxes)
|
||||||
|
protected.GET("/boxes/export.csv", app.handleAccountBoxesExport)
|
||||||
|
protected.POST("/boxes/bulk/expire", app.handleAccountBoxesBulkExpire)
|
||||||
|
protected.POST("/boxes/bulk/delete", app.handleAccountBoxesBulkDelete)
|
||||||
|
protected.POST("/boxes/bulk/bump-expiry", app.handleAccountBoxesBulkBumpExpiry)
|
||||||
|
protected.POST("/boxes/delete-largest", app.handleAccountBoxesDeleteLargest)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountLogin(ctx *gin.Context) {
|
||||||
|
if app.isAccountSessionValid(ctx) {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
app.renderAccountLogin(ctx, "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountLoginPost(ctx *gin.Context) {
|
||||||
|
if !app.adminLoginEnabled {
|
||||||
|
app.renderAccountLogin(ctx, "Account login is disabled.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
username := strings.TrimSpace(ctx.PostForm("username"))
|
||||||
|
password := ctx.PostForm("password")
|
||||||
|
user, ok, err := app.store.GetUserByUsername(username)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not load user")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !ok || user.Disabled || !metastore.VerifyPassword(user.PasswordHash, password) {
|
||||||
|
app.renderAccountLogin(ctx, "The username or password was not accepted.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if _, err := app.permissionsForUser(user); err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not load permissions")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
session, err := app.store.CreateSession(user.ID, time.Duration(app.config.SessionTTLSeconds)*time.Second)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not create session")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.SetSameSite(http.SameSiteLaxMode)
|
||||||
|
ctx.SetCookie(accountSessionCookie, session.Token, int(app.config.SessionTTLSeconds), "/account", "", app.config.AdminCookieSecure, true)
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountLogout(ctx *gin.Context) {
|
||||||
|
if token, err := ctx.Cookie(accountSessionCookie); err == nil {
|
||||||
|
_ = app.store.DeleteSession(token)
|
||||||
|
}
|
||||||
|
ctx.SetSameSite(http.SameSiteLaxMode)
|
||||||
|
ctx.SetCookie(accountSessionCookie, "", -1, "/account", "", app.config.AdminCookieSecure, true)
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) requireAccountSession(ctx *gin.Context) {
|
||||||
|
token, err := ctx.Cookie(accountSessionCookie)
|
||||||
|
if err != nil {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
session, ok, err := app.store.GetSession(token)
|
||||||
|
if err != nil || !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !validAdminCSRF(ctx, session) {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
user, ok, err := app.store.GetUser(session.UserID)
|
||||||
|
if err != nil || !ok || user.Disabled {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
perms, err := app.permissionsForUser(user)
|
||||||
|
if err != nil {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx.Set("accountUser", user)
|
||||||
|
ctx.Set("adminUser", user)
|
||||||
|
ctx.Set("accountPerms", perms)
|
||||||
|
ctx.Set("adminPerms", perms)
|
||||||
|
ctx.Set("accountSession", session)
|
||||||
|
ctx.Set("accountCSRFToken", session.CSRFToken)
|
||||||
|
ctx.Set("adminCSRFToken", session.CSRFToken)
|
||||||
|
ctx.Next()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) isAccountSessionValid(ctx *gin.Context) bool {
|
||||||
|
token, err := ctx.Cookie(accountSessionCookie)
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
session, ok, err := app.store.GetSession(token)
|
||||||
|
if err != nil || !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
user, ok, err := app.store.GetUser(session.UserID)
|
||||||
|
if err != nil || !ok || user.Disabled {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
_, err = app.permissionsForUser(user)
|
||||||
|
return err == nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) renderAccountLogin(ctx *gin.Context, errorMessage string) {
|
||||||
|
ctx.HTML(http.StatusOK, "account_login.html", gin.H{
|
||||||
|
"PageTitle": "WarpBox Account Login",
|
||||||
|
"AdminLoginEnabled": app.adminLoginEnabled,
|
||||||
|
"AccountLoginEnabled": app.adminLoginEnabled,
|
||||||
|
"Error": errorMessage,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func currentAccountUser(ctx *gin.Context) (metastore.User, bool) {
|
||||||
|
if current, ok := ctx.Get("accountUser"); ok {
|
||||||
|
if user, ok := current.(metastore.User); ok {
|
||||||
|
return user, true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if current, ok := ctx.Get("adminUser"); ok {
|
||||||
|
if user, ok := current.(metastore.User); ok {
|
||||||
|
return user, true
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return metastore.User{}, false
|
||||||
|
}
|
||||||
454
lib/server/account_boxes.go
Normal file
454
lib/server/account_boxes.go
Normal file
@@ -0,0 +1,454 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"bytes"
|
||||||
|
"encoding/csv"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/boxstore"
|
||||||
|
"warpbox/lib/helpers"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
"warpbox/lib/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
type BoxIndexView struct {
|
||||||
|
PageTitle string
|
||||||
|
WindowTitle string
|
||||||
|
WindowIcon string
|
||||||
|
AccountNav AccountNavView
|
||||||
|
CSRFToken string
|
||||||
|
Filters BoxFiltersView
|
||||||
|
Rows []BoxRowView
|
||||||
|
Stats BoxIndexStats
|
||||||
|
Page int
|
||||||
|
PageSize int
|
||||||
|
Total int
|
||||||
|
TotalPages int
|
||||||
|
HasPrev bool
|
||||||
|
HasNext bool
|
||||||
|
PrevURL string
|
||||||
|
NextURL string
|
||||||
|
CanManage bool
|
||||||
|
PolicySummary string
|
||||||
|
Error string
|
||||||
|
}
|
||||||
|
|
||||||
|
type BoxFiltersView struct {
|
||||||
|
Query string
|
||||||
|
Owner string
|
||||||
|
Status string
|
||||||
|
Flag string
|
||||||
|
Sort string
|
||||||
|
PageSize int
|
||||||
|
}
|
||||||
|
|
||||||
|
type BoxIndexStats struct {
|
||||||
|
Visible int
|
||||||
|
Total int
|
||||||
|
Expired int
|
||||||
|
Storage string
|
||||||
|
}
|
||||||
|
|
||||||
|
type BoxRowView struct {
|
||||||
|
ID string
|
||||||
|
Owner string
|
||||||
|
Status string
|
||||||
|
FileCount int
|
||||||
|
Size string
|
||||||
|
CreatedAt string
|
||||||
|
ExpiresAt string
|
||||||
|
Flags string
|
||||||
|
Policy string
|
||||||
|
CanManage bool
|
||||||
|
ManageURL string
|
||||||
|
OpenURL string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountBoxes(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
view, err := app.ListBoxes(ctx, actor, boxFiltersFromRequest(ctx), boxPageFromRequest(ctx))
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.HTML(http.StatusOK, "account_boxes.html", view)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountBoxesBulkExpire(ctx *gin.Context) {
|
||||||
|
app.handleAccountBoxesBulkAction(ctx, app.ExpireBoxes)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountBoxesBulkDelete(ctx *gin.Context) {
|
||||||
|
app.handleAccountBoxesBulkAction(ctx, app.DeleteBoxes)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountBoxesBulkBumpExpiry(ctx *gin.Context) {
|
||||||
|
app.handleAccountBoxesBulkAction(ctx, func(ctx *gin.Context, actor metastore.User, ids []string) error {
|
||||||
|
seconds := parsePositiveInt64Default(ctx.PostForm("bump_seconds"), app.config.BoxOwnerMaxRefreshAmountSeconds)
|
||||||
|
return app.BumpBoxExpiries(ctx, actor, ids, seconds)
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountBoxesDeleteLargest(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
filters := boxFiltersFromRequest(ctx)
|
||||||
|
filters.Sort = "largest"
|
||||||
|
page := metastore.BoxPageRequest{Page: 1, PageSize: 25}
|
||||||
|
boxPage, err := app.visibleBoxRecords(ctx, actor, filters, page)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ids := make([]string, 0, 10)
|
||||||
|
for _, row := range boxPage.Rows {
|
||||||
|
if len(ids) == 10 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
ids = append(ids, row.ID)
|
||||||
|
}
|
||||||
|
if err := app.DeleteBoxes(ctx, actor, ids); err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/boxes")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountBoxesExport(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
page, err := app.visibleBoxRecords(ctx, actor, boxFiltersFromRequest(ctx), metastore.BoxPageRequest{Page: 1, PageSize: 100})
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
var buffer bytes.Buffer
|
||||||
|
writer := csv.NewWriter(&buffer)
|
||||||
|
_ = writer.Write([]string{"id", "owner", "status", "file_count", "total_size", "created_at", "expires_at", "flags"})
|
||||||
|
for _, record := range page.Rows {
|
||||||
|
_ = writer.Write([]string{record.ID, record.OwnerUsername, boxStatus(record), strconv.Itoa(record.FileCount), strconv.FormatInt(record.TotalSize, 10), record.CreatedAt.Format(time.RFC3339), record.ExpiresAt.Format(time.RFC3339), boxFlags(record)})
|
||||||
|
}
|
||||||
|
writer.Flush()
|
||||||
|
ctx.Header("Content-Disposition", `attachment; filename="warpbox-boxes.csv"`)
|
||||||
|
ctx.Data(http.StatusOK, "text/csv; charset=utf-8", buffer.Bytes())
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountBoxesBulkAction(ctx *gin.Context, action func(*gin.Context, metastore.User, []string) error) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if err := action(ctx, actor, ctx.PostFormArray("box_ids")); err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/boxes")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) ListBoxes(ctx *gin.Context, actor metastore.User, filters metastore.BoxFilters, page metastore.BoxPageRequest) (BoxIndexView, error) {
|
||||||
|
boxPage, err := app.visibleBoxRecords(ctx, actor, filters, page)
|
||||||
|
if err != nil {
|
||||||
|
return BoxIndexView{}, err
|
||||||
|
}
|
||||||
|
rows := make([]BoxRowView, 0, len(boxPage.Rows))
|
||||||
|
stats := BoxIndexStats{Visible: len(boxPage.Rows), Total: boxPage.Total}
|
||||||
|
totalSize := int64(0)
|
||||||
|
for _, record := range boxPage.Rows {
|
||||||
|
totalSize += record.TotalSize
|
||||||
|
if boxExpired(record) {
|
||||||
|
stats.Expired++
|
||||||
|
}
|
||||||
|
rows = append(rows, app.boxRowView(ctx, actor, record))
|
||||||
|
}
|
||||||
|
stats.Storage = helpers.FormatBytes(totalSize)
|
||||||
|
nav := app.accountNavView(ctx, "boxes")
|
||||||
|
nav.AlertCount, nav.AlertSeverity = app.openAlertSummary()
|
||||||
|
return BoxIndexView{
|
||||||
|
PageTitle: "WarpBox Boxes",
|
||||||
|
WindowTitle: "WarpBox Boxes",
|
||||||
|
WindowIcon: "B",
|
||||||
|
AccountNav: nav,
|
||||||
|
CSRFToken: app.currentCSRFToken(ctx),
|
||||||
|
Filters: BoxFiltersView{Query: filters.Query, Owner: filters.Owner, Status: filters.Status, Flag: filters.Flag, Sort: filters.Sort, PageSize: boxPage.PageSize},
|
||||||
|
Rows: rows,
|
||||||
|
Stats: stats,
|
||||||
|
Page: boxPage.Page,
|
||||||
|
PageSize: boxPage.PageSize,
|
||||||
|
Total: boxPage.Total,
|
||||||
|
TotalPages: boxPage.TotalPages,
|
||||||
|
HasPrev: boxPage.HasPrev,
|
||||||
|
HasNext: boxPage.HasNext,
|
||||||
|
PrevURL: boxPageURL(ctx, boxPage.PrevPage),
|
||||||
|
NextURL: boxPageURL(ctx, boxPage.NextPage),
|
||||||
|
CanManage: currentAccountPermissions(ctx).AdminBoxesView,
|
||||||
|
PolicySummary: app.boxPolicySummary(),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) ExpireBoxes(ctx *gin.Context, actor metastore.User, ids []string) error {
|
||||||
|
records, err := app.authorizedBoxRecords(ctx, actor, ids)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
now := time.Now().UTC().Add(-time.Second)
|
||||||
|
for _, record := range records {
|
||||||
|
manifest, err := boxstore.ReadManifest(record.ID)
|
||||||
|
if err == nil {
|
||||||
|
manifest.ExpiresAt = now
|
||||||
|
_ = boxstore.WriteManifest(record.ID, manifest)
|
||||||
|
}
|
||||||
|
record.ExpiresAt = now
|
||||||
|
if err := app.store.UpsertBoxRecord(record); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) DeleteBoxes(ctx *gin.Context, actor metastore.User, ids []string) error {
|
||||||
|
records, err := app.authorizedBoxRecords(ctx, actor, ids)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, record := range records {
|
||||||
|
if err := boxstore.DeleteBox(record.ID); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := app.store.DeleteBoxRecord(record.ID); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) BumpBoxExpiries(ctx *gin.Context, actor metastore.User, ids []string, seconds int64) error {
|
||||||
|
if seconds <= 0 {
|
||||||
|
return fmt.Errorf("bump expiry requires a positive duration")
|
||||||
|
}
|
||||||
|
if !app.config.BoxOwnerRefreshEnabled {
|
||||||
|
return fmt.Errorf("box owner refresh policy is disabled")
|
||||||
|
}
|
||||||
|
if app.config.BoxOwnerMaxRefreshAmountSeconds > 0 && seconds > app.config.BoxOwnerMaxRefreshAmountSeconds {
|
||||||
|
return fmt.Errorf("bump expiry exceeds maximum refresh amount")
|
||||||
|
}
|
||||||
|
records, err := app.authorizedBoxRecords(ctx, actor, ids)
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for _, record := range records {
|
||||||
|
if record.OneTimeDownload {
|
||||||
|
return fmt.Errorf("one-time boxes cannot be refreshed")
|
||||||
|
}
|
||||||
|
if app.config.BoxOwnerMaxRefreshCount > 0 && record.RefreshCount >= app.config.BoxOwnerMaxRefreshCount {
|
||||||
|
return fmt.Errorf("box refresh count limit reached")
|
||||||
|
}
|
||||||
|
base := record.ExpiresAt
|
||||||
|
if base.IsZero() || time.Now().UTC().After(base) {
|
||||||
|
base = time.Now().UTC()
|
||||||
|
}
|
||||||
|
newExpiry := base.Add(time.Duration(seconds) * time.Second)
|
||||||
|
if app.config.BoxOwnerMaxTotalExpirySeconds > 0 && !record.CreatedAt.IsZero() && newExpiry.After(record.CreatedAt.Add(time.Duration(app.config.BoxOwnerMaxTotalExpirySeconds)*time.Second)) {
|
||||||
|
return fmt.Errorf("bump expiry exceeds maximum total expiry")
|
||||||
|
}
|
||||||
|
manifest, err := boxstore.ReadManifest(record.ID)
|
||||||
|
if err == nil {
|
||||||
|
manifest.ExpiresAt = newExpiry
|
||||||
|
_ = boxstore.WriteManifest(record.ID, manifest)
|
||||||
|
}
|
||||||
|
record.ExpiresAt = newExpiry
|
||||||
|
record.RefreshCount++
|
||||||
|
if err := app.store.UpsertBoxRecord(record); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) visibleBoxRecords(ctx *gin.Context, actor metastore.User, filters metastore.BoxFilters, page metastore.BoxPageRequest) (metastore.BoxRecordPage, error) {
|
||||||
|
perms := currentAccountPermissions(ctx)
|
||||||
|
if !perms.AdminBoxesView {
|
||||||
|
filters.Owner = actor.ID
|
||||||
|
}
|
||||||
|
return app.store.ListBoxRecords(filters, page)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) authorizedBoxRecords(ctx *gin.Context, actor metastore.User, ids []string) ([]metastore.BoxRecord, error) {
|
||||||
|
ids = uniqueNonEmpty(ids)
|
||||||
|
if len(ids) == 0 {
|
||||||
|
return nil, fmt.Errorf("no boxes selected")
|
||||||
|
}
|
||||||
|
perms := currentAccountPermissions(ctx)
|
||||||
|
records := make([]metastore.BoxRecord, 0, len(ids))
|
||||||
|
for _, id := range ids {
|
||||||
|
record, ok, err := app.store.GetBoxRecord(id)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
if !ok {
|
||||||
|
return nil, fmt.Errorf("box %s not found", id)
|
||||||
|
}
|
||||||
|
if !perms.AdminBoxesView && record.OwnerID != actor.ID {
|
||||||
|
return nil, fmt.Errorf("permission denied")
|
||||||
|
}
|
||||||
|
if !perms.AdminBoxesView && !app.config.BoxOwnerEditEnabled {
|
||||||
|
return nil, fmt.Errorf("box owner edit policy is disabled")
|
||||||
|
}
|
||||||
|
records = append(records, record)
|
||||||
|
}
|
||||||
|
return records, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) boxRowView(ctx *gin.Context, actor metastore.User, record metastore.BoxRecord) BoxRowView {
|
||||||
|
owner := record.OwnerUsername
|
||||||
|
if owner == "" {
|
||||||
|
owner = "guest"
|
||||||
|
}
|
||||||
|
return BoxRowView{
|
||||||
|
ID: record.ID,
|
||||||
|
Owner: owner,
|
||||||
|
Status: boxStatus(record),
|
||||||
|
FileCount: record.FileCount,
|
||||||
|
Size: helpers.FormatBytes(record.TotalSize),
|
||||||
|
CreatedAt: formatAdminTime(record.CreatedAt),
|
||||||
|
ExpiresAt: formatAdminTime(record.ExpiresAt),
|
||||||
|
Flags: boxFlags(record),
|
||||||
|
Policy: app.boxRecordPolicy(record),
|
||||||
|
CanManage: currentAccountPermissions(ctx).AdminBoxesView || record.OwnerID == actor.ID,
|
||||||
|
ManageURL: "/account/boxes/" + record.ID,
|
||||||
|
OpenURL: "/box/" + record.ID,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) indexBoxFromManifest(boxID string) {
|
||||||
|
manifest, err := boxstore.ReadManifest(boxID)
|
||||||
|
if err != nil {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
_ = app.store.UpsertBoxRecord(boxRecordFromManifest(boxID, manifest))
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxRecordFromManifest(boxID string, manifest models.BoxManifest) metastore.BoxRecord {
|
||||||
|
total := int64(0)
|
||||||
|
names := make([]string, 0, len(manifest.Files))
|
||||||
|
for _, file := range manifest.Files {
|
||||||
|
total += file.Size
|
||||||
|
names = append(names, file.Name)
|
||||||
|
}
|
||||||
|
return metastore.BoxRecord{
|
||||||
|
ID: boxID,
|
||||||
|
OwnerID: manifest.OwnerID,
|
||||||
|
OwnerUsername: manifest.OwnerUsername,
|
||||||
|
FileNames: names,
|
||||||
|
FileCount: len(manifest.Files),
|
||||||
|
TotalSize: total,
|
||||||
|
CreatedAt: manifest.CreatedAt,
|
||||||
|
ExpiresAt: manifest.ExpiresAt,
|
||||||
|
PasswordProtected: boxstore.IsPasswordProtected(manifest),
|
||||||
|
OneTimeDownload: manifest.OneTimeDownload,
|
||||||
|
DisableZip: manifest.DisableZip,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxFiltersFromRequest(ctx *gin.Context) metastore.BoxFilters {
|
||||||
|
return metastore.BoxFilters{
|
||||||
|
Query: strings.TrimSpace(ctx.Query("q")),
|
||||||
|
Owner: emptyAsAll(ctx.Query("owner")),
|
||||||
|
Status: emptyAsAll(ctx.Query("status")),
|
||||||
|
Flag: emptyAsAll(ctx.Query("flag")),
|
||||||
|
Sort: emptyAsNewest(ctx.Query("sort")),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxPageFromRequest(ctx *gin.Context) metastore.BoxPageRequest {
|
||||||
|
page, _ := strconv.Atoi(ctx.DefaultQuery("page", "1"))
|
||||||
|
pageSize, _ := strconv.Atoi(ctx.DefaultQuery("page_size", "25"))
|
||||||
|
return metastore.BoxPageRequest{Page: page, PageSize: pageSize}
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxStatus(record metastore.BoxRecord) string {
|
||||||
|
if boxExpired(record) {
|
||||||
|
return "expired"
|
||||||
|
}
|
||||||
|
if record.ExpiresAt.IsZero() {
|
||||||
|
return "pending"
|
||||||
|
}
|
||||||
|
return "active"
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxExpired(record metastore.BoxRecord) bool {
|
||||||
|
return !record.ExpiresAt.IsZero() && time.Now().UTC().After(record.ExpiresAt)
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxFlags(record metastore.BoxRecord) string {
|
||||||
|
flags := []string{}
|
||||||
|
if record.PasswordProtected {
|
||||||
|
flags = append(flags, "password")
|
||||||
|
}
|
||||||
|
if record.OneTimeDownload {
|
||||||
|
flags = append(flags, "one-time")
|
||||||
|
}
|
||||||
|
if record.DisableZip {
|
||||||
|
flags = append(flags, "zip disabled")
|
||||||
|
}
|
||||||
|
if boxExpired(record) {
|
||||||
|
flags = append(flags, "expired")
|
||||||
|
}
|
||||||
|
if len(flags) == 0 {
|
||||||
|
return "normal"
|
||||||
|
}
|
||||||
|
return strings.Join(flags, ", ")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) boxRecordPolicy(record metastore.BoxRecord) string {
|
||||||
|
if record.OneTimeDownload {
|
||||||
|
return "one-time: no refresh"
|
||||||
|
}
|
||||||
|
if !app.config.BoxOwnerEditEnabled {
|
||||||
|
return "owner edits disabled"
|
||||||
|
}
|
||||||
|
if !app.config.BoxOwnerRefreshEnabled {
|
||||||
|
return "editable, no refresh"
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("editable, refresh %d/%d", record.RefreshCount, app.config.BoxOwnerMaxRefreshCount)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) boxPolicySummary() string {
|
||||||
|
if !app.config.BoxOwnerEditEnabled {
|
||||||
|
return "Owners cannot edit boxes by default."
|
||||||
|
}
|
||||||
|
if !app.config.BoxOwnerRefreshEnabled {
|
||||||
|
return "Owners can edit boxes but cannot refresh expiry."
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("Owners can edit and refresh up to %d times by %s.", app.config.BoxOwnerMaxRefreshCount, formatDurationForSettings(app.config.BoxOwnerMaxRefreshAmountSeconds))
|
||||||
|
}
|
||||||
|
|
||||||
|
func boxPageURL(ctx *gin.Context, page int) string {
|
||||||
|
query := ctx.Request.URL.Query()
|
||||||
|
query.Set("page", strconv.Itoa(page))
|
||||||
|
return "/account/boxes?" + query.Encode()
|
||||||
|
}
|
||||||
|
|
||||||
|
func parsePositiveInt64Default(raw string, fallback int64) int64 {
|
||||||
|
value, err := strconv.ParseInt(strings.TrimSpace(raw), 10, 64)
|
||||||
|
if err != nil || value <= 0 {
|
||||||
|
return fallback
|
||||||
|
}
|
||||||
|
return value
|
||||||
|
}
|
||||||
220
lib/server/account_boxes_test.go
Normal file
220
lib/server/account_boxes_test.go
Normal file
@@ -0,0 +1,220 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"net/url"
|
||||||
|
"os"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"warpbox/lib/boxstore"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
"warpbox/lib/models"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestAccountBoxesAdminListsBoxes(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
createIndexedBox(t, app, "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa", "", "", 10, false)
|
||||||
|
|
||||||
|
response := getAccountBoxes(router, session, "/account/boxes")
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected boxes page, got %d body=%s", response.Code, response.Body.String())
|
||||||
|
}
|
||||||
|
if !strings.Contains(response.Body.String(), "aaaaaaaaaaaaaaaaaaaaaaaaaaaaaaaa") {
|
||||||
|
t.Fatal("expected indexed box in admin list")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountBoxesRegularUserSeesOnlyOwnBoxes(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
user, err := app.store.CreateUserWithPassword("box-user", "box-user@example.test", "secret", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateUserWithPassword returned error: %v", err)
|
||||||
|
}
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
createIndexedBox(t, app, "bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb", user.ID, user.Username, 10, false)
|
||||||
|
createIndexedBox(t, app, "cccccccccccccccccccccccccccccccc", "other", "other", 20, false)
|
||||||
|
|
||||||
|
response := getAccountBoxes(router, session, "/account/boxes")
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected boxes page, got %d", response.Code)
|
||||||
|
}
|
||||||
|
body := response.Body.String()
|
||||||
|
if !strings.Contains(body, "bbbbbbbbbbbbbbbbbbbbbbbbbbbbbbbb") {
|
||||||
|
t.Fatal("expected own box")
|
||||||
|
}
|
||||||
|
if strings.Contains(body, "cccccccccccccccccccccccccccccccc") {
|
||||||
|
t.Fatal("did not expect other user's box")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountBoxesFiltersSortAndPagination(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
createIndexedBox(t, app, "11111111111111111111111111111111", "", "", 10, false)
|
||||||
|
createIndexedBox(t, app, "22222222222222222222222222222222", "", "", 1000, true)
|
||||||
|
createIndexedBox(t, app, "33333333333333333333333333333333", "", "", 500, false)
|
||||||
|
|
||||||
|
response := getAccountBoxes(router, session, "/account/boxes?flag=password&sort=largest&page_size=25")
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected boxes page, got %d", response.Code)
|
||||||
|
}
|
||||||
|
body := response.Body.String()
|
||||||
|
if !strings.Contains(body, "22222222222222222222222222222222") {
|
||||||
|
t.Fatal("expected password filtered box")
|
||||||
|
}
|
||||||
|
if strings.Contains(body, "11111111111111111111111111111111") {
|
||||||
|
t.Fatal("did not expect unfiltered box")
|
||||||
|
}
|
||||||
|
|
||||||
|
page, err := app.store.ListBoxRecords(metastore.BoxFilters{Sort: "largest"}, metastore.BoxPageRequest{Page: 1, PageSize: 25})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("ListBoxRecords returned error: %v", err)
|
||||||
|
}
|
||||||
|
if len(page.Rows) != 3 || page.Rows[0].ID != "22222222222222222222222222222222" {
|
||||||
|
t.Fatalf("expected largest sort first, got %#v", page.Rows)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountBoxesBulkExpireAndDelete(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
id := "dddddddddddddddddddddddddddddddd"
|
||||||
|
createIndexedBox(t, app, id, "", "", 10, false)
|
||||||
|
|
||||||
|
values := url.Values{"box_ids": []string{id}}
|
||||||
|
response := postAccountBoxesForm(router, session, "/account/boxes/bulk/expire", values)
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected expire redirect, got %d", response.Code)
|
||||||
|
}
|
||||||
|
record, ok, err := app.store.GetBoxRecord(id)
|
||||||
|
if err != nil || !ok {
|
||||||
|
t.Fatalf("GetBoxRecord returned ok=%v err=%v", ok, err)
|
||||||
|
}
|
||||||
|
if record.ExpiresAt.After(time.Now().UTC()) {
|
||||||
|
t.Fatal("expected box to be expired")
|
||||||
|
}
|
||||||
|
|
||||||
|
response = postAccountBoxesForm(router, session, "/account/boxes/bulk/delete", values)
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected delete redirect, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if _, ok, err := app.store.GetBoxRecord(id); err != nil || ok {
|
||||||
|
t.Fatalf("expected deleted record, ok=%v err=%v", ok, err)
|
||||||
|
}
|
||||||
|
if _, err := os.Stat(boxstore.BoxPath(id)); !os.IsNotExist(err) {
|
||||||
|
t.Fatalf("expected box directory deleted, stat err=%v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountBoxesBulkDeletePermissionDenied(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
user, err := app.store.CreateUserWithPassword("box-limited", "box-limited@example.test", "secret", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateUserWithPassword returned error: %v", err)
|
||||||
|
}
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
id := "eeeeeeeeeeeeeeeeeeeeeeeeeeeeeeee"
|
||||||
|
createIndexedBox(t, app, id, "other", "other", 10, false)
|
||||||
|
|
||||||
|
response := postAccountBoxesForm(router, session, "/account/boxes/bulk/delete", url.Values{"box_ids": []string{id}})
|
||||||
|
if response.Code != http.StatusForbidden {
|
||||||
|
t.Fatalf("expected permission denied, got %d", response.Code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountBoxesBumpExpiryPolicyRejection(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
app.config.BoxOwnerRefreshEnabled = false
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
id := "ffffffffffffffffffffffffffffffff"
|
||||||
|
createIndexedBox(t, app, id, "", "", 10, false)
|
||||||
|
|
||||||
|
response := postAccountBoxesForm(router, session, "/account/boxes/bulk/bump-expiry", url.Values{"box_ids": []string{id}, "bump_seconds": []string{"60"}})
|
||||||
|
if response.Code != http.StatusForbidden {
|
||||||
|
t.Fatalf("expected policy rejection, got %d", response.Code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountBoxesDeleteLargest(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
small := "12345123451234512345123451234512"
|
||||||
|
large := "99999999999999999999999999999999"
|
||||||
|
createIndexedBox(t, app, small, "", "", 10, false)
|
||||||
|
createIndexedBox(t, app, large, "", "", 1000, false)
|
||||||
|
|
||||||
|
response := postAccountBoxesForm(router, session, "/account/boxes/delete-largest", nil)
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected delete-largest redirect, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if _, ok, err := app.store.GetBoxRecord(large); err != nil || ok {
|
||||||
|
t.Fatalf("expected largest deleted, ok=%v err=%v", ok, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func createIndexedBox(t *testing.T, app *App, id string, ownerID string, ownerUsername string, size int64, password bool) {
|
||||||
|
t.Helper()
|
||||||
|
if err := os.MkdirAll(boxstore.BoxPath(id), 0755); err != nil {
|
||||||
|
t.Fatalf("MkdirAll returned error: %v", err)
|
||||||
|
}
|
||||||
|
filename := "file-" + id[:4] + ".txt"
|
||||||
|
if err := os.WriteFile(filepath.Join(boxstore.BoxPath(id), filename), []byte(strings.Repeat("x", int(size))), 0644); err != nil {
|
||||||
|
t.Fatalf("WriteFile returned error: %v", err)
|
||||||
|
}
|
||||||
|
manifest := models.BoxManifest{
|
||||||
|
OwnerID: ownerID,
|
||||||
|
OwnerUsername: ownerUsername,
|
||||||
|
Files: []models.BoxFile{{
|
||||||
|
ID: "abcdabcdabcdabcd",
|
||||||
|
Name: filename,
|
||||||
|
Size: size,
|
||||||
|
Status: models.FileStatusReady,
|
||||||
|
}},
|
||||||
|
CreatedAt: time.Now().UTC().Add(-time.Duration(size) * time.Second),
|
||||||
|
ExpiresAt: time.Now().UTC().Add(time.Hour),
|
||||||
|
RetentionSecs: 3600,
|
||||||
|
}
|
||||||
|
if password {
|
||||||
|
manifest.PasswordHash = "hash"
|
||||||
|
manifest.AuthToken = "token"
|
||||||
|
}
|
||||||
|
if err := boxstore.WriteManifest(id, manifest); err != nil {
|
||||||
|
t.Fatalf("WriteManifest returned error: %v", err)
|
||||||
|
}
|
||||||
|
if err := app.store.UpsertBoxRecord(boxRecordFromManifest(id, manifest)); err != nil {
|
||||||
|
t.Fatalf("UpsertBoxRecord returned error: %v", err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func getAccountBoxes(router http.Handler, session metastore.Session, path string) *httptest.ResponseRecorder {
|
||||||
|
request := httptest.NewRequest(http.MethodGet, path, nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
return response
|
||||||
|
}
|
||||||
|
|
||||||
|
func postAccountBoxesForm(router http.Handler, session metastore.Session, path string, values url.Values) *httptest.ResponseRecorder {
|
||||||
|
if values == nil {
|
||||||
|
values = url.Values{}
|
||||||
|
}
|
||||||
|
values.Set("csrf_token", session.CSRFToken)
|
||||||
|
request := httptest.NewRequest(http.MethodPost, path, strings.NewReader(values.Encode()))
|
||||||
|
request.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
return response
|
||||||
|
}
|
||||||
61
lib/server/account_nav.go
Normal file
61
lib/server/account_nav.go
Normal file
@@ -0,0 +1,61 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AccountNavView struct {
|
||||||
|
Username string
|
||||||
|
IsAdmin bool
|
||||||
|
ActiveSection string
|
||||||
|
AlertCount int
|
||||||
|
AlertSeverity string
|
||||||
|
CanViewBoxes bool
|
||||||
|
CanViewAlerts bool
|
||||||
|
CanViewUsers bool
|
||||||
|
CanViewAPIKeys bool
|
||||||
|
CanViewSettings bool
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) accountNavView(ctx *gin.Context, activeSection string) AccountNavView {
|
||||||
|
perms := currentAccountPermissions(ctx)
|
||||||
|
isAdmin := perms.AdminAccess
|
||||||
|
|
||||||
|
return AccountNavView{
|
||||||
|
Username: app.currentAdminUsername(ctx),
|
||||||
|
IsAdmin: isAdmin,
|
||||||
|
ActiveSection: activeSection,
|
||||||
|
AlertSeverity: "ok",
|
||||||
|
CanViewBoxes: true,
|
||||||
|
CanViewAlerts: true,
|
||||||
|
CanViewUsers: perms.AdminUsersManage,
|
||||||
|
CanViewAPIKeys: true,
|
||||||
|
CanViewSettings: perms.AdminSettingsManage,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func currentAccountPermissions(ctx *gin.Context) metastore.EffectivePermissions {
|
||||||
|
value, ok := ctx.Get("adminPerms")
|
||||||
|
if !ok {
|
||||||
|
return metastore.EffectivePermissions{}
|
||||||
|
}
|
||||||
|
perms, ok := value.(metastore.EffectivePermissions)
|
||||||
|
if !ok {
|
||||||
|
return metastore.EffectivePermissions{}
|
||||||
|
}
|
||||||
|
return perms
|
||||||
|
}
|
||||||
|
|
||||||
|
func normalizeAlertSeverity(severity string) string {
|
||||||
|
normalized := strings.ToLower(strings.TrimSpace(severity))
|
||||||
|
switch normalized {
|
||||||
|
case "danger", "warning", "info", "ok":
|
||||||
|
return normalized
|
||||||
|
default:
|
||||||
|
return "ok"
|
||||||
|
}
|
||||||
|
}
|
||||||
253
lib/server/account_pages.go
Normal file
253
lib/server/account_pages.go
Normal file
@@ -0,0 +1,253 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/boxstore"
|
||||||
|
"warpbox/lib/helpers"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
type AccountDashboardView struct {
|
||||||
|
PageTitle string
|
||||||
|
WindowTitle string
|
||||||
|
WindowIcon string
|
||||||
|
PageScripts []string
|
||||||
|
AccountNav AccountNavView
|
||||||
|
CSRFToken string
|
||||||
|
Stats AccountDashboardStats
|
||||||
|
Statuses []accountStatusRow
|
||||||
|
Alerts []accountAlertPreviewRow
|
||||||
|
RecentBoxes []accountDashboardBoxRow
|
||||||
|
RecentActivity []accountActivityRow
|
||||||
|
ShowUsersStat bool
|
||||||
|
CanManageBoxes bool
|
||||||
|
CanManageUsers bool
|
||||||
|
CanViewSettings bool
|
||||||
|
HasAlertsPreview bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type AccountDashboardStats struct {
|
||||||
|
ActiveBoxes int
|
||||||
|
StorageUsedLabel string
|
||||||
|
AlertCount int
|
||||||
|
TotalUsers int
|
||||||
|
ActiveUsers int
|
||||||
|
DisabledUsers int
|
||||||
|
}
|
||||||
|
|
||||||
|
type accountStatusRow struct {
|
||||||
|
Label string
|
||||||
|
Value string
|
||||||
|
Severity string
|
||||||
|
}
|
||||||
|
|
||||||
|
type accountAlertPreviewRow struct {
|
||||||
|
Severity string
|
||||||
|
Title string
|
||||||
|
Detail string
|
||||||
|
}
|
||||||
|
|
||||||
|
type accountDashboardBoxRow struct {
|
||||||
|
ID string
|
||||||
|
FileCount int
|
||||||
|
TotalSizeLabel string
|
||||||
|
CreatedAt string
|
||||||
|
ExpiresAt string
|
||||||
|
Flags string
|
||||||
|
CanManage bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type accountActivityRow struct {
|
||||||
|
Time string
|
||||||
|
Title string
|
||||||
|
Meta string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountDashboard(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
view, err := app.GetAccountDashboard(ctx, actor)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not load account dashboard")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.HTML(http.StatusOK, "account_dashboard.html", view)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) GetAccountDashboard(ctx *gin.Context, actor metastore.User) (AccountDashboardView, error) {
|
||||||
|
perms := currentAccountPermissions(ctx)
|
||||||
|
nav := app.accountNavView(ctx, "dashboard")
|
||||||
|
|
||||||
|
totalSize := int64(0)
|
||||||
|
activeBoxes := 0
|
||||||
|
recentBoxes := []accountDashboardBoxRow{}
|
||||||
|
if perms.AdminBoxesView {
|
||||||
|
summaries, err := boxstore.ListBoxSummaries()
|
||||||
|
if err != nil {
|
||||||
|
return AccountDashboardView{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
recentBoxes = make([]accountDashboardBoxRow, 0, minInt(len(summaries), 10))
|
||||||
|
for _, summary := range summaries {
|
||||||
|
totalSize += summary.TotalSize
|
||||||
|
if !summary.Expired {
|
||||||
|
activeBoxes++
|
||||||
|
}
|
||||||
|
if len(recentBoxes) < 10 {
|
||||||
|
recentBoxes = append(recentBoxes, accountDashboardBoxRow{
|
||||||
|
ID: summary.ID,
|
||||||
|
FileCount: summary.FileCount,
|
||||||
|
TotalSizeLabel: summary.TotalSizeLabel,
|
||||||
|
CreatedAt: formatAdminTime(summary.CreatedAt),
|
||||||
|
ExpiresAt: formatAdminTime(summary.ExpiresAt),
|
||||||
|
Flags: accountBoxFlags(summary.Expired, summary.OneTimeDownload, summary.PasswordProtected),
|
||||||
|
CanManage: true,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stats := AccountDashboardStats{
|
||||||
|
ActiveBoxes: activeBoxes,
|
||||||
|
StorageUsedLabel: helpers.FormatBytes(totalSize),
|
||||||
|
}
|
||||||
|
alertPreview := []accountAlertPreviewRow{}
|
||||||
|
if perms.AdminAccess {
|
||||||
|
stats.AlertCount, nav.AlertSeverity = app.openAlertSummary()
|
||||||
|
nav.AlertCount = stats.AlertCount
|
||||||
|
alertPreview = app.accountDashboardAlertPreview()
|
||||||
|
}
|
||||||
|
|
||||||
|
showUsersStat := perms.AdminUsersManage
|
||||||
|
if showUsersStat {
|
||||||
|
users, err := app.store.ListUsers()
|
||||||
|
if err != nil {
|
||||||
|
return AccountDashboardView{}, err
|
||||||
|
}
|
||||||
|
stats.TotalUsers = len(users)
|
||||||
|
for _, user := range users {
|
||||||
|
if user.Disabled {
|
||||||
|
stats.DisabledUsers++
|
||||||
|
} else {
|
||||||
|
stats.ActiveUsers++
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return AccountDashboardView{
|
||||||
|
PageTitle: "WarpBox Account",
|
||||||
|
WindowTitle: "WarpBox Account Control Panel",
|
||||||
|
WindowIcon: "W",
|
||||||
|
AccountNav: nav,
|
||||||
|
CSRFToken: app.currentCSRFToken(ctx),
|
||||||
|
Stats: stats,
|
||||||
|
Statuses: app.accountDashboardStatuses(),
|
||||||
|
Alerts: alertPreview,
|
||||||
|
RecentBoxes: recentBoxes,
|
||||||
|
RecentActivity: accountPlaceholderActivity(actor, ctx),
|
||||||
|
ShowUsersStat: showUsersStat,
|
||||||
|
CanManageBoxes: perms.AdminBoxesView,
|
||||||
|
CanManageUsers: perms.AdminUsersManage,
|
||||||
|
CanViewSettings: perms.AdminSettingsManage,
|
||||||
|
HasAlertsPreview: true,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) accountDashboardStatuses() []accountStatusRow {
|
||||||
|
return []accountStatusRow{
|
||||||
|
{Label: "Guest uploads", Value: enabledLabel(app.config.GuestUploadsEnabled), Severity: boolSeverity(app.config.GuestUploadsEnabled)},
|
||||||
|
{Label: "API", Value: enabledLabel(app.config.APIEnabled), Severity: boolSeverity(app.config.APIEnabled)},
|
||||||
|
{Label: "ZIP downloads", Value: enabledLabel(app.config.ZipDownloadsEnabled), Severity: boolSeverity(app.config.ZipDownloadsEnabled)},
|
||||||
|
{Label: "One-time boxes", Value: enabledLabel(app.config.OneTimeDownloadsEnabled), Severity: boolSeverity(app.config.OneTimeDownloadsEnabled)},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) accountDashboardAlertPreview() []accountAlertPreviewRow {
|
||||||
|
alerts, err := app.store.ListAlerts(metastore.AlertFilters{Status: metastore.AlertStatusOpen, Sort: "severity"})
|
||||||
|
if err != nil {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
rows := make([]accountAlertPreviewRow, 0, minInt(len(alerts), 6))
|
||||||
|
for _, alert := range alerts {
|
||||||
|
if len(rows) == 6 {
|
||||||
|
break
|
||||||
|
}
|
||||||
|
rows = append(rows, accountAlertPreviewRow{
|
||||||
|
Severity: alert.Severity,
|
||||||
|
Title: alert.Title,
|
||||||
|
Detail: alert.Description,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return rows
|
||||||
|
}
|
||||||
|
|
||||||
|
func accountPlaceholderActivity(actor metastore.User, ctx *gin.Context) []accountActivityRow {
|
||||||
|
now := time.Now().UTC()
|
||||||
|
if value, ok := ctx.Get("accountSession"); ok {
|
||||||
|
if session, ok := value.(metastore.Session); ok {
|
||||||
|
now = session.CreatedAt
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return []accountActivityRow{
|
||||||
|
{
|
||||||
|
Time: formatAdminTime(now),
|
||||||
|
Title: "Signed in",
|
||||||
|
Meta: actor.Username + " opened the account dashboard.",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
Time: "pending",
|
||||||
|
Title: "Audit log not implemented",
|
||||||
|
Meta: "Recent account activity will use the audit model in a later pass.",
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func accountBoxFlags(expired bool, oneTime bool, passwordProtected bool) string {
|
||||||
|
flags := []string{}
|
||||||
|
if expired {
|
||||||
|
flags = append(flags, "expired")
|
||||||
|
}
|
||||||
|
if oneTime {
|
||||||
|
flags = append(flags, "one-time")
|
||||||
|
}
|
||||||
|
if passwordProtected {
|
||||||
|
flags = append(flags, "password")
|
||||||
|
}
|
||||||
|
if len(flags) == 0 {
|
||||||
|
return "normal"
|
||||||
|
}
|
||||||
|
out := flags[0]
|
||||||
|
for _, flag := range flags[1:] {
|
||||||
|
out += ", " + flag
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func enabledLabel(enabled bool) string {
|
||||||
|
if enabled {
|
||||||
|
return "enabled"
|
||||||
|
}
|
||||||
|
return "disabled"
|
||||||
|
}
|
||||||
|
|
||||||
|
func boolSeverity(enabled bool) string {
|
||||||
|
if enabled {
|
||||||
|
return "ok"
|
||||||
|
}
|
||||||
|
return "warn"
|
||||||
|
}
|
||||||
|
|
||||||
|
func minInt(a int, b int) int {
|
||||||
|
if a < b {
|
||||||
|
return a
|
||||||
|
}
|
||||||
|
return b
|
||||||
|
}
|
||||||
506
lib/server/account_settings.go
Normal file
506
lib/server/account_settings.go
Normal file
@@ -0,0 +1,506 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/config"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
type SettingsView struct {
|
||||||
|
PageTitle string
|
||||||
|
WindowTitle string
|
||||||
|
WindowIcon string
|
||||||
|
PageScripts []string
|
||||||
|
AccountNav AccountNavView
|
||||||
|
CSRFToken string
|
||||||
|
Groups []SettingsGroupView
|
||||||
|
OverridesAllowed bool
|
||||||
|
CanEdit bool
|
||||||
|
Error string
|
||||||
|
Notice string
|
||||||
|
}
|
||||||
|
|
||||||
|
type SettingsGroupView struct {
|
||||||
|
Key string
|
||||||
|
Label string
|
||||||
|
Description string
|
||||||
|
Rows []SettingsRowView
|
||||||
|
}
|
||||||
|
|
||||||
|
type SettingsRowView struct {
|
||||||
|
Key string
|
||||||
|
Label string
|
||||||
|
Description string
|
||||||
|
Type config.SettingType
|
||||||
|
Value string
|
||||||
|
DisplayValue string
|
||||||
|
Source string
|
||||||
|
EnvName string
|
||||||
|
Editable bool
|
||||||
|
LockedReason string
|
||||||
|
Future bool
|
||||||
|
}
|
||||||
|
|
||||||
|
type SettingsBackup struct {
|
||||||
|
Version int `json:"version"`
|
||||||
|
ExportedAt string `json:"exported_at"`
|
||||||
|
Settings map[string]string `json:"settings"`
|
||||||
|
Metadata map[string]string `json:"metadata,omitempty"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type ImportResult struct {
|
||||||
|
Applied int `json:"applied"`
|
||||||
|
Keys []string `json:"keys"`
|
||||||
|
}
|
||||||
|
|
||||||
|
type settingsMeta struct {
|
||||||
|
Group string
|
||||||
|
Description string
|
||||||
|
Units string
|
||||||
|
Future bool
|
||||||
|
}
|
||||||
|
|
||||||
|
var settingsGroups = []SettingsGroupView{
|
||||||
|
{Key: "uploads", Label: "Uploads", Description: "Guest uploads and upload size defaults."},
|
||||||
|
{Key: "downloads", Label: "Downloads", Description: "ZIP and one-time download behavior."},
|
||||||
|
{Key: "retention", Label: "Retention", Description: "Expiry and renewal defaults."},
|
||||||
|
{Key: "accounts", Label: "Accounts", Description: "Session and account defaults."},
|
||||||
|
{Key: "api", Label: "API", Description: "API surface toggles."},
|
||||||
|
{Key: "storage", Label: "Storage", Description: "Storage paths and hard capacity limits."},
|
||||||
|
{Key: "workers", Label: "Workers", Description: "Background worker timing."},
|
||||||
|
{Key: "box_policy", Label: "Box policy", Description: "Defaults for future owner-managed boxes."},
|
||||||
|
}
|
||||||
|
|
||||||
|
var settingsMetadata = map[string]settingsMeta{
|
||||||
|
config.SettingGuestUploadsEnabled: {Group: "uploads", Description: "Allow guests to create upload boxes."},
|
||||||
|
config.SettingDefaultUserMaxFileBytes: {Group: "uploads", Description: "Default per-user file size limit. Zero means unlimited.", Units: "bytes"},
|
||||||
|
config.SettingDefaultUserMaxBoxBytes: {Group: "uploads", Description: "Default per-user total box size limit. Zero means unlimited.", Units: "bytes"},
|
||||||
|
config.SettingZipDownloadsEnabled: {Group: "downloads", Description: "Allow ZIP downloads when a box permits it."},
|
||||||
|
config.SettingOneTimeDownloadsEnabled: {Group: "downloads", Description: "Allow one-time ZIP handoff boxes."},
|
||||||
|
config.SettingOneTimeDownloadExpirySecs: {Group: "downloads", Description: "How long one-time downloads stay retryable or pending.", Units: "duration"},
|
||||||
|
config.SettingOneTimeDownloadRetryFail: {Group: "downloads", Description: "Keep one-time boxes retryable after a ZIP writer failure."},
|
||||||
|
config.SettingDefaultGuestExpirySecs: {Group: "retention", Description: "Default guest box expiry.", Units: "duration"},
|
||||||
|
config.SettingMaxGuestExpirySecs: {Group: "retention", Description: "Maximum guest box expiry.", Units: "duration"},
|
||||||
|
config.SettingRenewOnAccessEnabled: {Group: "retention", Description: "Allow expiry renewal when a box is opened."},
|
||||||
|
config.SettingRenewOnDownloadEnabled: {Group: "retention", Description: "Allow expiry renewal when files are downloaded."},
|
||||||
|
config.SettingSessionTTLSeconds: {Group: "accounts", Description: "Account session lifetime.", Units: "duration"},
|
||||||
|
config.SettingAPIEnabled: {Group: "api", Description: "Expose API-style upload/status endpoints."},
|
||||||
|
config.SettingDataDir: {Group: "storage", Description: "Base data directory. Environment only."},
|
||||||
|
config.SettingGlobalMaxFileSizeBytes: {Group: "storage", Description: "Hard global file size cap. Environment only.", Units: "bytes"},
|
||||||
|
config.SettingGlobalMaxBoxSizeBytes: {Group: "storage", Description: "Hard global box size cap. Environment only.", Units: "bytes"},
|
||||||
|
config.SettingBoxPollIntervalMS: {Group: "workers", Description: "Browser polling cadence for box status.", Units: "milliseconds"},
|
||||||
|
config.SettingThumbnailBatchSize: {Group: "workers", Description: "Thumbnail worker batch size."},
|
||||||
|
config.SettingThumbnailIntervalSeconds: {Group: "workers", Description: "Thumbnail worker interval.", Units: "duration"},
|
||||||
|
config.SettingBoxOwnerEditEnabled: {Group: "box_policy", Description: "Default: owners may edit their boxes."},
|
||||||
|
config.SettingBoxOwnerRefreshEnabled: {Group: "box_policy", Description: "Default: owners may refresh box expiry."},
|
||||||
|
config.SettingBoxOwnerMaxRefreshCount: {Group: "box_policy", Description: "Default maximum number of owner refreshes."},
|
||||||
|
config.SettingBoxOwnerMaxRefreshAmount: {Group: "box_policy", Description: "Default maximum expiry added per owner refresh.", Units: "duration"},
|
||||||
|
config.SettingBoxOwnerMaxTotalExpiry: {Group: "box_policy", Description: "Default maximum total box expiry for owner-managed boxes.", Units: "duration"},
|
||||||
|
config.SettingBoxOwnerPasswordEdit: {Group: "box_policy", Description: "Default: owners may edit box passwords."},
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountSettings(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
view, err := app.ListSettings(ctx, actor)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.HTML(http.StatusOK, "account_settings.html", view)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountSettingsPost(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := ctx.Request.ParseForm(); err != nil {
|
||||||
|
app.renderSettingsWithMessage(ctx, actor, "could not parse settings form", "")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
editable := map[string]config.SettingDefinition{}
|
||||||
|
for _, def := range config.EditableDefinitions() {
|
||||||
|
editable[def.Key] = def
|
||||||
|
}
|
||||||
|
for key := range ctx.Request.PostForm {
|
||||||
|
if key == "csrf_token" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, ok := editable[key]; ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, ok := config.Definition(key); ok {
|
||||||
|
app.renderSettingsWithMessage(ctx, actor, fmt.Sprintf("setting %q is locked", key), "")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
app.renderSettingsWithMessage(ctx, actor, fmt.Sprintf("unknown setting %q", key), "")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
changes := map[string]string{}
|
||||||
|
for _, def := range editable {
|
||||||
|
if def.Type == config.SettingTypeBool {
|
||||||
|
value := "false"
|
||||||
|
if ctx.PostForm(def.Key) == "true" {
|
||||||
|
value = "true"
|
||||||
|
}
|
||||||
|
changes[def.Key] = value
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
if _, exists := ctx.GetPostForm(def.Key); exists {
|
||||||
|
changes[def.Key] = ctx.PostForm(def.Key)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if err := app.UpdateSettings(ctx, actor, changes); err != nil {
|
||||||
|
app.renderSettingsWithMessage(ctx, actor, err.Error(), "")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/settings")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountSettingsReset(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if err := app.ResetSettingOverride(ctx, actor, ctx.PostForm("key")); err != nil {
|
||||||
|
app.renderSettingsWithMessage(ctx, actor, err.Error(), "")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/settings")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountSettingsExport(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
backup, err := app.ExportSettings(ctx, actor)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Header("Content-Disposition", `attachment; filename="warpbox-settings.json"`)
|
||||||
|
ctx.JSON(http.StatusOK, backup)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAccountSettingsImport(ctx *gin.Context) {
|
||||||
|
actor, ok := currentAccountUser(ctx)
|
||||||
|
if !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !strings.HasPrefix(strings.ToLower(ctx.GetHeader("Content-Type")), "application/json") {
|
||||||
|
ctx.JSON(http.StatusUnsupportedMediaType, gin.H{"error": "settings import requires application/json"})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
var backup SettingsBackup
|
||||||
|
if err := json.NewDecoder(ctx.Request.Body).Decode(&backup); err != nil {
|
||||||
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": "invalid settings JSON"})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
result, err := app.ImportSettings(ctx, actor, backup)
|
||||||
|
if err != nil {
|
||||||
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.JSON(http.StatusOK, result)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) ListSettings(ctx *gin.Context, actor metastore.User) (SettingsView, error) {
|
||||||
|
perms := currentAccountPermissions(ctx)
|
||||||
|
if !perms.AdminSettingsManage {
|
||||||
|
return SettingsView{}, fmt.Errorf("permission denied")
|
||||||
|
}
|
||||||
|
|
||||||
|
rows := app.settingsRows(perms.AdminSettingsManage && app.config.AllowAdminSettingsOverride)
|
||||||
|
groups := make([]SettingsGroupView, 0, len(settingsGroups))
|
||||||
|
for _, group := range settingsGroups {
|
||||||
|
copyGroup := group
|
||||||
|
copyGroup.Rows = rows[group.Key]
|
||||||
|
groups = append(groups, copyGroup)
|
||||||
|
}
|
||||||
|
|
||||||
|
return SettingsView{
|
||||||
|
PageTitle: "WarpBox Settings",
|
||||||
|
WindowTitle: "WarpBox Account Settings",
|
||||||
|
WindowIcon: "S",
|
||||||
|
PageScripts: []string{"/static/js/account-settings.js"},
|
||||||
|
AccountNav: app.accountNavView(ctx, "settings"),
|
||||||
|
CSRFToken: app.currentCSRFToken(ctx),
|
||||||
|
Groups: groups,
|
||||||
|
OverridesAllowed: app.config.AllowAdminSettingsOverride,
|
||||||
|
CanEdit: app.config.AllowAdminSettingsOverride,
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) UpdateSettings(ctx *gin.Context, actor metastore.User, changes map[string]string) error {
|
||||||
|
if err := app.requireSettingsEdit(ctx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if !app.config.AllowAdminSettingsOverride {
|
||||||
|
return fmt.Errorf("admin settings overrides are disabled")
|
||||||
|
}
|
||||||
|
if err := validateSettingChanges(changes); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for key, value := range changes {
|
||||||
|
if err := app.store.SetSetting(key, value); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return app.reloadRuntimeConfig()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) ResetSettingOverride(ctx *gin.Context, actor metastore.User, key string) error {
|
||||||
|
if err := app.requireSettingsEdit(ctx); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
def, ok := config.Definition(strings.TrimSpace(key))
|
||||||
|
if !ok {
|
||||||
|
return fmt.Errorf("unknown setting %q", key)
|
||||||
|
}
|
||||||
|
if !def.Editable || def.HardLimit {
|
||||||
|
return fmt.Errorf("setting %q cannot be reset from account settings", key)
|
||||||
|
}
|
||||||
|
if err := app.store.DeleteSetting(def.Key); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return app.reloadRuntimeConfig()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) ExportSettings(ctx *gin.Context, actor metastore.User) (SettingsBackup, error) {
|
||||||
|
perms := currentAccountPermissions(ctx)
|
||||||
|
if !perms.AdminSettingsManage {
|
||||||
|
return SettingsBackup{}, fmt.Errorf("permission denied")
|
||||||
|
}
|
||||||
|
settings := map[string]string{}
|
||||||
|
for _, def := range config.EditableDefinitions() {
|
||||||
|
settings[def.Key] = app.config.SettingValue(def.Key)
|
||||||
|
}
|
||||||
|
return SettingsBackup{
|
||||||
|
Version: 1,
|
||||||
|
ExportedAt: time.Now().UTC().Format(time.RFC3339),
|
||||||
|
Settings: settings,
|
||||||
|
Metadata: map[string]string{
|
||||||
|
"app": "WarpBox",
|
||||||
|
},
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) ImportSettings(ctx *gin.Context, actor metastore.User, backup SettingsBackup) (ImportResult, error) {
|
||||||
|
if err := app.requireSettingsEdit(ctx); err != nil {
|
||||||
|
return ImportResult{}, err
|
||||||
|
}
|
||||||
|
if !app.config.AllowAdminSettingsOverride {
|
||||||
|
return ImportResult{}, fmt.Errorf("admin settings overrides are disabled")
|
||||||
|
}
|
||||||
|
if backup.Settings == nil {
|
||||||
|
return ImportResult{}, fmt.Errorf("settings backup has no settings")
|
||||||
|
}
|
||||||
|
if err := validateSettingChanges(backup.Settings); err != nil {
|
||||||
|
return ImportResult{}, err
|
||||||
|
}
|
||||||
|
|
||||||
|
keys := make([]string, 0, len(backup.Settings))
|
||||||
|
for key := range backup.Settings {
|
||||||
|
keys = append(keys, key)
|
||||||
|
}
|
||||||
|
sort.Strings(keys)
|
||||||
|
for _, key := range keys {
|
||||||
|
if err := app.store.SetSetting(key, backup.Settings[key]); err != nil {
|
||||||
|
return ImportResult{}, err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if err := app.reloadRuntimeConfig(); err != nil {
|
||||||
|
return ImportResult{}, err
|
||||||
|
}
|
||||||
|
return ImportResult{Applied: len(keys), Keys: keys}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) renderSettingsWithMessage(ctx *gin.Context, actor metastore.User, errorMessage string, notice string) {
|
||||||
|
view, err := app.ListSettings(ctx, actor)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
view.Error = errorMessage
|
||||||
|
view.Notice = notice
|
||||||
|
ctx.HTML(http.StatusOK, "account_settings.html", view)
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) requireSettingsEdit(ctx *gin.Context) error {
|
||||||
|
perms := currentAccountPermissions(ctx)
|
||||||
|
if !perms.AdminSettingsManage {
|
||||||
|
return fmt.Errorf("permission denied")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) settingsRows(canEdit bool) map[string][]SettingsRowView {
|
||||||
|
out := map[string][]SettingsRowView{}
|
||||||
|
for _, row := range app.config.SettingRows() {
|
||||||
|
meta := settingsMetadata[row.Definition.Key]
|
||||||
|
group := meta.Group
|
||||||
|
if group == "" {
|
||||||
|
group = "accounts"
|
||||||
|
}
|
||||||
|
editable := canEdit && row.Definition.Editable && !row.Definition.HardLimit
|
||||||
|
out[group] = append(out[group], SettingsRowView{
|
||||||
|
Key: row.Definition.Key,
|
||||||
|
Label: row.Definition.Label,
|
||||||
|
Description: meta.Description,
|
||||||
|
Type: row.Definition.Type,
|
||||||
|
Value: row.Value,
|
||||||
|
DisplayValue: settingDisplayValue(row.Value, meta.Units),
|
||||||
|
Source: settingSourceLabel(row.Source, row.Definition),
|
||||||
|
EnvName: row.Definition.EnvName,
|
||||||
|
Editable: editable,
|
||||||
|
LockedReason: settingLockedReason(row.Definition, canEdit),
|
||||||
|
Future: meta.Future,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func validateSettingChanges(changes map[string]string) error {
|
||||||
|
if len(changes) == 0 {
|
||||||
|
return fmt.Errorf("no settings provided")
|
||||||
|
}
|
||||||
|
cfg, err := config.Load()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
for key, value := range changes {
|
||||||
|
if _, ok := config.Definition(key); !ok {
|
||||||
|
return fmt.Errorf("unknown setting %q", key)
|
||||||
|
}
|
||||||
|
if err := cfg.ApplyOverride(key, value); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) reloadRuntimeConfig() error {
|
||||||
|
cfg, err := config.Load()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
overrides, err := app.store.ListSettings()
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
if err := cfg.ApplyOverrides(overrides); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
app.config = cfg
|
||||||
|
applyBoxstoreRuntimeConfig(cfg)
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func settingSourceLabel(source config.Source, def config.SettingDefinition) string {
|
||||||
|
if def.HardLimit {
|
||||||
|
return "hard env"
|
||||||
|
}
|
||||||
|
if !def.Editable {
|
||||||
|
return "locked"
|
||||||
|
}
|
||||||
|
switch source {
|
||||||
|
case config.SourceDB:
|
||||||
|
return "override"
|
||||||
|
case config.SourceEnv:
|
||||||
|
return "env"
|
||||||
|
default:
|
||||||
|
return "default"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func settingLockedReason(def config.SettingDefinition, canEdit bool) string {
|
||||||
|
if !canEdit {
|
||||||
|
return "settings changes disabled"
|
||||||
|
}
|
||||||
|
if def.HardLimit {
|
||||||
|
return "hard environment limit"
|
||||||
|
}
|
||||||
|
if !def.Editable {
|
||||||
|
return "runtime editing not supported"
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func settingDisplayValue(value string, units string) string {
|
||||||
|
switch units {
|
||||||
|
case "bytes":
|
||||||
|
parsed, ok := parseInt64String(value)
|
||||||
|
if !ok {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
if parsed == 0 {
|
||||||
|
return "unlimited"
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s (%s bytes)", formatBytesForSettings(parsed), value)
|
||||||
|
case "duration":
|
||||||
|
parsed, ok := parseInt64String(value)
|
||||||
|
if !ok {
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%s (%s seconds)", formatDurationForSettings(parsed), value)
|
||||||
|
case "milliseconds":
|
||||||
|
return value + " ms"
|
||||||
|
default:
|
||||||
|
return value
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseInt64String(value string) (int64, bool) {
|
||||||
|
var parsed int64
|
||||||
|
if _, err := fmt.Sscan(strings.TrimSpace(value), &parsed); err != nil {
|
||||||
|
return 0, false
|
||||||
|
}
|
||||||
|
return parsed, true
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatBytesForSettings(value int64) string {
|
||||||
|
units := []string{"B", "KiB", "MiB", "GiB", "TiB"}
|
||||||
|
size := float64(value)
|
||||||
|
unit := 0
|
||||||
|
for size >= 1024 && unit < len(units)-1 {
|
||||||
|
size /= 1024
|
||||||
|
unit++
|
||||||
|
}
|
||||||
|
return fmt.Sprintf("%.1f %s", size, units[unit])
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatDurationForSettings(seconds int64) string {
|
||||||
|
switch {
|
||||||
|
case seconds == 0:
|
||||||
|
return "none"
|
||||||
|
case seconds%86400 == 0:
|
||||||
|
return fmt.Sprintf("%d days", seconds/86400)
|
||||||
|
case seconds%3600 == 0:
|
||||||
|
return fmt.Sprintf("%d hours", seconds/3600)
|
||||||
|
case seconds%60 == 0:
|
||||||
|
return fmt.Sprintf("%d minutes", seconds/60)
|
||||||
|
default:
|
||||||
|
return fmt.Sprintf("%d seconds", seconds)
|
||||||
|
}
|
||||||
|
}
|
||||||
197
lib/server/account_settings_test.go
Normal file
197
lib/server/account_settings_test.go
Normal file
@@ -0,0 +1,197 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"net/url"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"warpbox/lib/config"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestAccountSettingsPermissionDenied(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
user, err := app.store.CreateUserWithPassword("regular", "regular@example.test", "secret", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateUserWithPassword returned error: %v", err)
|
||||||
|
}
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account/settings", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusForbidden {
|
||||||
|
t.Fatalf("expected permission denied, got %d", response.Code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountSettingsPageLoadsForAdmin(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account/settings", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected settings page, got %d body=%s", response.Code, response.Body.String())
|
||||||
|
}
|
||||||
|
for _, text := range []string{"Uploads", "Downloads", "Box policy", "Save Settings"} {
|
||||||
|
if !strings.Contains(response.Body.String(), text) {
|
||||||
|
t.Fatalf("expected settings page to contain %q", text)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountSettingsValidUpdate(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
|
||||||
|
form := url.Values{}
|
||||||
|
form.Set("csrf_token", session.CSRFToken)
|
||||||
|
form.Set(config.SettingAPIEnabled, "false")
|
||||||
|
response := postAccountSettingsForm(router, session, form)
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected settings redirect, got %d body=%s", response.Code, response.Body.String())
|
||||||
|
}
|
||||||
|
if app.config.APIEnabled {
|
||||||
|
t.Fatal("expected API setting to be disabled")
|
||||||
|
}
|
||||||
|
value, ok, err := app.store.GetSetting(config.SettingAPIEnabled)
|
||||||
|
if err != nil || !ok || value != "false" {
|
||||||
|
t.Fatalf("expected API setting override false, got value=%q ok=%v err=%v", value, ok, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountSettingsInvalidUpdate(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
|
||||||
|
form := url.Values{}
|
||||||
|
form.Set("csrf_token", session.CSRFToken)
|
||||||
|
form.Set(config.SettingSessionTTLSeconds, "1")
|
||||||
|
response := postAccountSettingsForm(router, session, form)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected settings form render, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if !strings.Contains(response.Body.String(), "must be at least 60") {
|
||||||
|
t.Fatal("expected validation error in response")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountSettingsLockedSettingCannotChange(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
|
||||||
|
form := url.Values{}
|
||||||
|
form.Set("csrf_token", session.CSRFToken)
|
||||||
|
form.Set(config.SettingGlobalMaxFileSizeBytes, "1")
|
||||||
|
response := postAccountSettingsForm(router, session, form)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected settings form render, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if !strings.Contains(response.Body.String(), "locked") {
|
||||||
|
t.Fatal("expected locked setting error")
|
||||||
|
}
|
||||||
|
if value, ok, err := app.store.GetSetting(config.SettingGlobalMaxFileSizeBytes); err != nil || ok || value != "" {
|
||||||
|
t.Fatalf("expected no locked setting override, got value=%q ok=%v err=%v", value, ok, err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountSettingsImportRejectsUnknownOrInvalidSettings(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
|
||||||
|
for _, body := range []string{
|
||||||
|
`{"version":1,"settings":{"not_real":"true"}}`,
|
||||||
|
`{"version":1,"settings":{"session_ttl_seconds":"1"}}`,
|
||||||
|
} {
|
||||||
|
response := postAccountSettingsJSON(router, session, body)
|
||||||
|
if response.Code != http.StatusBadRequest {
|
||||||
|
t.Fatalf("expected bad import for %s, got %d", body, response.Code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountSettingsImportAppliesValidSettings(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
|
||||||
|
response := postAccountSettingsJSON(router, session, `{"version":1,"settings":{"api_enabled":"false","box_owner_max_refresh_count":"7"}}`)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected import success, got %d body=%s", response.Code, response.Body.String())
|
||||||
|
}
|
||||||
|
if app.config.APIEnabled {
|
||||||
|
t.Fatal("expected imported API setting to be disabled")
|
||||||
|
}
|
||||||
|
if app.config.BoxOwnerMaxRefreshCount != 7 {
|
||||||
|
t.Fatalf("expected imported box owner refresh count 7, got %d", app.config.BoxOwnerMaxRefreshCount)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountSettingsExportShape(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session := createAccountTestSession(t, app, user)
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account/settings/export.json", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected export success, got %d", response.Code)
|
||||||
|
}
|
||||||
|
var backup SettingsBackup
|
||||||
|
if err := json.Unmarshal(response.Body.Bytes(), &backup); err != nil {
|
||||||
|
t.Fatalf("Unmarshal returned error: %v", err)
|
||||||
|
}
|
||||||
|
if backup.Version != 1 {
|
||||||
|
t.Fatalf("expected version 1, got %d", backup.Version)
|
||||||
|
}
|
||||||
|
if _, ok := backup.Settings[config.SettingBoxOwnerMaxRefreshCount]; !ok {
|
||||||
|
t.Fatal("expected export to include box owner policy setting")
|
||||||
|
}
|
||||||
|
if _, ok := backup.Settings[config.SettingDataDir]; ok {
|
||||||
|
t.Fatal("did not expect locked data dir in export settings")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func createAccountTestSession(t *testing.T, app *App, user metastore.User) metastore.Session {
|
||||||
|
t.Helper()
|
||||||
|
session, err := app.store.CreateSession(user.ID, time.Hour)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateSession returned error: %v", err)
|
||||||
|
}
|
||||||
|
return session
|
||||||
|
}
|
||||||
|
|
||||||
|
func postAccountSettingsForm(router http.Handler, session metastore.Session, form url.Values) *httptest.ResponseRecorder {
|
||||||
|
request := httptest.NewRequest(http.MethodPost, "/account/settings", strings.NewReader(form.Encode()))
|
||||||
|
request.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
return response
|
||||||
|
}
|
||||||
|
|
||||||
|
func postAccountSettingsJSON(router http.Handler, session metastore.Session, body string) *httptest.ResponseRecorder {
|
||||||
|
request := httptest.NewRequest(http.MethodPost, "/account/settings/import.json", strings.NewReader(body))
|
||||||
|
request.Header.Set("Content-Type", "application/json")
|
||||||
|
request.Header.Set("X-CSRF-Token", session.CSRFToken)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
return response
|
||||||
|
}
|
||||||
245
lib/server/account_test.go
Normal file
245
lib/server/account_test.go
Normal file
@@ -0,0 +1,245 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"html/template"
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
|
"net/url"
|
||||||
|
"path/filepath"
|
||||||
|
"strings"
|
||||||
|
"testing"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/boxstore"
|
||||||
|
"warpbox/lib/config"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestAccountLoginSuccess(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
|
||||||
|
response := postAccountLogin(router, "admin", "secret")
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected login redirect, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if location := response.Header().Get("Location"); location != "/account" {
|
||||||
|
t.Fatalf("expected redirect to /account, got %q", location)
|
||||||
|
}
|
||||||
|
if cookie := findResponseCookie(response, accountSessionCookie); cookie == nil || cookie.Value == "" {
|
||||||
|
t.Fatal("expected account session cookie")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountLoginFailure(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
|
||||||
|
response := postAccountLogin(router, "admin", "wrong")
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected failed login to render form, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if cookie := findResponseCookie(response, accountSessionCookie); cookie != nil {
|
||||||
|
t.Fatal("did not expect account session cookie")
|
||||||
|
}
|
||||||
|
if !strings.Contains(response.Body.String(), "not accepted") {
|
||||||
|
t.Fatal("expected login failure message")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountDisabledUserLoginFailure(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
user.Disabled = true
|
||||||
|
if err := app.store.UpdateUser(user); err != nil {
|
||||||
|
t.Fatalf("UpdateUser returned error: %v", err)
|
||||||
|
}
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
|
||||||
|
response := postAccountLogin(router, "admin", "secret")
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected disabled login to render form, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if cookie := findResponseCookie(response, accountSessionCookie); cookie != nil {
|
||||||
|
t.Fatal("did not expect account session cookie")
|
||||||
|
}
|
||||||
|
if !strings.Contains(response.Body.String(), "not accepted") {
|
||||||
|
t.Fatal("expected login failure message")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountLogoutRequiresCSRF(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session, err := app.store.CreateSession(user.ID, time.Hour)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateSession returned error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodPost, "/account/logout", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusForbidden {
|
||||||
|
t.Fatalf("expected missing CSRF token to be forbidden, got %d", response.Code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountDashboardRequiresAuth(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account", nil)
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected dashboard redirect, got %d", response.Code)
|
||||||
|
}
|
||||||
|
if location := response.Header().Get("Location"); location != "/account/login" {
|
||||||
|
t.Fatalf("expected redirect to /account/login, got %q", location)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountDashboardLoadsForBootstrapAdmin(t *testing.T) {
|
||||||
|
app, user := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session, err := app.store.CreateSession(user.ID, time.Hour)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateSession returned error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected dashboard to load, got %d", response.Code)
|
||||||
|
}
|
||||||
|
body := response.Body.String()
|
||||||
|
for _, text := range []string{"Dashboard", "Recent Boxes", "Users"} {
|
||||||
|
if !strings.Contains(body, text) {
|
||||||
|
t.Fatalf("expected dashboard body to contain %q", text)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAccountDashboardHidesAdminOnlyLinksForRegularUser(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
user, err := app.store.CreateUserWithPassword("maya", "maya@example.test", "secret", nil)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateUserWithPassword returned error: %v", err)
|
||||||
|
}
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
session, err := app.store.CreateSession(user.ID, time.Hour)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateSession returned error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodGet, "/account", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: accountSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusOK {
|
||||||
|
t.Fatalf("expected dashboard to load, got %d", response.Code)
|
||||||
|
}
|
||||||
|
body := response.Body.String()
|
||||||
|
for _, text := range []string{">Users<", ">Settings<"} {
|
||||||
|
if strings.Contains(body, text) {
|
||||||
|
t.Fatalf("expected dashboard body to hide %q", text)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestAdminEntryRedirectsToAccount(t *testing.T) {
|
||||||
|
app, _ := setupAccountTestApp(t)
|
||||||
|
router := setupAccountTestRouter(t, app)
|
||||||
|
|
||||||
|
cases := map[string]string{
|
||||||
|
"/admin/login": "/account/login",
|
||||||
|
"/admin": "/account",
|
||||||
|
}
|
||||||
|
for path, wantLocation := range cases {
|
||||||
|
request := httptest.NewRequest(http.MethodGet, path, nil)
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusSeeOther {
|
||||||
|
t.Fatalf("expected %s redirect, got %d", path, response.Code)
|
||||||
|
}
|
||||||
|
if location := response.Header().Get("Location"); location != wantLocation {
|
||||||
|
t.Fatalf("expected %s to redirect to %s, got %q", path, wantLocation, location)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func setupAccountTestApp(t *testing.T) (*App, metastore.User) {
|
||||||
|
t.Helper()
|
||||||
|
gin.SetMode(gin.TestMode)
|
||||||
|
|
||||||
|
restoreUploadRoot := boxstore.UploadRoot()
|
||||||
|
t.Cleanup(func() { boxstore.SetUploadRoot(restoreUploadRoot) })
|
||||||
|
boxstore.SetUploadRoot(t.TempDir())
|
||||||
|
|
||||||
|
store, err := metastore.Open(t.TempDir())
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Open returned error: %v", err)
|
||||||
|
}
|
||||||
|
t.Cleanup(func() { _ = store.Close() })
|
||||||
|
|
||||||
|
cfg, err := config.Load()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Load returned error: %v", err)
|
||||||
|
}
|
||||||
|
cfg.AdminUsername = "admin"
|
||||||
|
cfg.AdminPassword = "secret"
|
||||||
|
cfg.AdminEmail = "admin@example.test"
|
||||||
|
cfg.AdminEnabled = config.AdminEnabledAuto
|
||||||
|
cfg.SessionTTLSeconds = 3600
|
||||||
|
bootstrap, err := metastore.BootstrapAdmin(cfg, store)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("BootstrapAdmin returned error: %v", err)
|
||||||
|
}
|
||||||
|
if bootstrap.AdminUser == nil {
|
||||||
|
t.Fatal("expected bootstrap admin user")
|
||||||
|
}
|
||||||
|
|
||||||
|
app := &App{
|
||||||
|
config: cfg,
|
||||||
|
store: store,
|
||||||
|
adminLoginEnabled: bootstrap.AdminLoginEnabled,
|
||||||
|
}
|
||||||
|
return app, *bootstrap.AdminUser
|
||||||
|
}
|
||||||
|
|
||||||
|
func setupAccountTestRouter(t *testing.T, app *App) *gin.Engine {
|
||||||
|
t.Helper()
|
||||||
|
router := gin.New()
|
||||||
|
templates, err := template.ParseGlob(filepath.Join("..", "..", "templates", "*.html"))
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("ParseGlob returned error: %v", err)
|
||||||
|
}
|
||||||
|
router.SetHTMLTemplate(templates)
|
||||||
|
app.registerAccountRoutes(router)
|
||||||
|
app.registerAdminRoutes(router)
|
||||||
|
return router
|
||||||
|
}
|
||||||
|
|
||||||
|
func postAccountLogin(router *gin.Engine, username string, password string) *httptest.ResponseRecorder {
|
||||||
|
form := url.Values{}
|
||||||
|
form.Set("username", username)
|
||||||
|
form.Set("password", password)
|
||||||
|
request := httptest.NewRequest(http.MethodPost, "/account/login", strings.NewReader(form.Encode()))
|
||||||
|
request.Header.Set("Content-Type", "application/x-www-form-urlencoded")
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
return response
|
||||||
|
}
|
||||||
|
|
||||||
|
func findResponseCookie(response *httptest.ResponseRecorder, name string) *http.Cookie {
|
||||||
|
for _, cookie := range response.Result().Cookies() {
|
||||||
|
if cookie.Name == name {
|
||||||
|
return cookie
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
@@ -1,173 +0,0 @@
|
|||||||
package server
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net/http"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
|
|
||||||
"warpbox/lib/alerts"
|
|
||||||
"warpbox/lib/config"
|
|
||||||
"warpbox/lib/security"
|
|
||||||
)
|
|
||||||
|
|
||||||
const adminSessionCookie = "warpbox_admin_session"
|
|
||||||
const adminSessionMarker = "1"
|
|
||||||
|
|
||||||
func (app *App) adminLoginEnabled() bool {
|
|
||||||
return app.config.AdminLoginEnabled(app.config.AdminPassword != "")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) adminAuthMiddleware(ctx *gin.Context) {
|
|
||||||
if !app.adminLoginEnabled() {
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/")
|
|
||||||
ctx.Abort()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
token, err := ctx.Cookie(adminSessionCookie)
|
|
||||||
if err != nil || token != app.adminSessionToken() {
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/admin/login")
|
|
||||||
ctx.Abort()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.Next()
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) adminSessionToken() string {
|
|
||||||
// A simple deterministic token derived from the admin credentials.
|
|
||||||
// This will improve when proper user/session storage is added.
|
|
||||||
return app.config.AdminUsername + ":" + app.config.AdminPassword
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminLogin(ctx *gin.Context) {
|
|
||||||
if !app.adminLoginEnabled() {
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
// Already logged in.
|
|
||||||
if token, err := ctx.Cookie(adminSessionCookie); err == nil && token == app.adminSessionToken() {
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/admin/dashboard")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.HTML(http.StatusOK, "admin/login.html", gin.H{})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminLoginPost(ctx *gin.Context) {
|
|
||||||
if !app.adminLoginEnabled() {
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ip := app.clientIP(ctx)
|
|
||||||
guard := app.securityGuard
|
|
||||||
if app.securityFeaturesEnabled() && guard == nil {
|
|
||||||
guard = security.NewGuard()
|
|
||||||
app.securityGuard = guard
|
|
||||||
}
|
|
||||||
if app.securityFeaturesEnabled() && guard != nil && !guard.IsAdminWhitelisted(ip) && guard.IsBanned(ip) {
|
|
||||||
app.logActivity("auth.admin.block", "high", "Blocked admin login from banned IP", ctx, nil)
|
|
||||||
ctx.HTML(http.StatusTooManyRequests, "admin/login.html", gin.H{
|
|
||||||
"ErrorMessage": "Too many failed attempts. Try again later.",
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
username := strings.TrimSpace(ctx.PostForm("username"))
|
|
||||||
password := ctx.PostForm("password")
|
|
||||||
|
|
||||||
if username != app.config.AdminUsername || password != app.config.AdminPassword {
|
|
||||||
if app.securityFeaturesEnabled() && guard != nil && !guard.IsAdminWhitelisted(ip) {
|
|
||||||
banned, attempts := guard.RegisterFailedLogin(ip, app.config.SecurityLoginWindowSeconds, app.config.SecurityLoginMaxAttempts, app.config.SecurityBanSeconds)
|
|
||||||
app.logActivity("auth.admin.failed", "medium", "Failed admin login", ctx, map[string]string{"attempts": strconv.Itoa(attempts)})
|
|
||||||
if banned {
|
|
||||||
app.createAlert("Admin login brute-force blocked", "high", "security", "401", "auth.admin.bruteforce", "Too many failed admin logins triggered temporary ban.", map[string]string{"ip": ip, "attempts": strconv.Itoa(attempts)})
|
|
||||||
app.logActivity("security.ban", "high", "Auto-banned IP after admin login failures", ctx, map[string]string{"attempts": strconv.Itoa(attempts)})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ctx.HTML(http.StatusUnauthorized, "admin/login.html", gin.H{
|
|
||||||
"ErrorMessage": "Invalid username or password.",
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
app.logActivity("auth.admin.success", "low", "Admin login successful", ctx, nil)
|
|
||||||
secure := app.config.AdminCookieSecure
|
|
||||||
maxAge := int(app.config.SessionTTLSeconds)
|
|
||||||
|
|
||||||
ctx.SetCookie(adminSessionCookie, app.adminSessionToken(), maxAge, "/admin", "", secure, true)
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/admin/dashboard")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminLogout(ctx *gin.Context) {
|
|
||||||
secure := app.config.AdminCookieSecure
|
|
||||||
ctx.SetCookie(adminSessionCookie, "", -1, "/admin", "", secure, true)
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/admin/login")
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminDashboard(ctx *gin.Context) {
|
|
||||||
if !app.adminLoginEnabled() {
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
dashboardEnabled := config.AdminEnabledTrue
|
|
||||||
if cfgVal := app.config.AdminEnabled; cfgVal == config.AdminEnabledAuto || cfgVal == config.AdminEnabledTrue {
|
|
||||||
dashboardEnabled = cfgVal
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.HTML(http.StatusOK, "admin/dashboard.html", gin.H{
|
|
||||||
"AdminUsername": app.config.AdminUsername,
|
|
||||||
"AdminEmail": app.config.AdminEmail,
|
|
||||||
"ActivePage": "dashboard",
|
|
||||||
"DashboardEnabled": string(dashboardEnabled),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminAlerts(ctx *gin.Context) {
|
|
||||||
if !app.adminLoginEnabled() {
|
|
||||||
ctx.Redirect(http.StatusSeeOther, "/")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
alertsList := []alerts.Alert{}
|
|
||||||
if app.alertStore != nil {
|
|
||||||
var err error
|
|
||||||
alertsList, err = app.alertStore.List(500)
|
|
||||||
if err != nil {
|
|
||||||
ctx.String(http.StatusInternalServerError, "Could not load alerts")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
}
|
|
||||||
openCount := 0
|
|
||||||
highCount := 0
|
|
||||||
ackedCount := 0
|
|
||||||
closedCount := 0
|
|
||||||
for _, alert := range alertsList {
|
|
||||||
switch string(alert.Status) {
|
|
||||||
case "open":
|
|
||||||
openCount++
|
|
||||||
case "acked":
|
|
||||||
ackedCount++
|
|
||||||
case "closed":
|
|
||||||
closedCount++
|
|
||||||
}
|
|
||||||
if alert.Severity == "high" && string(alert.Status) != "closed" {
|
|
||||||
highCount++
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.HTML(http.StatusOK, "admin/alerts.html", gin.H{
|
|
||||||
"AdminUsername": app.config.AdminUsername,
|
|
||||||
"AdminEmail": app.config.AdminEmail,
|
|
||||||
"ActivePage": "alerts",
|
|
||||||
"Alerts": alertsList,
|
|
||||||
"OpenCount": strconv.Itoa(openCount),
|
|
||||||
"HighCount": strconv.Itoa(highCount),
|
|
||||||
"AckCount": strconv.Itoa(ackedCount),
|
|
||||||
"ClosedCount": strconv.Itoa(closedCount),
|
|
||||||
})
|
|
||||||
}
|
|
||||||
195
lib/server/admin_auth.go
Normal file
195
lib/server/admin_auth.go
Normal file
@@ -0,0 +1,195 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"crypto/subtle"
|
||||||
|
"net/http"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
const adminSessionCookie = "warpbox_admin_session"
|
||||||
|
|
||||||
|
func (app *App) handleAdminLogin(ctx *gin.Context) {
|
||||||
|
if app.isAdminSessionValid(ctx) {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
app.renderAdminLogin(ctx, "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAdminLoginPost(ctx *gin.Context) {
|
||||||
|
if !app.adminLoginEnabled {
|
||||||
|
app.renderAdminLogin(ctx, "Administrator login is disabled.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
username := strings.TrimSpace(ctx.PostForm("username"))
|
||||||
|
password := ctx.PostForm("password")
|
||||||
|
user, ok, err := app.store.GetUserByUsername(username)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not load user")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !ok || user.Disabled || !metastore.VerifyPassword(user.PasswordHash, password) {
|
||||||
|
app.renderAdminLogin(ctx, "The username or password was not accepted.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
perms, err := app.permissionsForUser(user)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not load permissions")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !perms.AdminAccess {
|
||||||
|
app.renderAdminLogin(ctx, "This user does not have administrator access.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
session, err := app.store.CreateSession(user.ID, time.Duration(app.config.SessionTTLSeconds)*time.Second)
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not create session")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.SetSameSite(http.SameSiteLaxMode)
|
||||||
|
ctx.SetCookie(adminSessionCookie, session.Token, int(app.config.SessionTTLSeconds), "/admin", "", app.config.AdminCookieSecure, true)
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAdminLogout(ctx *gin.Context) {
|
||||||
|
if token, err := ctx.Cookie(adminSessionCookie); err == nil {
|
||||||
|
_ = app.store.DeleteSession(token)
|
||||||
|
}
|
||||||
|
ctx.SetSameSite(http.SameSiteLaxMode)
|
||||||
|
ctx.SetCookie(adminSessionCookie, "", -1, "/admin", "", app.config.AdminCookieSecure, true)
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin/login")
|
||||||
|
}
|
||||||
|
func (app *App) requireAdminSession(ctx *gin.Context) {
|
||||||
|
token, err := ctx.Cookie(adminSessionCookie)
|
||||||
|
if err != nil {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin/login")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
session, ok, err := app.store.GetSession(token)
|
||||||
|
if err != nil || !ok {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin/login")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if !validAdminCSRF(ctx, session) {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
user, ok, err := app.store.GetUser(session.UserID)
|
||||||
|
if err != nil || !ok || user.Disabled {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin/login")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
perms, err := app.permissionsForUser(user)
|
||||||
|
if err != nil || !perms.AdminAccess {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin/login")
|
||||||
|
ctx.Abort()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Set("adminUser", user)
|
||||||
|
ctx.Set("adminPerms", perms)
|
||||||
|
ctx.Set("adminCSRFToken", session.CSRFToken)
|
||||||
|
ctx.Next()
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) isAdminSessionValid(ctx *gin.Context) bool {
|
||||||
|
token, err := ctx.Cookie(adminSessionCookie)
|
||||||
|
if err != nil {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
session, ok, err := app.store.GetSession(token)
|
||||||
|
if err != nil || !ok {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
user, ok, err := app.store.GetUser(session.UserID)
|
||||||
|
if err != nil || !ok || user.Disabled {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
perms, err := app.permissionsForUser(user)
|
||||||
|
return err == nil && perms.AdminAccess
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) permissionsForUser(user metastore.User) (metastore.EffectivePermissions, error) {
|
||||||
|
tags, err := app.store.TagsByID(user.TagIDs)
|
||||||
|
if err != nil {
|
||||||
|
return metastore.EffectivePermissions{}, err
|
||||||
|
}
|
||||||
|
return metastore.ResolveUserPermissions(app.config, user, tags), nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) requireAdminFlag(ctx *gin.Context, allowed func(metastore.EffectivePermissions) bool) bool {
|
||||||
|
value, ok := ctx.Get("adminPerms")
|
||||||
|
if !ok {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
perms, ok := value.(metastore.EffectivePermissions)
|
||||||
|
if !ok || !allowed(perms) {
|
||||||
|
ctx.String(http.StatusForbidden, "Permission denied")
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) currentAdminUsername(ctx *gin.Context) string {
|
||||||
|
if current, ok := ctx.Get("adminUser"); ok {
|
||||||
|
if user, ok := current.(metastore.User); ok {
|
||||||
|
return user.Username
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) currentCSRFToken(ctx *gin.Context) string {
|
||||||
|
if value, ok := ctx.Get("adminCSRFToken"); ok {
|
||||||
|
if token, ok := value.(string); ok {
|
||||||
|
return token
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return ""
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) renderAdminLogin(ctx *gin.Context, errorMessage string) {
|
||||||
|
ctx.HTML(http.StatusOK, "admin_login.html", gin.H{
|
||||||
|
"AdminLoginEnabled": app.adminLoginEnabled,
|
||||||
|
"Error": errorMessage,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
func noStoreAdminHeaders(ctx *gin.Context) {
|
||||||
|
ctx.Header("Cache-Control", "no-store")
|
||||||
|
ctx.Header("Pragma", "no-cache")
|
||||||
|
ctx.Header("X-Content-Type-Options", "nosniff")
|
||||||
|
ctx.Next()
|
||||||
|
}
|
||||||
|
|
||||||
|
func validAdminCSRF(ctx *gin.Context, session metastore.Session) bool {
|
||||||
|
switch ctx.Request.Method {
|
||||||
|
case http.MethodGet, http.MethodHead, http.MethodOptions:
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
token := ctx.PostForm("csrf_token")
|
||||||
|
if token == "" {
|
||||||
|
token = ctx.GetHeader("X-CSRF-Token")
|
||||||
|
}
|
||||||
|
return token != "" && subtleConstantTimeEqual(token, session.CSRFToken)
|
||||||
|
}
|
||||||
|
|
||||||
|
func subtleConstantTimeEqual(a string, b string) bool {
|
||||||
|
if len(a) != len(b) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
return subtle.ConstantTimeCompare([]byte(a), []byte(b)) == 1
|
||||||
|
}
|
||||||
@@ -1,337 +1,63 @@
|
|||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"sort"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
"warpbox/lib/boxstore"
|
"warpbox/lib/boxstore"
|
||||||
|
"warpbox/lib/helpers"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
)
|
)
|
||||||
|
|
||||||
type adminBoxesActionRequest struct {
|
type adminBoxRow struct {
|
||||||
Action string `json:"action"`
|
ID string
|
||||||
BoxIDs []string `json:"box_ids"`
|
FileCount int
|
||||||
DeltaSeconds int64 `json:"delta_seconds,omitempty"`
|
TotalSizeLabel string
|
||||||
}
|
CreatedAt string
|
||||||
|
ExpiresAt string
|
||||||
type adminBoxFileView struct {
|
Expired bool
|
||||||
Name string `json:"name"`
|
OneTimeDownload bool
|
||||||
SizeLabel string `json:"size_label"`
|
PasswordProtected bool
|
||||||
MimeType string `json:"mime_type"`
|
|
||||||
Status string `json:"status"`
|
|
||||||
StatusLabel string `json:"status_label"`
|
|
||||||
DownloadPath string `json:"download_path"`
|
|
||||||
ThumbnailURL string `json:"thumbnail_url"`
|
|
||||||
IsComplete bool `json:"is_complete"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type adminBoxView struct {
|
|
||||||
ID string `json:"id"`
|
|
||||||
Status string `json:"status"`
|
|
||||||
StatusLabel string `json:"status_label"`
|
|
||||||
FileCount int `json:"file_count"`
|
|
||||||
CompleteFiles int `json:"complete_files"`
|
|
||||||
PendingFiles int `json:"pending_files"`
|
|
||||||
FailedFiles int `json:"failed_files"`
|
|
||||||
TotalSizeLabel string `json:"total_size_label"`
|
|
||||||
CreatedAtLabel string `json:"created_at_label"`
|
|
||||||
CreatedAtISO string `json:"created_at_iso"`
|
|
||||||
ExpiresAtLabel string `json:"expires_at_label"`
|
|
||||||
ExpiresAtISO string `json:"expires_at_iso"`
|
|
||||||
RetentionLabel string `json:"retention_label"`
|
|
||||||
PasswordProtected bool `json:"password_protected"`
|
|
||||||
OneTimeDownload bool `json:"one_time_download"`
|
|
||||||
ZipDisabled bool `json:"zip_disabled"`
|
|
||||||
ZipAvailable bool `json:"zip_available"`
|
|
||||||
Consumed bool `json:"consumed"`
|
|
||||||
HasManifest bool `json:"has_manifest"`
|
|
||||||
OpenURL string `json:"open_url"`
|
|
||||||
ZipURL string `json:"zip_url"`
|
|
||||||
Flags []string `json:"flags"`
|
|
||||||
Files []adminBoxFileView `json:"files"`
|
|
||||||
SearchText string `json:"search_text"`
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (app *App) handleAdminBoxes(ctx *gin.Context) {
|
func (app *App) handleAdminBoxes(ctx *gin.Context) {
|
||||||
if !app.adminLoginEnabled() {
|
if !app.requireAdminFlag(ctx, func(perms metastore.EffectivePermissions) bool { return perms.AdminBoxesView }) {
|
||||||
ctx.Redirect(http.StatusSeeOther, "/")
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
boxes, err := app.listAdminBoxes()
|
|
||||||
if err != nil {
|
|
||||||
ctx.String(http.StatusInternalServerError, "Could not load boxes")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.HTML(http.StatusOK, "admin/boxes.html", gin.H{
|
|
||||||
"AdminUsername": app.config.AdminUsername,
|
|
||||||
"AdminEmail": app.config.AdminEmail,
|
|
||||||
"ActivePage": "boxes",
|
|
||||||
"Boxes": boxes,
|
|
||||||
"ZipDownloadsOn": app.config.ZipDownloadsEnabled,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminBoxesAction(ctx *gin.Context) {
|
|
||||||
var request adminBoxesActionRequest
|
|
||||||
if err := ctx.ShouldBindJSON(&request); err != nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Invalid action payload"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch request.Action {
|
|
||||||
case "delete", "expire", "bump", "cleanup_expired":
|
|
||||||
default:
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Unknown action"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if request.Action != "cleanup_expired" && len(request.BoxIDs) == 0 {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Select one or more boxes first"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if request.Action == "bump" && request.DeltaSeconds <= 0 {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Missing bump duration"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if request.Action == "cleanup_expired" {
|
|
||||||
result, err := app.runExpiredCleanup("admin")
|
|
||||||
if err != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Expired cleanup job failed"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
boxes, listErr := app.listAdminBoxes()
|
|
||||||
if listErr != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Cleanup finished, but boxes could not be reloaded"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{
|
|
||||||
"ok": len(result.Warnings) == 0,
|
|
||||||
"message": fmt.Sprintf("Expired cleanup done: deleted %d box(es), skipped %d", result.Deleted, result.Skipped),
|
|
||||||
"warnings": result.Warnings,
|
|
||||||
"boxes": boxes,
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
processed := 0
|
|
||||||
warnings := make([]string, 0)
|
|
||||||
|
|
||||||
for _, boxID := range request.BoxIDs {
|
|
||||||
if !boxstore.ValidBoxID(boxID) {
|
|
||||||
warnings = append(warnings, fmt.Sprintf("%s: invalid box id", boxID))
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
|
|
||||||
var err error
|
|
||||||
switch request.Action {
|
|
||||||
case "delete":
|
|
||||||
err = boxstore.DeleteBox(boxID)
|
|
||||||
case "expire":
|
|
||||||
_, err = boxstore.ExpireBox(boxID)
|
|
||||||
case "bump":
|
|
||||||
_, err = boxstore.BumpBoxExpiry(boxID, time.Duration(request.DeltaSeconds)*time.Second)
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
warnings = append(warnings, fmt.Sprintf("%s: %v", boxID, err))
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
processed++
|
|
||||||
}
|
|
||||||
|
|
||||||
boxes, err := app.listAdminBoxes()
|
|
||||||
if err != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Action finished, but boxes could not be reloaded"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
status := http.StatusOK
|
|
||||||
if processed == 0 && len(warnings) > 0 {
|
|
||||||
status = http.StatusBadRequest
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.JSON(status, gin.H{
|
|
||||||
"ok": len(warnings) == 0,
|
|
||||||
"message": adminBoxesActionMessage(request.Action, processed, request.DeltaSeconds),
|
|
||||||
"warnings": warnings,
|
|
||||||
"boxes": boxes,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) listAdminBoxes() ([]adminBoxView, error) {
|
|
||||||
summaries, err := boxstore.ListBoxSummaries()
|
summaries, err := boxstore.ListBoxSummaries()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
return nil, err
|
ctx.String(http.StatusInternalServerError, "Could not list boxes")
|
||||||
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
boxes := make([]adminBoxView, 0, len(summaries))
|
rows := make([]adminBoxRow, 0, len(summaries))
|
||||||
|
totalSize := int64(0)
|
||||||
|
expiredCount := 0
|
||||||
for _, summary := range summaries {
|
for _, summary := range summaries {
|
||||||
boxView, err := app.buildAdminBoxView(summary.ID)
|
totalSize += summary.TotalSize
|
||||||
if err != nil {
|
if summary.Expired {
|
||||||
continue
|
expiredCount++
|
||||||
}
|
}
|
||||||
boxes = append(boxes, boxView)
|
rows = append(rows, adminBoxRow{
|
||||||
}
|
|
||||||
|
|
||||||
sort.Slice(boxes, func(i, j int) bool {
|
|
||||||
return boxes[i].CreatedAtISO > boxes[j].CreatedAtISO
|
|
||||||
})
|
|
||||||
return boxes, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) buildAdminBoxView(boxID string) (adminBoxView, error) {
|
|
||||||
summary, err := boxstore.BoxSummary(boxID)
|
|
||||||
if err != nil {
|
|
||||||
return adminBoxView{}, err
|
|
||||||
}
|
|
||||||
|
|
||||||
files, err := boxstore.ListFiles(boxID)
|
|
||||||
if err != nil {
|
|
||||||
return adminBoxView{}, err
|
|
||||||
}
|
|
||||||
|
|
||||||
manifest, manifestErr := boxstore.ReadManifest(boxID)
|
|
||||||
hasManifest := manifestErr == nil
|
|
||||||
|
|
||||||
boxView := adminBoxView{
|
|
||||||
ID: summary.ID,
|
ID: summary.ID,
|
||||||
FileCount: summary.FileCount,
|
FileCount: summary.FileCount,
|
||||||
TotalSizeLabel: summary.TotalSizeLabel,
|
TotalSizeLabel: summary.TotalSizeLabel,
|
||||||
CreatedAtLabel: adminTimeLabel(summary.CreatedAt),
|
CreatedAt: formatAdminTime(summary.CreatedAt),
|
||||||
CreatedAtISO: formatBrowserTime(summary.CreatedAt),
|
ExpiresAt: formatAdminTime(summary.ExpiresAt),
|
||||||
ExpiresAtLabel: "Not set",
|
Expired: summary.Expired,
|
||||||
ExpiresAtISO: formatBrowserTime(summary.ExpiresAt),
|
|
||||||
RetentionLabel: "Legacy / unmanaged",
|
|
||||||
PasswordProtected: summary.PasswordProtected,
|
|
||||||
OneTimeDownload: summary.OneTimeDownload,
|
OneTimeDownload: summary.OneTimeDownload,
|
||||||
HasManifest: hasManifest,
|
PasswordProtected: summary.PasswordProtected,
|
||||||
OpenURL: "/box/" + summary.ID,
|
|
||||||
Files: make([]adminBoxFileView, 0, len(files)),
|
|
||||||
}
|
|
||||||
|
|
||||||
if !summary.ExpiresAt.IsZero() {
|
|
||||||
boxView.ExpiresAtLabel = adminTimeLabel(summary.ExpiresAt)
|
|
||||||
}
|
|
||||||
|
|
||||||
searchParts := []string{summary.ID, summary.TotalSizeLabel}
|
|
||||||
for _, file := range files {
|
|
||||||
if file.IsComplete {
|
|
||||||
boxView.CompleteFiles++
|
|
||||||
}
|
|
||||||
if file.Status == "failed" {
|
|
||||||
boxView.FailedFiles++
|
|
||||||
}
|
|
||||||
if !file.IsComplete && file.Status != "failed" {
|
|
||||||
boxView.PendingFiles++
|
|
||||||
}
|
|
||||||
|
|
||||||
boxView.Files = append(boxView.Files, adminBoxFileView{
|
|
||||||
Name: file.Name,
|
|
||||||
SizeLabel: file.SizeLabel,
|
|
||||||
MimeType: file.MimeType,
|
|
||||||
Status: file.Status,
|
|
||||||
StatusLabel: file.StatusLabel,
|
|
||||||
DownloadPath: file.DownloadPath,
|
|
||||||
ThumbnailURL: file.ThumbnailURL,
|
|
||||||
IsComplete: file.IsComplete,
|
|
||||||
})
|
})
|
||||||
searchParts = append(searchParts, file.Name, file.MimeType, file.StatusLabel)
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if hasManifest {
|
ctx.HTML(http.StatusOK, "admin_boxes.html", gin.H{
|
||||||
boxView.RetentionLabel = manifest.RetentionLabel
|
"AdminSection": "boxes",
|
||||||
boxView.ZipDisabled = manifest.DisableZip
|
"CurrentUser": app.currentAdminUsername(ctx),
|
||||||
boxView.Consumed = manifest.Consumed
|
"Boxes": rows,
|
||||||
} else {
|
"TotalBoxes": len(rows),
|
||||||
boxView.ZipDisabled = false
|
"TotalStorage": helpers.FormatBytes(totalSize),
|
||||||
}
|
"ExpiredBoxes": expiredCount,
|
||||||
|
})
|
||||||
boxView.ZipAvailable = app.config.ZipDownloadsEnabled && !boxView.ZipDisabled && !boxView.Consumed && boxView.FileCount > 0 && boxView.PendingFiles == 0
|
|
||||||
if boxView.ZipAvailable {
|
|
||||||
boxView.ZipURL = "/box/" + summary.ID + "/download"
|
|
||||||
}
|
|
||||||
|
|
||||||
boxView.Status, boxView.StatusLabel = deriveAdminBoxStatus(hasManifest, summary.Expired, boxView.PendingFiles, boxView.FailedFiles, boxView.Consumed)
|
|
||||||
boxView.Flags = deriveAdminBoxFlags(boxView)
|
|
||||||
searchParts = append(searchParts, boxView.StatusLabel, boxView.RetentionLabel)
|
|
||||||
boxView.SearchText = strings.ToLower(strings.Join(searchParts, " "))
|
|
||||||
|
|
||||||
return boxView, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func deriveAdminBoxStatus(hasManifest bool, expired bool, pendingFiles int, failedFiles int, consumed bool) (string, string) {
|
|
||||||
switch {
|
|
||||||
case !hasManifest:
|
|
||||||
return "legacy", "Legacy"
|
|
||||||
case consumed:
|
|
||||||
return "consumed", "Consumed"
|
|
||||||
case expired:
|
|
||||||
return "expired", "Expired"
|
|
||||||
case pendingFiles > 0:
|
|
||||||
return "uploading", "Uploading"
|
|
||||||
case failedFiles > 0:
|
|
||||||
return "attention", "Needs review"
|
|
||||||
default:
|
|
||||||
return "ready", "Ready"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func deriveAdminBoxFlags(box adminBoxView) []string {
|
|
||||||
flags := make([]string, 0, 5)
|
|
||||||
if box.PasswordProtected {
|
|
||||||
flags = append(flags, "protected")
|
|
||||||
}
|
|
||||||
if box.OneTimeDownload {
|
|
||||||
flags = append(flags, "one-time")
|
|
||||||
}
|
|
||||||
if box.ZipDisabled {
|
|
||||||
flags = append(flags, "zip off")
|
|
||||||
}
|
|
||||||
if !box.HasManifest {
|
|
||||||
flags = append(flags, "legacy")
|
|
||||||
}
|
|
||||||
if box.Consumed {
|
|
||||||
flags = append(flags, "consumed")
|
|
||||||
}
|
|
||||||
return flags
|
|
||||||
}
|
|
||||||
|
|
||||||
func adminTimeLabel(value time.Time) string {
|
|
||||||
if value.IsZero() {
|
|
||||||
return "Not set"
|
|
||||||
}
|
|
||||||
return value.UTC().Format("2006-01-02 15:04 UTC")
|
|
||||||
}
|
|
||||||
|
|
||||||
func adminBoxesActionMessage(action string, processed int, deltaSeconds int64) string {
|
|
||||||
switch action {
|
|
||||||
case "delete":
|
|
||||||
return fmt.Sprintf("Deleted %d box(es)", processed)
|
|
||||||
case "expire":
|
|
||||||
return fmt.Sprintf("Expired %d box(es)", processed)
|
|
||||||
case "bump":
|
|
||||||
return fmt.Sprintf("Extended %d box(es) by %s", processed, adminBoxesDeltaLabel(deltaSeconds))
|
|
||||||
case "cleanup_expired":
|
|
||||||
return fmt.Sprintf("Expired cleanup processed %d box(es)", processed)
|
|
||||||
default:
|
|
||||||
return "Action complete"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func adminBoxesDeltaLabel(deltaSeconds int64) string {
|
|
||||||
switch deltaSeconds {
|
|
||||||
case 24 * 60 * 60:
|
|
||||||
return "24h"
|
|
||||||
case 7 * 24 * 60 * 60:
|
|
||||||
return "7d"
|
|
||||||
default:
|
|
||||||
return (time.Duration(deltaSeconds) * time.Second).String()
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|||||||
14
lib/server/admin_dashboard.go
Normal file
14
lib/server/admin_dashboard.go
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (app *App) handleAdminDashboard(ctx *gin.Context) {
|
||||||
|
ctx.HTML(http.StatusOK, "admin.html", gin.H{
|
||||||
|
"CurrentUser": app.currentAdminUsername(ctx),
|
||||||
|
"CSRFToken": app.currentCSRFToken(ctx),
|
||||||
|
})
|
||||||
|
}
|
||||||
73
lib/server/admin_format.go
Normal file
73
lib/server/admin_format.go
Normal file
@@ -0,0 +1,73 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"errors"
|
||||||
|
"fmt"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func parseOptionalInt64(raw string) (*int64, error) {
|
||||||
|
raw = strings.TrimSpace(raw)
|
||||||
|
if raw == "" {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
value, err := strconv.ParseInt(raw, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return nil, errors.New("must be an integer")
|
||||||
|
}
|
||||||
|
if value < 0 {
|
||||||
|
return nil, errors.New("must be at least 0")
|
||||||
|
}
|
||||||
|
return &value, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func parseCSVInt64(raw string) ([]int64, error) {
|
||||||
|
raw = strings.TrimSpace(raw)
|
||||||
|
if raw == "" {
|
||||||
|
return nil, nil
|
||||||
|
}
|
||||||
|
parts := strings.Split(raw, ",")
|
||||||
|
values := make([]int64, 0, len(parts))
|
||||||
|
for _, part := range parts {
|
||||||
|
part = strings.TrimSpace(part)
|
||||||
|
if part == "" {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
value, err := strconv.ParseInt(part, 10, 64)
|
||||||
|
if err != nil {
|
||||||
|
return nil, fmt.Errorf("allowed expiry durations must be comma-separated seconds")
|
||||||
|
}
|
||||||
|
if value < 0 {
|
||||||
|
return nil, fmt.Errorf("allowed expiry durations must be at least 0")
|
||||||
|
}
|
||||||
|
values = append(values, value)
|
||||||
|
}
|
||||||
|
return values, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func optionalInt64Label(value *int64) string {
|
||||||
|
if value == nil {
|
||||||
|
return "-"
|
||||||
|
}
|
||||||
|
return strconv.FormatInt(*value, 10)
|
||||||
|
}
|
||||||
|
|
||||||
|
func joinInt64s(values []int64) string {
|
||||||
|
if len(values) == 0 {
|
||||||
|
return "-"
|
||||||
|
}
|
||||||
|
parts := make([]string, 0, len(values))
|
||||||
|
for _, value := range values {
|
||||||
|
parts = append(parts, strconv.FormatInt(value, 10))
|
||||||
|
}
|
||||||
|
return strings.Join(parts, ", ")
|
||||||
|
}
|
||||||
|
|
||||||
|
func formatAdminTime(value time.Time) string {
|
||||||
|
if value.IsZero() {
|
||||||
|
return "-"
|
||||||
|
}
|
||||||
|
return value.Local().Format("2006-01-02 15:04:05")
|
||||||
|
}
|
||||||
33
lib/server/admin_routes.go
Normal file
33
lib/server/admin_routes.go
Normal file
@@ -0,0 +1,33 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"net/http"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
)
|
||||||
|
|
||||||
|
func (app *App) registerAdminRoutes(router *gin.Engine) {
|
||||||
|
admin := router.Group("/admin")
|
||||||
|
admin.Use(noStoreAdminHeaders)
|
||||||
|
admin.GET("/login", func(ctx *gin.Context) {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account/login")
|
||||||
|
})
|
||||||
|
admin.POST("/login", app.handleAdminLoginPost)
|
||||||
|
admin.GET("", func(ctx *gin.Context) {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account")
|
||||||
|
})
|
||||||
|
admin.GET("/", func(ctx *gin.Context) {
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/account")
|
||||||
|
})
|
||||||
|
|
||||||
|
protected := admin.Group("")
|
||||||
|
protected.Use(app.requireAdminSession)
|
||||||
|
protected.POST("/logout", app.handleAdminLogout)
|
||||||
|
protected.GET("/boxes", app.handleAdminBoxes)
|
||||||
|
protected.GET("/users", app.handleAdminUsers)
|
||||||
|
protected.POST("/users", app.handleAdminUsersPost)
|
||||||
|
protected.GET("/tags", app.handleAdminTags)
|
||||||
|
protected.POST("/tags", app.handleAdminTagsPost)
|
||||||
|
protected.GET("/settings", app.handleAdminSettings)
|
||||||
|
protected.POST("/settings", app.handleAdminSettingsPost)
|
||||||
|
}
|
||||||
@@ -1,331 +0,0 @@
|
|||||||
package server
|
|
||||||
|
|
||||||
import (
|
|
||||||
"fmt"
|
|
||||||
"net"
|
|
||||||
"net/http"
|
|
||||||
"path/filepath"
|
|
||||||
"strconv"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
|
|
||||||
"warpbox/lib/activity"
|
|
||||||
"warpbox/lib/alerts"
|
|
||||||
"warpbox/lib/security"
|
|
||||||
)
|
|
||||||
|
|
||||||
type adminAlertsActionRequest struct {
|
|
||||||
Action string `json:"action"`
|
|
||||||
IDs []string `json:"ids"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type adminSecurityActionRequest struct {
|
|
||||||
Action string `json:"action"`
|
|
||||||
IP string `json:"ip"`
|
|
||||||
IPs []string `json:"ips"`
|
|
||||||
BanUntil string `json:"ban_until"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) reloadSecurityConfig() error {
|
|
||||||
if app == nil || app.config == nil {
|
|
||||||
return fmt.Errorf("app or config is nil")
|
|
||||||
}
|
|
||||||
if !app.securityFeaturesEnabled() {
|
|
||||||
if app.securityGuard != nil {
|
|
||||||
_ = app.securityGuard.Close()
|
|
||||||
}
|
|
||||||
app.securityGuard = nil
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
if app.securityGuard == nil {
|
|
||||||
app.securityGuard = security.NewGuard()
|
|
||||||
}
|
|
||||||
if err := app.securityGuard.EnableBanPersistence(filepath.Join(app.config.DBDir, "bans.badger")); err != nil {
|
|
||||||
return fmt.Errorf("enable ban persistence: %w", err)
|
|
||||||
}
|
|
||||||
if err := app.securityGuard.Reload(security.Config{
|
|
||||||
IPWhitelist: app.config.SecurityIPWhitelist,
|
|
||||||
AdminIPWhitelist: app.config.SecurityAdminIPWhitelist,
|
|
||||||
LoginWindowSeconds: app.config.SecurityLoginWindowSeconds,
|
|
||||||
LoginMaxAttempts: app.config.SecurityLoginMaxAttempts,
|
|
||||||
BanSeconds: app.config.SecurityBanSeconds,
|
|
||||||
ScanWindowSeconds: app.config.SecurityScanWindowSeconds,
|
|
||||||
ScanMaxAttempts: app.config.SecurityScanMaxAttempts,
|
|
||||||
UploadWindowSeconds: app.config.SecurityUploadWindowSeconds,
|
|
||||||
UploadMaxRequests: app.config.SecurityUploadMaxRequests,
|
|
||||||
UploadMaxBytes: app.config.SecurityUploadMaxBytes,
|
|
||||||
}); err != nil {
|
|
||||||
return fmt.Errorf("reload guard config: %w", err)
|
|
||||||
}
|
|
||||||
return nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) securityFeaturesEnabled() bool {
|
|
||||||
return app != nil && app.config != nil && app.config.SecurityEnabled
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) logActivity(kind string, severity string, message string, ctx *gin.Context, meta map[string]string) {
|
|
||||||
if app.activityStore == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
event := activity.Event{
|
|
||||||
Kind: kind,
|
|
||||||
Severity: severity,
|
|
||||||
Message: message,
|
|
||||||
CreatedAt: time.Now().UTC(),
|
|
||||||
Meta: meta,
|
|
||||||
}
|
|
||||||
if ctx != nil {
|
|
||||||
event.IP = app.clientIP(ctx)
|
|
||||||
event.Path = ctx.Request.URL.Path
|
|
||||||
event.Method = ctx.Request.Method
|
|
||||||
}
|
|
||||||
_ = app.activityStore.Append(event, app.config.ActivityRetentionSeconds)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) createAlert(title string, severity string, group string, code string, trace string, message string, meta map[string]string) {
|
|
||||||
if app.alertStore == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
_ = app.alertStore.Add(alerts.Alert{
|
|
||||||
Title: title,
|
|
||||||
Severity: severity,
|
|
||||||
Group: group,
|
|
||||||
Code: code,
|
|
||||||
Trace: trace,
|
|
||||||
Message: message,
|
|
||||||
Status: alerts.StatusOpen,
|
|
||||||
Meta: meta,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) securityMiddleware() gin.HandlerFunc {
|
|
||||||
return func(ctx *gin.Context) {
|
|
||||||
if !app.securityFeaturesEnabled() {
|
|
||||||
ctx.Next()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if app.securityGuard == nil {
|
|
||||||
ctx.Next()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ip := app.clientIP(ctx)
|
|
||||||
if app.securityGuard.IsWhitelisted(ip) || app.securityGuard.IsAdminWhitelisted(ip) {
|
|
||||||
ctx.Next()
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if app.securityGuard.IsBanned(ip) {
|
|
||||||
app.logActivity("security.block", "high", "Blocked banned IP", ctx, nil)
|
|
||||||
ctx.AbortWithStatusJSON(http.StatusTooManyRequests, gin.H{"error": "Too many abusive requests. Try again later."})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ctx.Next()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleNoRoute(ctx *gin.Context) {
|
|
||||||
if !app.securityFeaturesEnabled() {
|
|
||||||
ctx.JSON(http.StatusNotFound, gin.H{"error": "Not found"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if app.securityGuard == nil {
|
|
||||||
ctx.JSON(http.StatusNotFound, gin.H{"error": "Not found"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
path := strings.ToLower(ctx.Request.URL.Path)
|
|
||||||
suspicious := strings.Contains(path, "../") || strings.Contains(path, ".php") || strings.Contains(path, "wp-admin") || strings.Contains(path, ".env")
|
|
||||||
if suspicious {
|
|
||||||
ip := app.clientIP(ctx)
|
|
||||||
if !app.securityGuard.IsWhitelisted(ip) {
|
|
||||||
banned, attempts := app.securityGuard.RegisterScanAttempt(ip, app.config.SecurityScanWindowSeconds, app.config.SecurityScanMaxAttempts, app.config.SecurityBanSeconds)
|
|
||||||
app.logActivity("security.scan", "medium", "Suspicious path probe detected", ctx, map[string]string{"attempts": intToString(attempts)})
|
|
||||||
if banned {
|
|
||||||
app.createAlert("IP auto-banned after malicious path scans", "high", "security", "410", "security.scan.autoban", "Repeated malicious path scans triggered temporary ban.", map[string]string{"ip": ip, "attempts": intToString(attempts)})
|
|
||||||
app.logActivity("security.ban", "high", "IP auto-banned after scans", ctx, map[string]string{"attempts": intToString(attempts)})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
ctx.JSON(http.StatusNotFound, gin.H{"error": "Not found"})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminActivity(ctx *gin.Context) {
|
|
||||||
if app.activityStore == nil {
|
|
||||||
ctx.HTML(http.StatusOK, "admin/activity.html", gin.H{
|
|
||||||
"AdminUsername": app.config.AdminUsername,
|
|
||||||
"AdminEmail": app.config.AdminEmail,
|
|
||||||
"ActivePage": "activity",
|
|
||||||
"Events": []activity.Event{},
|
|
||||||
})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
events, err := app.activityStore.List(400, app.config.ActivityRetentionSeconds)
|
|
||||||
if err != nil {
|
|
||||||
ctx.String(http.StatusInternalServerError, "Could not load activity")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ctx.HTML(http.StatusOK, "admin/activity.html", gin.H{
|
|
||||||
"AdminUsername": app.config.AdminUsername,
|
|
||||||
"AdminEmail": app.config.AdminEmail,
|
|
||||||
"ActivePage": "activity",
|
|
||||||
"Events": events,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminSecurity(ctx *gin.Context) {
|
|
||||||
if !app.securityFeaturesEnabled() {
|
|
||||||
ctx.String(http.StatusNotFound, "Security features are disabled")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
events := []activity.Event{}
|
|
||||||
alertsList := []alerts.Alert{}
|
|
||||||
if app.activityStore != nil {
|
|
||||||
events, _ = app.activityStore.List(300, app.config.ActivityRetentionSeconds)
|
|
||||||
}
|
|
||||||
if app.alertStore != nil {
|
|
||||||
alertsList, _ = app.alertStore.List(120)
|
|
||||||
}
|
|
||||||
bans := []security.BanEntry{}
|
|
||||||
if app.securityGuard != nil {
|
|
||||||
bans = app.securityGuard.BanList()
|
|
||||||
}
|
|
||||||
ctx.HTML(http.StatusOK, "admin/security.html", gin.H{
|
|
||||||
"AdminUsername": app.config.AdminUsername,
|
|
||||||
"AdminEmail": app.config.AdminEmail,
|
|
||||||
"ActivePage": "security",
|
|
||||||
"Events": events,
|
|
||||||
"Alerts": alertsList,
|
|
||||||
"Bans": bans,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminAlertsAction(ctx *gin.Context) {
|
|
||||||
if app.alertStore == nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Alert store unavailable"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
var request adminAlertsActionRequest
|
|
||||||
if err := ctx.ShouldBindJSON(&request); err != nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Invalid action payload"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
switch request.Action {
|
|
||||||
case "ack":
|
|
||||||
if err := app.alertStore.SetStatus(request.IDs, alerts.StatusAcked); err != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Could not update alerts"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
case "close":
|
|
||||||
if err := app.alertStore.SetStatus(request.IDs, alerts.StatusClosed); err != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Could not update alerts"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
case "delete":
|
|
||||||
if err := app.alertStore.Delete(request.IDs); err != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Could not delete alerts"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
default:
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Unknown action"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
app.logActivity("alerts.action", "low", "Admin changed alert state", ctx, map[string]string{"action": request.Action, "count": intToString(len(request.IDs))})
|
|
||||||
alertsList, _ := app.alertStore.List(500)
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"ok": true, "alerts": alertsList})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) recordManualBanAction(ctx *gin.Context, kind string, message string, severity string, ip string, meta map[string]string, alertTitle string, alertSeverity string, code string, trace string, alertMessage string) {
|
|
||||||
metaCopy := map[string]string{"ip": ip}
|
|
||||||
for k, v := range meta {
|
|
||||||
metaCopy[k] = v
|
|
||||||
}
|
|
||||||
app.logActivity(kind, severity, message, ctx, metaCopy)
|
|
||||||
app.createAlert(alertTitle, alertSeverity, "security", code, trace, alertMessage, metaCopy)
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminSecurityAction(ctx *gin.Context) {
|
|
||||||
if !app.securityFeaturesEnabled() {
|
|
||||||
ctx.JSON(http.StatusNotFound, gin.H{"error": "Security features are disabled"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if app.securityGuard == nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Security guard unavailable"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
var request adminSecurityActionRequest
|
|
||||||
if err := ctx.ShouldBindJSON(&request); err != nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Invalid action payload"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
ip := strings.TrimSpace(request.IP)
|
|
||||||
if ip != "" && net.ParseIP(ip) == nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Invalid IP"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
switch request.Action {
|
|
||||||
case "ban":
|
|
||||||
if ip == "" {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Missing IP"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
app.securityGuard.Ban(ip, app.config.SecurityBanSeconds)
|
|
||||||
app.recordManualBanAction(ctx, "security.manual_ban", "Admin banned IP", "high", ip, nil, "IP manually banned by admin", "medium", "420", "security.manual.ban", "Admin manually applied temporary ban.")
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"ok": true, "message": "IP banned", "bans": app.securityGuard.BanList()})
|
|
||||||
case "ban_until":
|
|
||||||
if ip == "" {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Missing IP"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
until, err := time.Parse(time.RFC3339, strings.TrimSpace(request.BanUntil))
|
|
||||||
if err != nil || until.Before(time.Now().UTC()) {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Invalid ban expiration"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
app.securityGuard.BanUntil(ip, until)
|
|
||||||
meta := map[string]string{"until": until.UTC().Format(time.RFC3339)}
|
|
||||||
app.recordManualBanAction(ctx, "security.manual_ban_until", "Admin set custom ban expiration", "high", ip, meta, "Custom IP ban applied by admin", "medium", "421", "security.manual.ban_until", "Admin set explicit ban expiration date.")
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"ok": true, "message": "IP ban expiration updated", "bans": app.securityGuard.BanList()})
|
|
||||||
case "unban":
|
|
||||||
if ip == "" {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Missing IP"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
app.securityGuard.Unban(ip)
|
|
||||||
app.recordManualBanAction(ctx, "security.manual_unban", "Admin unbanned IP", "medium", ip, nil, "IP unbanned by admin", "low", "422", "security.manual.unban", "Admin manually removed temporary ban.")
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"ok": true, "message": "IP unbanned", "bans": app.securityGuard.BanList()})
|
|
||||||
case "bulk_unban":
|
|
||||||
if len(request.IPs) == 0 {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Missing IP list"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
count := 0
|
|
||||||
for _, candidate := range request.IPs {
|
|
||||||
candidate = strings.TrimSpace(candidate)
|
|
||||||
if net.ParseIP(candidate) == nil {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
app.securityGuard.Unban(candidate)
|
|
||||||
count++
|
|
||||||
}
|
|
||||||
app.logActivity("security.manual_bulk_unban", "high", "Admin unbanned multiple IPs", ctx, map[string]string{"count": intToString(count)})
|
|
||||||
app.createAlert("Bulk IP unban by admin", "medium", "security", "423", "security.manual.bulk_unban", "Admin manually removed multiple temporary bans.", map[string]string{"count": intToString(count)})
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"ok": true, "message": "Bulk unban complete", "bans": app.securityGuard.BanList()})
|
|
||||||
case "unban_all":
|
|
||||||
current := app.securityGuard.BanList()
|
|
||||||
for _, ban := range current {
|
|
||||||
app.securityGuard.Unban(ban.IP)
|
|
||||||
}
|
|
||||||
count := len(current)
|
|
||||||
app.logActivity("security.manual_unban_all", "high", "Admin cleared all active bans", ctx, map[string]string{"count": intToString(count)})
|
|
||||||
app.createAlert("All active bans cleared by admin", "medium", "security", "424", "security.manual.unban_all", "Admin manually removed all temporary bans.", map[string]string{"count": intToString(count)})
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"ok": true, "message": "All bans cleared", "bans": app.securityGuard.BanList()})
|
|
||||||
default:
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Unknown action"})
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func intToString(value int) string {
|
|
||||||
return strconv.Itoa(value)
|
|
||||||
}
|
|
||||||
@@ -1,125 +0,0 @@
|
|||||||
package server
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net/http"
|
|
||||||
"net/http/httptest"
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
|
|
||||||
"warpbox/lib/activity"
|
|
||||||
"warpbox/lib/alerts"
|
|
||||||
"warpbox/lib/config"
|
|
||||||
"warpbox/lib/security"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestAdminSecurityActionsWriteAuditTrail(t *testing.T) {
|
|
||||||
app, router := setupAdminSecurityTest(t)
|
|
||||||
|
|
||||||
for _, body := range []string{
|
|
||||||
`{"action":"ban","ip":"203.0.113.7"}`,
|
|
||||||
`{"action":"unban","ip":"203.0.113.7"}`,
|
|
||||||
} {
|
|
||||||
request := httptest.NewRequest(http.MethodPost, "/admin/security/actions", strings.NewReader(body))
|
|
||||||
request.Header.Set("Content-Type", "application/json")
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
if response.Code != http.StatusOK {
|
|
||||||
t.Fatalf("expected 200, got %d body=%s", response.Code, response.Body.String())
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
events, err := app.activityStore.List(100, app.config.ActivityRetentionSeconds)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("activity list error: %v", err)
|
|
||||||
}
|
|
||||||
if len(events) < 2 {
|
|
||||||
t.Fatalf("expected activity events, got %d", len(events))
|
|
||||||
}
|
|
||||||
alertsList, err := app.alertStore.List(100)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("alerts list error: %v", err)
|
|
||||||
}
|
|
||||||
if len(alertsList) < 2 {
|
|
||||||
t.Fatalf("expected alerts for manual actions, got %d", len(alertsList))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAdminSecurityBulkUnbanAndUnbanAll(t *testing.T) {
|
|
||||||
app, router := setupAdminSecurityTest(t)
|
|
||||||
app.securityGuard.Ban("203.0.113.8", 300)
|
|
||||||
app.securityGuard.Ban("203.0.113.9", 300)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodPost, "/admin/security/actions", strings.NewReader(`{"action":"bulk_unban","ips":["203.0.113.8"]}`))
|
|
||||||
request.Header.Set("Content-Type", "application/json")
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
if response.Code != http.StatusOK {
|
|
||||||
t.Fatalf("bulk_unban expected 200, got %d", response.Code)
|
|
||||||
}
|
|
||||||
if app.securityGuard.IsBanned("203.0.113.8") {
|
|
||||||
t.Fatal("expected selected IP to be unbanned")
|
|
||||||
}
|
|
||||||
if !app.securityGuard.IsBanned("203.0.113.9") {
|
|
||||||
t.Fatal("expected non-selected IP to remain banned")
|
|
||||||
}
|
|
||||||
|
|
||||||
requestAll := httptest.NewRequest(http.MethodPost, "/admin/security/actions", strings.NewReader(`{"action":"unban_all"}`))
|
|
||||||
requestAll.Header.Set("Content-Type", "application/json")
|
|
||||||
requestAll.AddCookie(authCookie(app))
|
|
||||||
responseAll := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(responseAll, requestAll)
|
|
||||||
if responseAll.Code != http.StatusOK {
|
|
||||||
t.Fatalf("unban_all expected 200, got %d", responseAll.Code)
|
|
||||||
}
|
|
||||||
if len(app.securityGuard.BanList()) != 0 {
|
|
||||||
t.Fatal("expected all bans to be removed")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func setupAdminSecurityTest(t *testing.T) (*App, *gin.Engine) {
|
|
||||||
t.Helper()
|
|
||||||
gin.SetMode(gin.TestMode)
|
|
||||||
cwd, _ := os.Getwd()
|
|
||||||
root := filepath.Clean(filepath.Join(cwd, "..", ".."))
|
|
||||||
if err := os.Chdir(root); err != nil {
|
|
||||||
t.Fatalf("chdir: %v", err)
|
|
||||||
}
|
|
||||||
t.Cleanup(func() { _ = os.Chdir(cwd) })
|
|
||||||
|
|
||||||
clearAdminSettingsEnv(t)
|
|
||||||
t.Setenv("WARPBOX_DATA_DIR", t.TempDir())
|
|
||||||
t.Setenv("WARPBOX_ADMIN_PASSWORD", "secret")
|
|
||||||
t.Setenv("WARPBOX_ADMIN_ENABLED", "true")
|
|
||||||
|
|
||||||
cfg, err := config.Load()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("config load: %v", err)
|
|
||||||
}
|
|
||||||
if err := cfg.EnsureDirectories(); err != nil {
|
|
||||||
t.Fatalf("ensure dirs: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
app := &App{
|
|
||||||
config: cfg,
|
|
||||||
activityStore: activity.NewStore(filepath.Join(cfg.DBDir, "activity.json")),
|
|
||||||
alertStore: alerts.NewStore(filepath.Join(cfg.DBDir, "alerts.json")),
|
|
||||||
securityGuard: security.NewGuard(),
|
|
||||||
}
|
|
||||||
if err := app.reloadSecurityConfig(); err != nil {
|
|
||||||
t.Fatalf("reload security config: %v", err)
|
|
||||||
}
|
|
||||||
t.Cleanup(func() { _ = app.securityGuard.Close() })
|
|
||||||
|
|
||||||
router := gin.New()
|
|
||||||
admin := router.Group("/admin")
|
|
||||||
admin.GET("/login", app.handleAdminLogin)
|
|
||||||
protected := router.Group("/admin", app.adminAuthMiddleware)
|
|
||||||
protected.POST("/security/actions", app.handleAdminSecurityAction)
|
|
||||||
return app, router
|
|
||||||
}
|
|
||||||
@@ -1,497 +1,58 @@
|
|||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"fmt"
|
|
||||||
"net/http"
|
"net/http"
|
||||||
"sort"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
"warpbox/lib/config"
|
"warpbox/lib/config"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
)
|
)
|
||||||
|
|
||||||
type adminSettingsCategoryView struct {
|
|
||||||
Key string
|
|
||||||
Label string
|
|
||||||
Icon string
|
|
||||||
Count int
|
|
||||||
Rows []adminSettingRowView
|
|
||||||
}
|
|
||||||
|
|
||||||
type adminSettingRowView struct {
|
|
||||||
Key string `json:"key"`
|
|
||||||
Label string `json:"label"`
|
|
||||||
EnvName string `json:"env_name"`
|
|
||||||
Category string `json:"category"`
|
|
||||||
CategoryLabel string `json:"category_label"`
|
|
||||||
Type string `json:"type"`
|
|
||||||
Value string `json:"value"`
|
|
||||||
DefaultValue string `json:"default_value"`
|
|
||||||
Source string `json:"source"`
|
|
||||||
SourceBadge string `json:"source_badge"`
|
|
||||||
Editable bool `json:"editable"`
|
|
||||||
Locked bool `json:"locked"`
|
|
||||||
HardLimit bool `json:"hard_limit"`
|
|
||||||
Minimum int64 `json:"minimum"`
|
|
||||||
Description string `json:"description"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type adminSettingsSaveRequest struct {
|
|
||||||
Values map[string]string `json:"values"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type adminSettingsImportRequest struct {
|
|
||||||
Settings map[string]string `json:"settings"`
|
|
||||||
EditableSettings map[string]string `json:"editable_settings"`
|
|
||||||
Values map[string]string `json:"values"`
|
|
||||||
Changes map[string]string `json:"changes"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type adminSettingsResetRequest struct {
|
|
||||||
Keys []string `json:"keys"`
|
|
||||||
}
|
|
||||||
|
|
||||||
type adminSettingsExportResponse struct {
|
|
||||||
Format string `json:"format"`
|
|
||||||
ExportedAt string `json:"exported_at"`
|
|
||||||
Settings map[string]string `json:"settings"`
|
|
||||||
EditableSettings map[string]string `json:"editable_settings"`
|
|
||||||
Rows []adminSettingRowView `json:"rows"`
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminSettings(ctx *gin.Context) {
|
func (app *App) handleAdminSettings(ctx *gin.Context) {
|
||||||
rows, categories := app.buildAdminSettingsRows()
|
if !app.requireAdminFlag(ctx, func(perms metastore.EffectivePermissions) bool { return perms.AdminSettingsManage }) {
|
||||||
ctx.HTML(http.StatusOK, "admin/settings.html", gin.H{
|
return
|
||||||
"AdminUsername": app.config.AdminUsername,
|
}
|
||||||
"AdminEmail": app.config.AdminEmail,
|
app.renderAdminSettings(ctx, "")
|
||||||
"ActivePage": "settings",
|
|
||||||
"Rows": rows,
|
|
||||||
"Categories": categories,
|
|
||||||
"RowsJSON": rows,
|
|
||||||
})
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (app *App) handleAdminSettingsExport(ctx *gin.Context) {
|
func (app *App) handleAdminSettingsPost(ctx *gin.Context) {
|
||||||
rows, _ := app.buildAdminSettingsRows()
|
if !app.requireAdminFlag(ctx, func(perms metastore.EffectivePermissions) bool { return perms.AdminSettingsManage }) {
|
||||||
ctx.JSON(http.StatusOK, app.buildSettingsExportPayload(rows))
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminSettingsSave(ctx *gin.Context) {
|
|
||||||
var request adminSettingsSaveRequest
|
|
||||||
if err := ctx.ShouldBindJSON(&request); err != nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Invalid save payload"})
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
currentOverrides, err := config.ReadAdminSettingsOverrides(app.settingsOverridesPath)
|
|
||||||
if err != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Could not load current settings overrides"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if currentOverrides == nil {
|
|
||||||
currentOverrides = map[string]string{}
|
|
||||||
}
|
|
||||||
for key, value := range request.Values {
|
|
||||||
currentOverrides[key] = value
|
|
||||||
}
|
|
||||||
|
|
||||||
rows, warnings, err := app.applySettingsOverrideSet(currentOverrides)
|
|
||||||
if err != nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{
|
|
||||||
"ok": true,
|
|
||||||
"message": fmt.Sprintf("Saved %d editable setting(s)", len(request.Values)),
|
|
||||||
"warnings": warnings,
|
|
||||||
"rows": rows,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminSettingsImport(ctx *gin.Context) {
|
|
||||||
var request adminSettingsImportRequest
|
|
||||||
if err := ctx.ShouldBindJSON(&request); err != nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "Invalid import payload"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
values := request.Values
|
|
||||||
if len(values) == 0 {
|
|
||||||
values = request.Settings
|
|
||||||
}
|
|
||||||
if len(values) == 0 {
|
|
||||||
values = request.EditableSettings
|
|
||||||
}
|
|
||||||
if len(values) == 0 {
|
|
||||||
values = request.Changes
|
|
||||||
}
|
|
||||||
if len(values) == 0 {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": "No importable settings found"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
editable := map[string]bool{}
|
|
||||||
for _, def := range config.EditableDefinitions() {
|
|
||||||
editable[def.Key] = true
|
|
||||||
}
|
|
||||||
filtered := make(map[string]string, len(values))
|
|
||||||
warnings := make([]string, 0)
|
|
||||||
for key, value := range values {
|
|
||||||
if editable[key] {
|
|
||||||
filtered[key] = value
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
if _, found := config.Definition(key); found {
|
|
||||||
warnings = append(warnings, fmt.Sprintf("%s skipped: locked", key))
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
warnings = append(warnings, fmt.Sprintf("%s skipped: unknown key", key))
|
|
||||||
}
|
|
||||||
currentOverrides, err := config.ReadAdminSettingsOverrides(app.settingsOverridesPath)
|
|
||||||
if err != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Could not load current settings overrides"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
for key, value := range currentOverrides {
|
|
||||||
if _, exists := filtered[key]; !exists {
|
|
||||||
filtered[key] = value
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rows, applyWarnings, err := app.applySettingsOverrideSet(filtered)
|
|
||||||
if err != nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
warnings = append(warnings, applyWarnings...)
|
|
||||||
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{
|
|
||||||
"ok": true,
|
|
||||||
"message": fmt.Sprintf("Imported %d setting value(s)", len(values)),
|
|
||||||
"warnings": warnings,
|
|
||||||
"rows": rows,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) handleAdminSettingsReset(ctx *gin.Context) {
|
|
||||||
var request adminSettingsResetRequest
|
|
||||||
_ = ctx.ShouldBindJSON(&request)
|
|
||||||
|
|
||||||
overrideSet, err := config.ReadAdminSettingsOverrides(app.settingsOverridesPath)
|
|
||||||
if err != nil {
|
|
||||||
ctx.JSON(http.StatusInternalServerError, gin.H{"error": "Could not load settings overrides"})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
if overrideSet == nil {
|
|
||||||
overrideSet = map[string]string{}
|
|
||||||
}
|
|
||||||
targetKeys := map[string]bool{}
|
|
||||||
for _, key := range request.Keys {
|
|
||||||
targetKeys[config.NormalizeLegacySettingKey(key)] = true
|
|
||||||
}
|
|
||||||
|
|
||||||
if len(targetKeys) == 0 {
|
|
||||||
overrideSet = map[string]string{}
|
|
||||||
} else {
|
|
||||||
for key := range targetKeys {
|
|
||||||
delete(overrideSet, key)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rows, warnings, err := app.applySettingsOverrideSet(overrideSet)
|
|
||||||
if err != nil {
|
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{
|
|
||||||
"ok": true,
|
|
||||||
"message": "Selected overrides cleared; environment and defaults now apply",
|
|
||||||
"warnings": warnings,
|
|
||||||
"rows": rows,
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) applySettingsOverrideSet(values map[string]string) ([]adminSettingRowView, []string, error) {
|
|
||||||
if !app.config.AllowAdminSettingsOverride {
|
if !app.config.AllowAdminSettingsOverride {
|
||||||
return nil, nil, fmt.Errorf("runtime admin setting overrides are disabled by environment")
|
app.renderAdminSettings(ctx, "Admin settings overrides are disabled by environment configuration.")
|
||||||
}
|
return
|
||||||
if values == nil {
|
|
||||||
values = map[string]string{}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
overrideSet := make(map[string]string, len(values))
|
|
||||||
warnings := make([]string, 0)
|
|
||||||
editable := map[string]config.SettingDefinition{}
|
|
||||||
for _, def := range config.EditableDefinitions() {
|
for _, def := range config.EditableDefinitions() {
|
||||||
editable[def.Key] = def
|
value := ctx.PostForm(def.Key)
|
||||||
|
if def.Type == config.SettingTypeBool {
|
||||||
|
value = "false"
|
||||||
|
if ctx.PostForm(def.Key) == "true" {
|
||||||
|
value = "true"
|
||||||
}
|
}
|
||||||
|
|
||||||
keys := make([]string, 0, len(values))
|
|
||||||
for key := range values {
|
|
||||||
keys = append(keys, key)
|
|
||||||
}
|
}
|
||||||
sort.Strings(keys)
|
if err := app.config.ApplyOverride(def.Key, value); err != nil {
|
||||||
|
app.renderAdminSettings(ctx, err.Error())
|
||||||
for _, key := range keys {
|
return
|
||||||
normalizedKey, normalizedValue, err := config.NormalizeOverrideInput(key, strings.TrimSpace(values[key]))
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, fmt.Errorf("%s: %w", key, err)
|
|
||||||
}
|
}
|
||||||
key = normalizedKey
|
if err := app.store.SetSetting(def.Key, value); err != nil {
|
||||||
value := normalizedValue
|
app.renderAdminSettings(ctx, err.Error())
|
||||||
def, ok := editable[key]
|
return
|
||||||
if !ok {
|
|
||||||
if _, found := config.Definition(key); found {
|
|
||||||
return nil, nil, fmt.Errorf("setting %q is locked and cannot be changed", key)
|
|
||||||
}
|
}
|
||||||
warnings = append(warnings, fmt.Sprintf("%s skipped: unknown key", key))
|
|
||||||
continue
|
|
||||||
}
|
}
|
||||||
if value == "" && def.Type != config.SettingTypeText {
|
|
||||||
return nil, nil, fmt.Errorf("setting %q cannot be blank", key)
|
|
||||||
}
|
|
||||||
overrideSet[key] = value
|
|
||||||
}
|
|
||||||
|
|
||||||
nextCfg, err := config.Load()
|
|
||||||
if err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
if err := nextCfg.ApplyOverrides(overrideSet); err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
if err := config.WriteAdminSettingsOverrides(app.settingsOverridesPath, overrideSet); err != nil {
|
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
|
|
||||||
app.config = nextCfg
|
|
||||||
applyBoxstoreRuntimeConfig(app.config)
|
applyBoxstoreRuntimeConfig(app.config)
|
||||||
if err := app.reloadSecurityConfig(); err != nil {
|
ctx.Redirect(http.StatusSeeOther, "/admin/settings")
|
||||||
return nil, nil, err
|
|
||||||
}
|
|
||||||
rows, _ := app.buildAdminSettingsRows()
|
|
||||||
return rows, warnings, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func (app *App) buildSettingsExportPayload(rows []adminSettingRowView) adminSettingsExportResponse {
|
func (app *App) renderAdminSettings(ctx *gin.Context, errorMessage string) {
|
||||||
settings := make(map[string]string, len(rows))
|
ctx.HTML(http.StatusOK, "admin_settings.html", gin.H{
|
||||||
editable := make(map[string]string)
|
"AdminSection": "settings",
|
||||||
for _, row := range rows {
|
"CurrentUser": app.currentAdminUsername(ctx),
|
||||||
settings[row.Key] = row.Value
|
"CSRFToken": app.currentCSRFToken(ctx),
|
||||||
if row.Editable && !row.Locked {
|
"Rows": app.config.SettingRows(),
|
||||||
editable[row.Key] = row.Value
|
"OverridesAllowed": app.config.AllowAdminSettingsOverride,
|
||||||
}
|
"Error": errorMessage,
|
||||||
}
|
|
||||||
return adminSettingsExportResponse{
|
|
||||||
Format: "warpbox.settings.export.v1",
|
|
||||||
ExportedAt: time.Now().UTC().Format(time.RFC3339),
|
|
||||||
Settings: settings,
|
|
||||||
EditableSettings: editable,
|
|
||||||
Rows: rows,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) buildAdminSettingsRows() ([]adminSettingRowView, []adminSettingsCategoryView) {
|
|
||||||
cfgRows := app.config.SettingRows()
|
|
||||||
rows := make([]adminSettingRowView, 0, len(cfgRows)+5)
|
|
||||||
for _, row := range cfgRows {
|
|
||||||
rows = append(rows, app.makeDefinitionSettingRow(row))
|
|
||||||
}
|
|
||||||
rows = append(rows,
|
|
||||||
app.makeLockedSettingRow("admin_username", "Admin username", "WARPBOX_ADMIN_USERNAME", "accounts", "admin", app.config.AdminUsername, "Environment-controlled admin login name."),
|
|
||||||
app.makeLockedSettingRow("admin_email", "Admin email", "WARPBOX_ADMIN_EMAIL", "accounts", "admin", app.config.AdminEmail, "Administrative contact address used for future account and alert workflows."),
|
|
||||||
app.makeLockedSettingRow("admin_enabled", "Admin enabled mode", "WARPBOX_ADMIN_ENABLED", "accounts", "admin", string(app.config.AdminEnabled), "Controls whether administrative login is disabled, forced on, or auto-detected."),
|
|
||||||
app.makeLockedSettingRow("admin_cookie_secure", "Admin cookie secure", "WARPBOX_ADMIN_COOKIE_SECURE", "accounts", "bool", boolString(app.config.AdminCookieSecure), "Secure admin cookie flag. Locking this avoids accidental auth regressions."),
|
|
||||||
app.makeLockedSettingRow("allow_admin_settings_override", "Admin settings override allowed", "WARPBOX_ALLOW_ADMIN_SETTINGS_OVERRIDE", "accounts", "bool", boolString(app.config.AllowAdminSettingsOverride), "Master switch for runtime admin setting overrides."),
|
|
||||||
)
|
|
||||||
|
|
||||||
sort.Slice(rows, func(i, j int) bool {
|
|
||||||
if rows[i].Category == rows[j].Category {
|
|
||||||
return rows[i].Label < rows[j].Label
|
|
||||||
}
|
|
||||||
return settingsCategoryRank(rows[i].Category) < settingsCategoryRank(rows[j].Category)
|
|
||||||
})
|
})
|
||||||
|
|
||||||
categoryMeta := settingsCategoryMeta()
|
|
||||||
categories := make([]adminSettingsCategoryView, 0, len(categoryMeta)+1)
|
|
||||||
allCategory := adminSettingsCategoryView{Key: "all", Label: "All settings", Icon: "▤", Count: len(rows)}
|
|
||||||
categories = append(categories, allCategory)
|
|
||||||
|
|
||||||
grouped := map[string][]adminSettingRowView{}
|
|
||||||
for _, row := range rows {
|
|
||||||
grouped[row.Category] = append(grouped[row.Category], row)
|
|
||||||
}
|
|
||||||
|
|
||||||
for _, meta := range categoryMeta {
|
|
||||||
categories = append(categories, adminSettingsCategoryView{
|
|
||||||
Key: meta.Key,
|
|
||||||
Label: meta.Label,
|
|
||||||
Icon: meta.Icon,
|
|
||||||
Count: len(grouped[meta.Key]),
|
|
||||||
Rows: grouped[meta.Key],
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|
||||||
return rows, categories
|
|
||||||
}
|
|
||||||
|
|
||||||
func boolString(value bool) string {
|
|
||||||
if value {
|
|
||||||
return "true"
|
|
||||||
}
|
|
||||||
return "false"
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) makeDefinitionSettingRow(row config.SettingRow) adminSettingRowView {
|
|
||||||
def := row.Definition
|
|
||||||
locked := !def.Editable || def.HardLimit
|
|
||||||
source := string(row.Source)
|
|
||||||
sourceBadge := source
|
|
||||||
if locked {
|
|
||||||
sourceBadge = "hard env"
|
|
||||||
}
|
|
||||||
return adminSettingRowView{
|
|
||||||
Key: def.Key,
|
|
||||||
Label: def.Label,
|
|
||||||
EnvName: def.EnvName,
|
|
||||||
Category: settingsCategoryForKey(def.Key),
|
|
||||||
CategoryLabel: settingsCategoryLabel(settingsCategoryForKey(def.Key)),
|
|
||||||
Type: string(def.Type),
|
|
||||||
Value: row.Value,
|
|
||||||
DefaultValue: app.config.DefaultValue(def.Key),
|
|
||||||
Source: source,
|
|
||||||
SourceBadge: sourceBadge,
|
|
||||||
Editable: def.Editable && !def.HardLimit,
|
|
||||||
Locked: locked,
|
|
||||||
HardLimit: def.HardLimit,
|
|
||||||
Minimum: def.Minimum,
|
|
||||||
Description: settingsDescription(def.Key),
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) makeLockedSettingRow(key string, label string, envName string, category string, rowType string, value string, description string) adminSettingRowView {
|
|
||||||
return adminSettingRowView{
|
|
||||||
Key: key,
|
|
||||||
Label: label,
|
|
||||||
EnvName: envName,
|
|
||||||
Category: category,
|
|
||||||
CategoryLabel: settingsCategoryLabel(category),
|
|
||||||
Type: rowType,
|
|
||||||
Value: value,
|
|
||||||
DefaultValue: "",
|
|
||||||
Source: "environment",
|
|
||||||
SourceBadge: "hard env",
|
|
||||||
Editable: false,
|
|
||||||
Locked: true,
|
|
||||||
HardLimit: true,
|
|
||||||
Description: description,
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
type settingsCategoryInfo struct {
|
|
||||||
Key string
|
|
||||||
Label string
|
|
||||||
Icon string
|
|
||||||
}
|
|
||||||
|
|
||||||
func settingsCategoryMeta() []settingsCategoryInfo {
|
|
||||||
return []settingsCategoryInfo{
|
|
||||||
{Key: "uploads", Label: "Uploads", Icon: "↥"},
|
|
||||||
{Key: "downloads", Label: "Downloads", Icon: "↧"},
|
|
||||||
{Key: "retention", Label: "Retention", Icon: "⌛"},
|
|
||||||
{Key: "security", Label: "Security", Icon: "🔒"},
|
|
||||||
{Key: "activity", Label: "Activity", Icon: "☰"},
|
|
||||||
{Key: "accounts", Label: "Accounts", Icon: "☺"},
|
|
||||||
{Key: "api", Label: "API", Icon: "{ }"},
|
|
||||||
{Key: "storage", Label: "Storage", Icon: "▥"},
|
|
||||||
{Key: "workers", Label: "Workers", Icon: "⚙"},
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func settingsCategoryLabel(key string) string {
|
|
||||||
for _, meta := range settingsCategoryMeta() {
|
|
||||||
if meta.Key == key {
|
|
||||||
return meta.Label
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return "General"
|
|
||||||
}
|
|
||||||
|
|
||||||
func settingsCategoryRank(key string) int {
|
|
||||||
for index, meta := range settingsCategoryMeta() {
|
|
||||||
if meta.Key == key {
|
|
||||||
return index
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return len(settingsCategoryMeta()) + 1
|
|
||||||
}
|
|
||||||
|
|
||||||
func settingsCategoryForKey(key string) string {
|
|
||||||
switch key {
|
|
||||||
case config.SettingGuestUploadsEnabled, config.SettingDefaultUserMaxFileBytes, config.SettingDefaultUserMaxBoxBytes, config.SettingGlobalMaxFileSizeBytes, config.SettingGlobalMaxBoxSizeBytes:
|
|
||||||
return "uploads"
|
|
||||||
case config.SettingSecurityUploadWindowSecs, config.SettingSecurityUploadMaxRequests, config.SettingSecurityUploadMaxGB:
|
|
||||||
return "uploads"
|
|
||||||
case config.SettingZipDownloadsEnabled, config.SettingOneTimeDownloadsEnabled, config.SettingOneTimeDownloadExpirySecs, config.SettingRenewOnDownloadEnabled:
|
|
||||||
return "downloads"
|
|
||||||
case config.SettingRenewOnAccessEnabled, config.SettingDefaultGuestExpirySecs, config.SettingMaxGuestExpirySecs, config.SettingOneTimeDownloadRetryFail:
|
|
||||||
return "retention"
|
|
||||||
case config.SettingSecurityEnabled, config.SettingSecurityIPWhitelist, config.SettingSecurityAdminIPWhitelist, config.SettingSecurityLoginWindowSecs, config.SettingSecurityLoginMaxAttempts, config.SettingSecurityBanSeconds, config.SettingSecurityScanWindowSecs, config.SettingSecurityScanMaxAttempts:
|
|
||||||
return "security"
|
|
||||||
case config.SettingActivityRetentionSeconds:
|
|
||||||
return "activity"
|
|
||||||
case config.SettingSessionTTLSeconds:
|
|
||||||
return "accounts"
|
|
||||||
case config.SettingAPIEnabled:
|
|
||||||
return "api"
|
|
||||||
case config.SettingDataDir:
|
|
||||||
return "storage"
|
|
||||||
case config.SettingBoxPollIntervalMS, config.SettingThumbnailBatchSize, config.SettingThumbnailIntervalSeconds:
|
|
||||||
return "workers"
|
|
||||||
case config.SettingExpiredCleanupIntervalSecs:
|
|
||||||
return "workers"
|
|
||||||
default:
|
|
||||||
return "accounts"
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func settingsDescription(key string) string {
|
|
||||||
descriptions := map[string]string{
|
|
||||||
config.SettingGuestUploadsEnabled: "Allow unauthenticated guests to create boxes through the public upload flow.",
|
|
||||||
config.SettingAPIEnabled: "Enable API endpoints used by the browser upload and status workflows.",
|
|
||||||
config.SettingZipDownloadsEnabled: "Allow archive downloads for full boxes when ZIP is supported.",
|
|
||||||
config.SettingOneTimeDownloadsEnabled: "Enable one-time download retention mode for boxes.",
|
|
||||||
config.SettingOneTimeDownloadExpirySecs: "Expiry window, in seconds, for one-time download boxes after upload completion.",
|
|
||||||
config.SettingOneTimeDownloadRetryFail: "When enabled by environment, failed one-time ZIP writes leave the box retryable.",
|
|
||||||
config.SettingRenewOnAccessEnabled: "Extend retention when a box page is viewed.",
|
|
||||||
config.SettingRenewOnDownloadEnabled: "Extend retention when file or ZIP downloads happen.",
|
|
||||||
config.SettingDefaultGuestExpirySecs: "Default retention presented to guest uploads.",
|
|
||||||
config.SettingMaxGuestExpirySecs: "Maximum retention guests may request.",
|
|
||||||
config.SettingGlobalMaxFileSizeBytes: "Global single-file upload ceiling in GB applied to future requests across the whole app. Decimal values allowed.",
|
|
||||||
config.SettingGlobalMaxBoxSizeBytes: "Global total box size ceiling in GB applied to future requests across the whole app. Decimal values allowed.",
|
|
||||||
config.SettingDefaultUserMaxFileBytes: "Default per-user file size ceiling in GB used by future account-aware flows. Decimal values allowed.",
|
|
||||||
config.SettingDefaultUserMaxBoxBytes: "Default per-user box size ceiling in GB used by future account-aware flows. Decimal values allowed.",
|
|
||||||
config.SettingSessionTTLSeconds: "Lifetime for authenticated browser sessions, including admin session cookies.",
|
|
||||||
config.SettingBoxPollIntervalMS: "Browser polling cadence for box status refreshes.",
|
|
||||||
config.SettingThumbnailBatchSize: "How many thumbnail jobs the worker handles per batch.",
|
|
||||||
config.SettingThumbnailIntervalSeconds: "Delay between thumbnail worker passes.",
|
|
||||||
config.SettingDataDir: "Root data path. Locked because moving storage roots live is risky.",
|
|
||||||
config.SettingActivityRetentionSeconds: "How long activity events stay stored before automatic prune.",
|
|
||||||
config.SettingSecurityEnabled: "Master switch for security middleware, automated bans, suspicious path detection, and upload throttling.",
|
|
||||||
config.SettingSecurityIPWhitelist: "Comma-separated IPs that bypass generic security bans and rate-limits.",
|
|
||||||
config.SettingSecurityAdminIPWhitelist: "Comma-separated IPs allowed to bypass admin login brute-force controls.",
|
|
||||||
config.SettingSecurityLoginWindowSecs: "Window used for failed admin login counting.",
|
|
||||||
config.SettingSecurityLoginMaxAttempts: "Max failed admin logins per window before temporary ban.",
|
|
||||||
config.SettingSecurityBanSeconds: "Duration for automatic temporary IP bans.",
|
|
||||||
config.SettingSecurityScanWindowSecs: "Window used for malicious path scan detection.",
|
|
||||||
config.SettingSecurityScanMaxAttempts: "Max suspicious path probes per window before temporary ban.",
|
|
||||||
config.SettingSecurityUploadWindowSecs: "Window used for per-IP upload throttling.",
|
|
||||||
config.SettingSecurityUploadMaxRequests: "Max upload requests per IP per upload window.",
|
|
||||||
config.SettingSecurityUploadMaxGB: "Max upload volume in GB per IP per upload window.",
|
|
||||||
config.SettingExpiredCleanupIntervalSecs: "Background interval for deleting expired boxes. Set 0 to disable periodic cleanup.",
|
|
||||||
}
|
|
||||||
return descriptions[key]
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,300 +0,0 @@
|
|||||||
package server
|
|
||||||
|
|
||||||
import (
|
|
||||||
"encoding/json"
|
|
||||||
"net/http"
|
|
||||||
"net/http/httptest"
|
|
||||||
"os"
|
|
||||||
"path/filepath"
|
|
||||||
"strings"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
|
|
||||||
"warpbox/lib/config"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestAdminSettingsRequiresAuth(t *testing.T) {
|
|
||||||
app, router := setupAdminSettingsTest(t)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodGet, "/admin/settings", nil)
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
|
|
||||||
if response.Code != http.StatusSeeOther {
|
|
||||||
t.Fatalf("expected redirect, got %d", response.Code)
|
|
||||||
}
|
|
||||||
if location := response.Header().Get("Location"); location != "/admin/login" {
|
|
||||||
t.Fatalf("expected login redirect, got %q", location)
|
|
||||||
}
|
|
||||||
if app == nil {
|
|
||||||
t.Fatal("expected app setup")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAdminSettingsPageRenders(t *testing.T) {
|
|
||||||
app, router := setupAdminSettingsTest(t)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodGet, "/admin/settings", nil)
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
|
|
||||||
if response.Code != http.StatusOK {
|
|
||||||
t.Fatalf("expected 200, got %d", response.Code)
|
|
||||||
}
|
|
||||||
body := response.Body.String()
|
|
||||||
if !strings.Contains(body, "WarpBox Settings") {
|
|
||||||
t.Fatalf("expected settings page title, got %s", body)
|
|
||||||
}
|
|
||||||
if !strings.Contains(body, "WARPBOX_API_ENABLED") {
|
|
||||||
t.Fatalf("expected API env var in page body")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAdminSettingsExportIncludesCurrentValues(t *testing.T) {
|
|
||||||
app, router := setupAdminSettingsTest(t)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodGet, "/admin/settings/export", nil)
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
|
|
||||||
if response.Code != http.StatusOK {
|
|
||||||
t.Fatalf("expected 200, got %d", response.Code)
|
|
||||||
}
|
|
||||||
|
|
||||||
var payload struct {
|
|
||||||
Format string `json:"format"`
|
|
||||||
Settings map[string]string `json:"settings"`
|
|
||||||
}
|
|
||||||
if err := json.Unmarshal(response.Body.Bytes(), &payload); err != nil {
|
|
||||||
t.Fatalf("json.Unmarshal returned error: %v", err)
|
|
||||||
}
|
|
||||||
if payload.Format != "warpbox.settings.export.v1" {
|
|
||||||
t.Fatalf("unexpected export format: %q", payload.Format)
|
|
||||||
}
|
|
||||||
if payload.Settings[config.SettingAPIEnabled] != "false" {
|
|
||||||
t.Fatalf("expected api_enabled to reflect environment false, got %q", payload.Settings[config.SettingAPIEnabled])
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAdminSettingsSavePersistsEditableOverrides(t *testing.T) {
|
|
||||||
app, router := setupAdminSettingsTest(t)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodPost, "/admin/settings/save", strings.NewReader(`{"values":{"api_enabled":"true","box_poll_interval_ms":"6000","global_max_file_size_gb":"0.5"}}`))
|
|
||||||
request.Header.Set("Content-Type", "application/json")
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
|
|
||||||
if response.Code != http.StatusOK {
|
|
||||||
t.Fatalf("expected 200, got %d: %s", response.Code, response.Body.String())
|
|
||||||
}
|
|
||||||
if !app.config.APIEnabled {
|
|
||||||
t.Fatal("expected APIEnabled override to be applied")
|
|
||||||
}
|
|
||||||
if app.config.BoxPollIntervalMS != 6000 {
|
|
||||||
t.Fatalf("expected poll interval override, got %d", app.config.BoxPollIntervalMS)
|
|
||||||
}
|
|
||||||
if app.config.GlobalMaxFileSizeBytes != 512*1024*1024 {
|
|
||||||
t.Fatalf("expected size override in bytes, got %d", app.config.GlobalMaxFileSizeBytes)
|
|
||||||
}
|
|
||||||
|
|
||||||
overrides, err := config.ReadAdminSettingsOverrides(app.settingsOverridesPath)
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("ReadAdminSettingsOverrides returned error: %v", err)
|
|
||||||
}
|
|
||||||
if overrides[config.SettingAPIEnabled] != "true" {
|
|
||||||
t.Fatalf("expected persisted API override, got %#v", overrides)
|
|
||||||
}
|
|
||||||
if _, exists := overrides[config.SettingBoxPollIntervalMS]; !exists {
|
|
||||||
t.Fatalf("expected changed poll interval override to be persisted, got %#v", overrides)
|
|
||||||
}
|
|
||||||
if _, exists := overrides[config.SettingSessionTTLSeconds]; exists {
|
|
||||||
t.Fatalf("expected untouched setting to stay out of overrides, got %#v", overrides)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAdminSettingsSaveRejectsLockedSetting(t *testing.T) {
|
|
||||||
app, router := setupAdminSettingsTest(t)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodPost, "/admin/settings/save", strings.NewReader(`{"values":{"data_dir":"./other"}}`))
|
|
||||||
request.Header.Set("Content-Type", "application/json")
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
|
|
||||||
if response.Code != http.StatusBadRequest {
|
|
||||||
t.Fatalf("expected 400, got %d", response.Code)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAdminSettingsImportSkipsLockedAndUnknownKeys(t *testing.T) {
|
|
||||||
app, router := setupAdminSettingsTest(t)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodPost, "/admin/settings/import", strings.NewReader(`{"settings":{"api_enabled":"true","data_dir":"./other","bogus":"x"}}`))
|
|
||||||
request.Header.Set("Content-Type", "application/json")
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
|
|
||||||
if response.Code != http.StatusOK {
|
|
||||||
t.Fatalf("expected 200, got %d: %s", response.Code, response.Body.String())
|
|
||||||
}
|
|
||||||
if !app.config.APIEnabled {
|
|
||||||
t.Fatal("expected editable import value to apply")
|
|
||||||
}
|
|
||||||
|
|
||||||
var payload struct {
|
|
||||||
Warnings []string `json:"warnings"`
|
|
||||||
}
|
|
||||||
if err := json.Unmarshal(response.Body.Bytes(), &payload); err != nil {
|
|
||||||
t.Fatalf("json.Unmarshal returned error: %v", err)
|
|
||||||
}
|
|
||||||
if len(payload.Warnings) != 2 {
|
|
||||||
t.Fatalf("expected 2 warnings, got %#v", payload.Warnings)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAdminSettingsResetUsesBuiltInDefaults(t *testing.T) {
|
|
||||||
app, router := setupAdminSettingsTest(t)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodPost, "/admin/settings/reset", strings.NewReader(`{}`))
|
|
||||||
request.Header.Set("Content-Type", "application/json")
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
|
|
||||||
if response.Code != http.StatusOK {
|
|
||||||
t.Fatalf("expected 200, got %d: %s", response.Code, response.Body.String())
|
|
||||||
}
|
|
||||||
if app.config.APIEnabled {
|
|
||||||
t.Fatal("expected reset to respect environment and restore APIEnabled=false")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func setupAdminSettingsTest(t *testing.T) (*App, *gin.Engine) {
|
|
||||||
t.Helper()
|
|
||||||
gin.SetMode(gin.TestMode)
|
|
||||||
cwd, err := os.Getwd()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Getwd returned error: %v", err)
|
|
||||||
}
|
|
||||||
root := filepath.Clean(filepath.Join(cwd, "..", ".."))
|
|
||||||
if err := os.Chdir(root); err != nil {
|
|
||||||
t.Fatalf("Chdir returned error: %v", err)
|
|
||||||
}
|
|
||||||
t.Cleanup(func() {
|
|
||||||
_ = os.Chdir(cwd)
|
|
||||||
})
|
|
||||||
clearAdminSettingsEnv(t)
|
|
||||||
t.Setenv("WARPBOX_DATA_DIR", t.TempDir())
|
|
||||||
t.Setenv("WARPBOX_ADMIN_PASSWORD", "secret")
|
|
||||||
t.Setenv("WARPBOX_ADMIN_ENABLED", "true")
|
|
||||||
t.Setenv("WARPBOX_API_ENABLED", "false")
|
|
||||||
|
|
||||||
cfg, err := config.Load()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("Load returned error: %v", err)
|
|
||||||
}
|
|
||||||
if err := cfg.EnsureDirectories(); err != nil {
|
|
||||||
t.Fatalf("EnsureDirectories returned error: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
app := &App{
|
|
||||||
config: cfg,
|
|
||||||
settingsOverridesPath: filepath.Join(cfg.DBDir, config.AdminSettingsOverrideFilename),
|
|
||||||
}
|
|
||||||
|
|
||||||
htmlTemplates, err := loadHTMLTemplates()
|
|
||||||
if err != nil {
|
|
||||||
t.Fatalf("loadHTMLTemplates returned error: %v", err)
|
|
||||||
}
|
|
||||||
|
|
||||||
router := gin.New()
|
|
||||||
router.SetHTMLTemplate(htmlTemplates)
|
|
||||||
admin := router.Group("/admin")
|
|
||||||
admin.GET("/login", app.handleAdminLogin)
|
|
||||||
protected := router.Group("/admin", app.adminAuthMiddleware)
|
|
||||||
protected.GET("/settings", app.handleAdminSettings)
|
|
||||||
protected.GET("/settings/export", app.handleAdminSettingsExport)
|
|
||||||
protected.POST("/settings/save", app.handleAdminSettingsSave)
|
|
||||||
protected.POST("/settings/import", app.handleAdminSettingsImport)
|
|
||||||
protected.POST("/settings/reset", app.handleAdminSettingsReset)
|
|
||||||
return app, router
|
|
||||||
}
|
|
||||||
|
|
||||||
func authCookie(app *App) *http.Cookie {
|
|
||||||
return &http.Cookie{Name: adminSessionCookie, Value: app.adminSessionToken()}
|
|
||||||
}
|
|
||||||
|
|
||||||
func clearAdminSettingsEnv(t *testing.T) {
|
|
||||||
t.Helper()
|
|
||||||
for _, name := range []string{
|
|
||||||
"WARPBOX_DATA_DIR",
|
|
||||||
"WARPBOX_ADMIN_PASSWORD",
|
|
||||||
"WARPBOX_ADMIN_USERNAME",
|
|
||||||
"WARPBOX_ADMIN_EMAIL",
|
|
||||||
"WARPBOX_ADMIN_ENABLED",
|
|
||||||
"WARPBOX_ALLOW_ADMIN_SETTINGS_OVERRIDE",
|
|
||||||
"WARPBOX_ADMIN_COOKIE_SECURE",
|
|
||||||
"WARPBOX_GUEST_UPLOADS_ENABLED",
|
|
||||||
"WARPBOX_API_ENABLED",
|
|
||||||
"WARPBOX_ZIP_DOWNLOADS_ENABLED",
|
|
||||||
"WARPBOX_ONE_TIME_DOWNLOADS_ENABLED",
|
|
||||||
"WARPBOX_ONE_TIME_DOWNLOAD_EXPIRY_SECONDS",
|
|
||||||
"WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE",
|
|
||||||
"WARPBOX_RENEW_ON_ACCESS_ENABLED",
|
|
||||||
"WARPBOX_RENEW_ON_DOWNLOAD_ENABLED",
|
|
||||||
"WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS",
|
|
||||||
"WARPBOX_MAX_GUEST_EXPIRY_SECONDS",
|
|
||||||
"WARPBOX_GLOBAL_MAX_FILE_SIZE_GB",
|
|
||||||
"WARPBOX_GLOBAL_MAX_FILE_SIZE_MB",
|
|
||||||
"WARPBOX_GLOBAL_MAX_FILE_SIZE_BYTES",
|
|
||||||
"WARPBOX_GLOBAL_MAX_BOX_SIZE_GB",
|
|
||||||
"WARPBOX_GLOBAL_MAX_BOX_SIZE_MB",
|
|
||||||
"WARPBOX_GLOBAL_MAX_BOX_SIZE_BYTES",
|
|
||||||
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_GB",
|
|
||||||
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_MB",
|
|
||||||
"WARPBOX_DEFAULT_USER_MAX_FILE_SIZE_BYTES",
|
|
||||||
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_GB",
|
|
||||||
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_MB",
|
|
||||||
"WARPBOX_DEFAULT_USER_MAX_BOX_SIZE_BYTES",
|
|
||||||
"WARPBOX_SESSION_TTL_SECONDS",
|
|
||||||
"WARPBOX_BOX_POLL_INTERVAL_MS",
|
|
||||||
"WARPBOX_THUMBNAIL_BATCH_SIZE",
|
|
||||||
"WARPBOX_THUMBNAIL_INTERVAL_SECONDS",
|
|
||||||
"WARPBOX_SECURITY_ENABLED",
|
|
||||||
"WARPBOX_SECURITY_IP_WHITELIST",
|
|
||||||
"WARPBOX_SECURITY_ADMIN_IP_WHITELIST",
|
|
||||||
"WARPBOX_TRUSTED_PROXY_CIDRS",
|
|
||||||
"WARPBOX_SECURITY_LOGIN_WINDOW_SECONDS",
|
|
||||||
"WARPBOX_SECURITY_LOGIN_MAX_ATTEMPTS",
|
|
||||||
"WARPBOX_SECURITY_BAN_SECONDS",
|
|
||||||
"WARPBOX_SECURITY_SCAN_WINDOW_SECONDS",
|
|
||||||
"WARPBOX_SECURITY_SCAN_MAX_ATTEMPTS",
|
|
||||||
"WARPBOX_SECURITY_UPLOAD_WINDOW_SECONDS",
|
|
||||||
"WARPBOX_SECURITY_UPLOAD_MAX_REQUESTS",
|
|
||||||
"WARPBOX_SECURITY_UPLOAD_MAX_GB",
|
|
||||||
"WARPBOX_SECURITY_UPLOAD_MAX_MB",
|
|
||||||
"WARPBOX_SECURITY_UPLOAD_MAX_BYTES",
|
|
||||||
"WARPBOX_EXPIRED_CLEANUP_INTERVAL_SECONDS",
|
|
||||||
} {
|
|
||||||
t.Setenv(name, "")
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestAdminSettingsSaveRejectsInvalidTrustedProxyCIDR(t *testing.T) {
|
|
||||||
app, router := setupAdminSettingsTest(t)
|
|
||||||
|
|
||||||
request := httptest.NewRequest(http.MethodPost, "/admin/settings/save", strings.NewReader(`{"values":{"trusted_proxy_cidrs":"not-a-cidr"}}`))
|
|
||||||
request.Header.Set("Content-Type", "application/json")
|
|
||||||
request.AddCookie(authCookie(app))
|
|
||||||
response := httptest.NewRecorder()
|
|
||||||
router.ServeHTTP(response, request)
|
|
||||||
|
|
||||||
if response.Code != http.StatusBadRequest {
|
|
||||||
t.Fatalf("expected 400, got %d", response.Code)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
122
lib/server/admin_tags.go
Normal file
122
lib/server/admin_tags.go
Normal file
@@ -0,0 +1,122 @@
|
|||||||
|
package server
|
||||||
|
|
||||||
|
import (
|
||||||
|
"fmt"
|
||||||
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/metastore"
|
||||||
|
)
|
||||||
|
|
||||||
|
type adminTagRow struct {
|
||||||
|
ID string
|
||||||
|
Name string
|
||||||
|
Description string
|
||||||
|
Protected bool
|
||||||
|
AdminAccess bool
|
||||||
|
UploadAllowed bool
|
||||||
|
ZipDownloadAllowed bool
|
||||||
|
OneTimeDownloadAllowed bool
|
||||||
|
RenewableAllowed bool
|
||||||
|
MaxFileSizeBytes string
|
||||||
|
MaxBoxSizeBytes string
|
||||||
|
AllowedExpirySeconds string
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAdminTags(ctx *gin.Context) {
|
||||||
|
if !app.requireAdminFlag(ctx, func(perms metastore.EffectivePermissions) bool { return perms.AdminUsersManage }) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
app.renderAdminTags(ctx, "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAdminTagsPost(ctx *gin.Context) {
|
||||||
|
if !app.requireAdminFlag(ctx, func(perms metastore.EffectivePermissions) bool { return perms.AdminUsersManage }) {
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
perms, err := parseTagPermissions(ctx)
|
||||||
|
if err != nil {
|
||||||
|
app.renderAdminTags(ctx, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
tag := metastore.Tag{
|
||||||
|
Name: ctx.PostForm("name"),
|
||||||
|
Description: ctx.PostForm("description"),
|
||||||
|
Permissions: perms,
|
||||||
|
}
|
||||||
|
if err := app.store.CreateTag(&tag); err != nil {
|
||||||
|
app.renderAdminTags(ctx, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin/tags")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) renderAdminTags(ctx *gin.Context, errorMessage string) {
|
||||||
|
tags, err := app.store.ListTags()
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not list tags")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
sort.Slice(tags, func(i int, j int) bool {
|
||||||
|
return strings.ToLower(tags[i].Name) < strings.ToLower(tags[j].Name)
|
||||||
|
})
|
||||||
|
rows := make([]adminTagRow, 0, len(tags))
|
||||||
|
for _, tag := range tags {
|
||||||
|
rows = append(rows, adminTagRow{
|
||||||
|
ID: tag.ID,
|
||||||
|
Name: tag.Name,
|
||||||
|
Description: tag.Description,
|
||||||
|
Protected: tag.Protected,
|
||||||
|
AdminAccess: tag.Permissions.AdminAccess,
|
||||||
|
UploadAllowed: tag.Permissions.UploadAllowed,
|
||||||
|
ZipDownloadAllowed: tag.Permissions.ZipDownloadAllowed,
|
||||||
|
OneTimeDownloadAllowed: tag.Permissions.OneTimeDownloadAllowed,
|
||||||
|
RenewableAllowed: tag.Permissions.RenewableAllowed,
|
||||||
|
MaxFileSizeBytes: optionalInt64Label(tag.Permissions.MaxFileSizeBytes),
|
||||||
|
MaxBoxSizeBytes: optionalInt64Label(tag.Permissions.MaxBoxSizeBytes),
|
||||||
|
AllowedExpirySeconds: joinInt64s(tag.Permissions.AllowedExpirySeconds),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
ctx.HTML(http.StatusOK, "admin_tags.html", gin.H{
|
||||||
|
"AdminSection": "tags",
|
||||||
|
"CurrentUser": app.currentAdminUsername(ctx),
|
||||||
|
"CSRFToken": app.currentCSRFToken(ctx),
|
||||||
|
"Tags": rows,
|
||||||
|
"Error": errorMessage,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
func parseTagPermissions(ctx *gin.Context) (metastore.TagPermissions, error) {
|
||||||
|
maxFileSize, err := parseOptionalInt64(ctx.PostForm("max_file_size_bytes"))
|
||||||
|
if err != nil {
|
||||||
|
return metastore.TagPermissions{}, fmt.Errorf("max file size bytes %w", err)
|
||||||
|
}
|
||||||
|
maxBoxSize, err := parseOptionalInt64(ctx.PostForm("max_box_size_bytes"))
|
||||||
|
if err != nil {
|
||||||
|
return metastore.TagPermissions{}, fmt.Errorf("max box size bytes %w", err)
|
||||||
|
}
|
||||||
|
expirySeconds, err := parseCSVInt64(ctx.PostForm("allowed_expiry_seconds"))
|
||||||
|
if err != nil {
|
||||||
|
return metastore.TagPermissions{}, err
|
||||||
|
}
|
||||||
|
return metastore.TagPermissions{
|
||||||
|
UploadAllowed: checkbox(ctx, "upload_allowed"),
|
||||||
|
AllowedExpirySeconds: expirySeconds,
|
||||||
|
MaxFileSizeBytes: maxFileSize,
|
||||||
|
MaxBoxSizeBytes: maxBoxSize,
|
||||||
|
OneTimeDownloadAllowed: checkbox(ctx, "one_time_download_allowed"),
|
||||||
|
ZipDownloadAllowed: checkbox(ctx, "zip_download_allowed"),
|
||||||
|
RenewableAllowed: checkbox(ctx, "renewable_allowed"),
|
||||||
|
AdminAccess: checkbox(ctx, "admin_access"),
|
||||||
|
AdminUsersManage: checkbox(ctx, "admin_users_manage"),
|
||||||
|
AdminSettingsManage: checkbox(ctx, "admin_settings_manage"),
|
||||||
|
AdminBoxesView: checkbox(ctx, "admin_boxes_view"),
|
||||||
|
}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func checkbox(ctx *gin.Context, name string) bool {
|
||||||
|
return ctx.PostForm(name) == "true"
|
||||||
|
}
|
||||||
@@ -2,19 +2,120 @@ package server
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"net/http"
|
"net/http"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
|
"warpbox/lib/metastore"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type adminUserRow struct {
|
||||||
|
ID string
|
||||||
|
Username string
|
||||||
|
Email string
|
||||||
|
Tags string
|
||||||
|
CreatedAt string
|
||||||
|
Disabled bool
|
||||||
|
IsCurrent bool
|
||||||
|
}
|
||||||
|
|
||||||
func (app *App) handleAdminUsers(ctx *gin.Context) {
|
func (app *App) handleAdminUsers(ctx *gin.Context) {
|
||||||
if !app.adminLoginEnabled() {
|
if !app.requireAdminFlag(ctx, func(perms metastore.EffectivePermissions) bool { return perms.AdminUsersManage }) {
|
||||||
ctx.Redirect(http.StatusSeeOther, "/")
|
return
|
||||||
|
}
|
||||||
|
app.renderAdminUsers(ctx, "")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) handleAdminUsersPost(ctx *gin.Context) {
|
||||||
|
if !app.requireAdminFlag(ctx, func(perms metastore.EffectivePermissions) bool { return perms.AdminUsersManage }) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
ctx.HTML(http.StatusOK, "admin/users.html", gin.H{
|
if ctx.PostForm("action") == "toggle_disabled" {
|
||||||
"AdminUsername": app.config.AdminUsername,
|
userID := strings.TrimSpace(ctx.PostForm("user_id"))
|
||||||
"AdminEmail": app.config.AdminEmail,
|
user, ok, err := app.store.GetUser(userID)
|
||||||
"ActivePage": "users",
|
if err != nil || !ok {
|
||||||
|
app.renderAdminUsers(ctx, "User not found.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
if current, ok := ctx.Get("adminUser"); ok {
|
||||||
|
if currentUser, ok := current.(metastore.User); ok && currentUser.ID == user.ID {
|
||||||
|
app.renderAdminUsers(ctx, "You cannot disable the user for the active session.")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
user.Disabled = !user.Disabled
|
||||||
|
if err := app.store.UpdateUser(user); err != nil {
|
||||||
|
app.renderAdminUsers(ctx, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin/users")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
username := ctx.PostForm("username")
|
||||||
|
email := ctx.PostForm("email")
|
||||||
|
password := ctx.PostForm("password")
|
||||||
|
tagIDs := ctx.PostFormArray("tag_ids")
|
||||||
|
if _, err := app.store.CreateUserWithPassword(username, email, password, tagIDs); err != nil {
|
||||||
|
app.renderAdminUsers(ctx, err.Error())
|
||||||
|
return
|
||||||
|
}
|
||||||
|
ctx.Redirect(http.StatusSeeOther, "/admin/users")
|
||||||
|
}
|
||||||
|
|
||||||
|
func (app *App) renderAdminUsers(ctx *gin.Context, errorMessage string) {
|
||||||
|
users, err := app.store.ListUsers()
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not list users")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
tags, err := app.store.ListTags()
|
||||||
|
if err != nil {
|
||||||
|
ctx.String(http.StatusInternalServerError, "Could not list tags")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
tagNames := make(map[string]string, len(tags))
|
||||||
|
for _, tag := range tags {
|
||||||
|
tagNames[tag.ID] = tag.Name
|
||||||
|
}
|
||||||
|
sort.Slice(users, func(i int, j int) bool {
|
||||||
|
return strings.ToLower(users[i].Username) < strings.ToLower(users[j].Username)
|
||||||
|
})
|
||||||
|
|
||||||
|
currentID := ""
|
||||||
|
if current, ok := ctx.Get("adminUser"); ok {
|
||||||
|
if currentUser, ok := current.(metastore.User); ok {
|
||||||
|
currentID = currentUser.ID
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
rows := make([]adminUserRow, 0, len(users))
|
||||||
|
for _, user := range users {
|
||||||
|
names := make([]string, 0, len(user.TagIDs))
|
||||||
|
for _, tagID := range user.TagIDs {
|
||||||
|
if name := tagNames[tagID]; name != "" {
|
||||||
|
names = append(names, name)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
rows = append(rows, adminUserRow{
|
||||||
|
ID: user.ID,
|
||||||
|
Username: user.Username,
|
||||||
|
Email: user.Email,
|
||||||
|
Tags: strings.Join(names, ", "),
|
||||||
|
CreatedAt: formatAdminTime(user.CreatedAt),
|
||||||
|
Disabled: user.Disabled,
|
||||||
|
IsCurrent: user.ID == currentID,
|
||||||
|
})
|
||||||
|
}
|
||||||
|
|
||||||
|
ctx.HTML(http.StatusOK, "admin_users.html", gin.H{
|
||||||
|
"AdminSection": "users",
|
||||||
|
"CurrentUser": app.currentAdminUsername(ctx),
|
||||||
|
"CSRFToken": app.currentCSRFToken(ctx),
|
||||||
|
"Users": rows,
|
||||||
|
"Tags": tags,
|
||||||
|
"Error": errorMessage,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,62 +0,0 @@
|
|||||||
package server
|
|
||||||
|
|
||||||
import (
|
|
||||||
"log"
|
|
||||||
"strings"
|
|
||||||
"time"
|
|
||||||
|
|
||||||
"warpbox/lib/boxstore"
|
|
||||||
)
|
|
||||||
|
|
||||||
func (app *App) runExpiredCleanup(trigger string) (boxstore.CleanupExpiredResult, error) {
|
|
||||||
result, err := boxstore.CleanupExpiredBoxes()
|
|
||||||
if err != nil {
|
|
||||||
log.Printf("warpbox cleanup[%s] failed: %v", trigger, err)
|
|
||||||
app.logActivity("boxes.cleanup.failed", "high", "Expired boxes cleanup failed", nil, map[string]string{
|
|
||||||
"trigger": trigger,
|
|
||||||
"error": err.Error(),
|
|
||||||
})
|
|
||||||
return result, err
|
|
||||||
}
|
|
||||||
|
|
||||||
meta := map[string]string{
|
|
||||||
"trigger": trigger,
|
|
||||||
"scanned": intToString(result.Scanned),
|
|
||||||
"deleted": intToString(result.Deleted),
|
|
||||||
"skipped": intToString(result.Skipped),
|
|
||||||
}
|
|
||||||
if len(result.DeletedIDs) > 0 {
|
|
||||||
limit := len(result.DeletedIDs)
|
|
||||||
if limit > 20 {
|
|
||||||
limit = 20
|
|
||||||
}
|
|
||||||
meta["deleted_ids"] = strings.Join(result.DeletedIDs[:limit], ",")
|
|
||||||
}
|
|
||||||
if len(result.Warnings) > 0 {
|
|
||||||
limit := len(result.Warnings)
|
|
||||||
if limit > 3 {
|
|
||||||
limit = 3
|
|
||||||
}
|
|
||||||
meta["warnings"] = strings.Join(result.Warnings[:limit], " | ")
|
|
||||||
}
|
|
||||||
app.logActivity("boxes.cleanup", "medium", "Expired boxes cleanup run completed", nil, meta)
|
|
||||||
log.Printf("warpbox cleanup[%s] scanned=%d deleted=%d skipped=%d", trigger, result.Scanned, result.Deleted, result.Skipped)
|
|
||||||
return result, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func (app *App) startExpiredCleanupWorker() {
|
|
||||||
if app == nil || app.config == nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
go func() {
|
|
||||||
for {
|
|
||||||
interval := app.config.ExpiredCleanupIntervalSeconds
|
|
||||||
if interval <= 0 {
|
|
||||||
time.Sleep(30 * time.Second)
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
time.Sleep(time.Duration(interval) * time.Second)
|
|
||||||
_, _ = app.runExpiredCleanup("worker")
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
}
|
|
||||||
107
lib/server/ip.go
107
lib/server/ip.go
@@ -1,107 +0,0 @@
|
|||||||
package server
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net"
|
|
||||||
"net/http"
|
|
||||||
"strings"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
|
|
||||||
"warpbox/lib/security"
|
|
||||||
)
|
|
||||||
|
|
||||||
func (app *App) clientIP(ctx *gin.Context) string {
|
|
||||||
if ctx == nil || ctx.Request == nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
remoteIP := remoteAddrIP(ctx.Request)
|
|
||||||
trusted, err := security.ParseCIDRList(app.config.TrustedProxyCIDRs)
|
|
||||||
if err != nil {
|
|
||||||
return remoteIP
|
|
||||||
}
|
|
||||||
if !remoteIsTrusted(remoteIP, trusted) {
|
|
||||||
return remoteIP
|
|
||||||
}
|
|
||||||
for _, candidate := range headerIPs(ctx.Request.Header) {
|
|
||||||
if isPublicIP(candidate) {
|
|
||||||
return candidate
|
|
||||||
}
|
|
||||||
}
|
|
||||||
candidates := headerIPs(ctx.Request.Header)
|
|
||||||
if len(candidates) > 0 {
|
|
||||||
return candidates[0]
|
|
||||||
}
|
|
||||||
return remoteIP
|
|
||||||
}
|
|
||||||
|
|
||||||
func remoteIsTrusted(remoteIP string, trusted []net.IPNet) bool {
|
|
||||||
ip := net.ParseIP(strings.TrimSpace(remoteIP))
|
|
||||||
if ip == nil {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
for _, prefix := range trusted {
|
|
||||||
if prefix.Contains(ip) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|
||||||
func headerIPs(header http.Header) []string {
|
|
||||||
keys := []string{
|
|
||||||
"X-Forwarded-For",
|
|
||||||
"X-Real-Ip",
|
|
||||||
"CF-Connecting-IP",
|
|
||||||
"X-Envoy-External-Address",
|
|
||||||
"Fly-Client-IP",
|
|
||||||
}
|
|
||||||
out := make([]string, 0, 4)
|
|
||||||
seen := map[string]bool{}
|
|
||||||
for _, key := range keys {
|
|
||||||
raw := strings.TrimSpace(header.Get(key))
|
|
||||||
if raw == "" {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
for _, part := range strings.Split(raw, ",") {
|
|
||||||
ip := normalizeIP(strings.TrimSpace(part))
|
|
||||||
if ip == "" || seen[ip] {
|
|
||||||
continue
|
|
||||||
}
|
|
||||||
seen[ip] = true
|
|
||||||
out = append(out, ip)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return out
|
|
||||||
}
|
|
||||||
|
|
||||||
func remoteAddrIP(request *http.Request) string {
|
|
||||||
host, _, err := net.SplitHostPort(strings.TrimSpace(request.RemoteAddr))
|
|
||||||
if err != nil {
|
|
||||||
return normalizeIP(strings.TrimSpace(request.RemoteAddr))
|
|
||||||
}
|
|
||||||
return normalizeIP(host)
|
|
||||||
}
|
|
||||||
|
|
||||||
func normalizeIP(raw string) string {
|
|
||||||
ip := net.ParseIP(strings.TrimSpace(raw))
|
|
||||||
if ip == nil {
|
|
||||||
return ""
|
|
||||||
}
|
|
||||||
return ip.String()
|
|
||||||
}
|
|
||||||
|
|
||||||
func isPublicIP(value string) bool {
|
|
||||||
ip := net.ParseIP(value)
|
|
||||||
if ip == nil || !ip.IsGlobalUnicast() {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
return !isPrivateOrLoopback(value)
|
|
||||||
}
|
|
||||||
|
|
||||||
func isPrivateOrLoopback(value string) bool {
|
|
||||||
ip := net.ParseIP(value)
|
|
||||||
if ip == nil {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
return ip.IsLoopback() || ip.IsPrivate() || ip.IsLinkLocalUnicast() || ip.IsLinkLocalMulticast()
|
|
||||||
}
|
|
||||||
@@ -1,44 +0,0 @@
|
|||||||
package server
|
|
||||||
|
|
||||||
import (
|
|
||||||
"net/http"
|
|
||||||
"net/http/httptest"
|
|
||||||
"testing"
|
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
|
||||||
|
|
||||||
"warpbox/lib/config"
|
|
||||||
)
|
|
||||||
|
|
||||||
func TestClientIPDirectClient(t *testing.T) {
|
|
||||||
app := &App{config: &config.Config{TrustedProxyCIDRs: "10.0.0.0/8"}}
|
|
||||||
ctx, _ := gin.CreateTestContext(httptest.NewRecorder())
|
|
||||||
ctx.Request = httptest.NewRequest(http.MethodGet, "/", nil)
|
|
||||||
ctx.Request.RemoteAddr = "198.51.100.10:1234"
|
|
||||||
ctx.Request.Header.Set("X-Forwarded-For", "203.0.113.4")
|
|
||||||
if got := app.clientIP(ctx); got != "198.51.100.10" {
|
|
||||||
t.Fatalf("expected direct remote IP, got %q", got)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestClientIPTrustedProxyChain(t *testing.T) {
|
|
||||||
app := &App{config: &config.Config{TrustedProxyCIDRs: "10.0.0.0/8"}}
|
|
||||||
ctx, _ := gin.CreateTestContext(httptest.NewRecorder())
|
|
||||||
ctx.Request = httptest.NewRequest(http.MethodGet, "/", nil)
|
|
||||||
ctx.Request.RemoteAddr = "10.1.2.3:8080"
|
|
||||||
ctx.Request.Header.Set("X-Forwarded-For", "203.0.113.44, 10.0.0.5")
|
|
||||||
if got := app.clientIP(ctx); got != "203.0.113.44" {
|
|
||||||
t.Fatalf("expected forwarded public client IP, got %q", got)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func TestClientIPSpoofedHeaderFromUntrustedRemote(t *testing.T) {
|
|
||||||
app := &App{config: &config.Config{TrustedProxyCIDRs: "10.0.0.0/8"}}
|
|
||||||
ctx, _ := gin.CreateTestContext(httptest.NewRecorder())
|
|
||||||
ctx.Request = httptest.NewRequest(http.MethodGet, "/", nil)
|
|
||||||
ctx.Request.RemoteAddr = "203.0.113.200:8080"
|
|
||||||
ctx.Request.Header.Set("X-Forwarded-For", "198.51.100.55")
|
|
||||||
if got := app.clientIP(ctx); got != "203.0.113.200" {
|
|
||||||
t.Fatalf("expected untrusted remote IP, got %q", got)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,12 +1,17 @@
|
|||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"net/http"
|
||||||
|
"net/http/httptest"
|
||||||
"os"
|
"os"
|
||||||
"testing"
|
"testing"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
"warpbox/lib/boxstore"
|
"warpbox/lib/boxstore"
|
||||||
"warpbox/lib/config"
|
"warpbox/lib/config"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
"warpbox/lib/models"
|
"warpbox/lib/models"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -35,3 +40,40 @@ func TestValidateManifestFileUploadRejectsExpiredBox(t *testing.T) {
|
|||||||
t.Fatalf("expected expired box to be deleted, stat err=%v", err)
|
t.Fatalf("expected expired box to be deleted, stat err=%v", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func TestAdminProtectedPostRequiresCSRF(t *testing.T) {
|
||||||
|
gin.SetMode(gin.TestMode)
|
||||||
|
|
||||||
|
store, err := metastore.Open(t.TempDir())
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("Open returned error: %v", err)
|
||||||
|
}
|
||||||
|
defer store.Close()
|
||||||
|
|
||||||
|
adminTag, err := store.EnsureAdminTag()
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("EnsureAdminTag returned error: %v", err)
|
||||||
|
}
|
||||||
|
user, err := store.CreateUserWithPassword("admin", "", "secret", []string{adminTag.ID})
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateUserWithPassword returned error: %v", err)
|
||||||
|
}
|
||||||
|
session, err := store.CreateSession(user.ID, time.Hour)
|
||||||
|
if err != nil {
|
||||||
|
t.Fatalf("CreateSession returned error: %v", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
app := &App{config: &config.Config{}, store: store}
|
||||||
|
router := gin.New()
|
||||||
|
router.POST("/admin/test", app.requireAdminSession, func(ctx *gin.Context) {
|
||||||
|
ctx.Status(http.StatusNoContent)
|
||||||
|
})
|
||||||
|
|
||||||
|
request := httptest.NewRequest(http.MethodPost, "/admin/test", nil)
|
||||||
|
request.AddCookie(&http.Cookie{Name: adminSessionCookie, Value: session.Token})
|
||||||
|
response := httptest.NewRecorder()
|
||||||
|
router.ServeHTTP(response, request)
|
||||||
|
if response.Code != http.StatusForbidden {
|
||||||
|
t.Fatalf("expected missing CSRF token to be forbidden, got %d", response.Code)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|||||||
@@ -1,28 +1,22 @@
|
|||||||
package server
|
package server
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"fmt"
|
||||||
"html/template"
|
|
||||||
"path/filepath"
|
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/gin-contrib/gzip"
|
"github.com/gin-contrib/gzip"
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
|
|
||||||
"warpbox/lib/activity"
|
|
||||||
"warpbox/lib/alerts"
|
|
||||||
"warpbox/lib/boxstore"
|
"warpbox/lib/boxstore"
|
||||||
"warpbox/lib/config"
|
"warpbox/lib/config"
|
||||||
|
"warpbox/lib/metastore"
|
||||||
"warpbox/lib/routing"
|
"warpbox/lib/routing"
|
||||||
"warpbox/lib/security"
|
|
||||||
)
|
)
|
||||||
|
|
||||||
type App struct {
|
type App struct {
|
||||||
config *config.Config
|
config *config.Config
|
||||||
settingsOverridesPath string
|
store *metastore.Store
|
||||||
activityStore *activity.Store
|
adminLoginEnabled bool
|
||||||
alertStore *alerts.Store
|
|
||||||
securityGuard *security.Guard
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func Run(addr string) error {
|
func Run(addr string) error {
|
||||||
@@ -33,39 +27,39 @@ func Run(addr string) error {
|
|||||||
if err := cfg.EnsureDirectories(); err != nil {
|
if err := cfg.EnsureDirectories(); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
overridesPath := filepath.Join(cfg.DBDir, config.AdminSettingsOverrideFilename)
|
|
||||||
overrides, err := config.ReadAdminSettingsOverrides(overridesPath)
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
if err := cfg.ApplyOverrides(overrides); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
applyBoxstoreRuntimeConfig(cfg)
|
applyBoxstoreRuntimeConfig(cfg)
|
||||||
|
|
||||||
|
store, err := metastore.Open(cfg.DBDir)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("open metadata database: %w", err)
|
||||||
|
}
|
||||||
|
defer store.Close()
|
||||||
|
|
||||||
|
overrides, err := store.ListSettings()
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("load settings overrides: %w", err)
|
||||||
|
}
|
||||||
|
if err := cfg.ApplyOverrides(overrides); err != nil {
|
||||||
|
return fmt.Errorf("apply settings overrides: %w", err)
|
||||||
|
}
|
||||||
|
applyBoxstoreRuntimeConfig(cfg)
|
||||||
|
|
||||||
|
bootstrap, err := metastore.BootstrapAdmin(cfg, store)
|
||||||
|
if err != nil {
|
||||||
|
return fmt.Errorf("bootstrap admin metadata: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
app := &App{
|
app := &App{
|
||||||
config: cfg,
|
config: cfg,
|
||||||
settingsOverridesPath: overridesPath,
|
store: store,
|
||||||
activityStore: activity.NewStore(filepath.Join(cfg.DBDir, "activity_log.json")),
|
adminLoginEnabled: bootstrap.AdminLoginEnabled,
|
||||||
alertStore: alerts.NewStore(filepath.Join(cfg.DBDir, "alerts.json")),
|
|
||||||
securityGuard: security.NewGuard(),
|
|
||||||
}
|
|
||||||
if err := app.reloadSecurityConfig(); err != nil {
|
|
||||||
return err
|
|
||||||
}
|
}
|
||||||
|
|
||||||
router := gin.Default()
|
router := gin.Default()
|
||||||
router.Use(app.securityMiddleware())
|
router.LoadHTMLGlob("templates/*.html")
|
||||||
router.NoRoute(app.handleNoRoute)
|
|
||||||
htmlTemplates, err := loadHTMLTemplates()
|
|
||||||
if err != nil {
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
router.SetHTMLTemplate(htmlTemplates)
|
|
||||||
|
|
||||||
routing.Register(router, routing.Handlers{
|
routing.Register(router, routing.Handlers{
|
||||||
Health: app.handleHealth,
|
|
||||||
Index: app.handleIndex,
|
Index: app.handleIndex,
|
||||||
ShowBox: app.handleShowBox,
|
ShowBox: app.handleShowBox,
|
||||||
BoxLogin: handleBoxLogin,
|
BoxLogin: handleBoxLogin,
|
||||||
@@ -79,67 +73,19 @@ func Run(addr string) error {
|
|||||||
FileStatusUpdate: app.handleFileStatusUpdate,
|
FileStatusUpdate: app.handleFileStatusUpdate,
|
||||||
DirectBoxUpload: app.handleDirectBoxUpload,
|
DirectBoxUpload: app.handleDirectBoxUpload,
|
||||||
LegacyUpload: app.handleLegacyUpload,
|
LegacyUpload: app.handleLegacyUpload,
|
||||||
|
|
||||||
AdminLogin: app.handleAdminLogin,
|
|
||||||
AdminLoginPost: app.handleAdminLoginPost,
|
|
||||||
AdminLogout: app.handleAdminLogout,
|
|
||||||
AdminDashboard: app.handleAdminDashboard,
|
|
||||||
AdminAlerts: app.handleAdminAlerts,
|
|
||||||
AdminBoxes: app.handleAdminBoxes,
|
|
||||||
AdminBoxesAction: app.handleAdminBoxesAction,
|
|
||||||
AdminUsers: app.handleAdminUsers,
|
|
||||||
AdminActivity: app.handleAdminActivity,
|
|
||||||
AdminSecurity: app.handleAdminSecurity,
|
|
||||||
AdminAlertsAction: app.handleAdminAlertsAction,
|
|
||||||
AdminSecurityAction: app.handleAdminSecurityAction,
|
|
||||||
AdminSettings: app.handleAdminSettings,
|
|
||||||
AdminSettingsExport: app.handleAdminSettingsExport,
|
|
||||||
AdminSettingsSave: app.handleAdminSettingsSave,
|
|
||||||
AdminSettingsImport: app.handleAdminSettingsImport,
|
|
||||||
AdminSettingsReset: app.handleAdminSettingsReset,
|
|
||||||
AdminAuth: app.adminAuthMiddleware,
|
|
||||||
})
|
})
|
||||||
|
app.registerAccountRoutes(router)
|
||||||
|
app.registerAdminRoutes(router)
|
||||||
|
|
||||||
compressed := router.Group("/", gzip.Gzip(gzip.DefaultCompression))
|
compressed := router.Group("/", gzip.Gzip(gzip.DefaultCompression))
|
||||||
compressed.Static("/static", "./static")
|
compressed.Static("/static", "./static")
|
||||||
|
|
||||||
boxstore.StartThumbnailWorker(cfg.ThumbnailBatchSize, time.Duration(cfg.ThumbnailIntervalSeconds)*time.Second)
|
boxstore.StartThumbnailWorker(cfg.ThumbnailBatchSize, time.Duration(cfg.ThumbnailIntervalSeconds)*time.Second)
|
||||||
app.startExpiredCleanupWorker()
|
|
||||||
|
|
||||||
return router.Run(addr)
|
return router.Run(addr)
|
||||||
}
|
}
|
||||||
|
|
||||||
func loadHTMLTemplates() (*template.Template, error) {
|
|
||||||
tmpl := template.New("").Funcs(template.FuncMap{
|
|
||||||
"toJSON": func(value any) template.JS {
|
|
||||||
data, err := json.Marshal(value)
|
|
||||||
if err != nil {
|
|
||||||
return template.JS("null")
|
|
||||||
}
|
|
||||||
return template.JS(data)
|
|
||||||
},
|
|
||||||
})
|
|
||||||
for _, pattern := range []string{
|
|
||||||
"templates/*.html",
|
|
||||||
"templates/admin/*.html",
|
|
||||||
"templates/admin/partials/*.html",
|
|
||||||
} {
|
|
||||||
var err error
|
|
||||||
tmpl, err = tmpl.ParseGlob(pattern)
|
|
||||||
if err != nil {
|
|
||||||
return nil, err
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return tmpl, nil
|
|
||||||
}
|
|
||||||
|
|
||||||
func applyBoxstoreRuntimeConfig(cfg *config.Config) {
|
func applyBoxstoreRuntimeConfig(cfg *config.Config) {
|
||||||
boxstore.SetUploadRoot(cfg.UploadsDir)
|
boxstore.SetUploadRoot(cfg.UploadsDir)
|
||||||
boxstore.SetOneTimeDownloadExpiry(cfg.OneTimeDownloadExpirySeconds)
|
boxstore.SetOneTimeDownloadExpiry(cfg.OneTimeDownloadExpirySeconds)
|
||||||
}
|
}
|
||||||
|
|
||||||
func (app *App) handleHealth(c *gin.Context) {
|
|
||||||
c.JSON(200, gin.H{
|
|
||||||
"status": "healthy",
|
|
||||||
})
|
|
||||||
}
|
|
||||||
|
|||||||
@@ -39,19 +39,13 @@ func (app *App) handleCreateBox(ctx *gin.Context) {
|
|||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
totalSize := int64(0)
|
|
||||||
for _, file := range request.Files {
|
|
||||||
totalSize += file.Size
|
|
||||||
}
|
|
||||||
if !app.enforceUploadRateLimit(ctx, totalSize) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
files, err := boxstore.CreateManifest(boxID, request)
|
files, err := boxstore.CreateManifest(boxID, request)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
app.indexBoxFromManifest(boxID)
|
||||||
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"box_id": boxID, "box_url": "/box/" + boxID, "files": files})
|
ctx.JSON(http.StatusOK, gin.H{"box_id": boxID, "box_url": "/box/" + boxID, "files": files})
|
||||||
}
|
}
|
||||||
@@ -80,10 +74,6 @@ func (app *App) handleManifestFileUpload(ctx *gin.Context) {
|
|||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if !app.enforceUploadRateLimit(ctx, file.Size) {
|
|
||||||
boxstore.MarkFileStatus(boxID, fileID, models.FileStatusFailed)
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
savedFile, err := boxstore.SaveManifestUpload(boxID, fileID, file)
|
savedFile, err := boxstore.SaveManifestUpload(boxID, fileID, file)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -91,6 +81,7 @@ func (app *App) handleManifestFileUpload(ctx *gin.Context) {
|
|||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
app.indexBoxFromManifest(boxID)
|
||||||
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"box_id": boxID, "box_url": "/box/" + boxID, "file": savedFile})
|
ctx.JSON(http.StatusOK, gin.H{"box_id": boxID, "box_url": "/box/" + boxID, "file": savedFile})
|
||||||
}
|
}
|
||||||
@@ -127,6 +118,7 @@ func (app *App) handleFileStatusUpdate(ctx *gin.Context) {
|
|||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
app.indexBoxFromManifest(boxID)
|
||||||
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"file": file})
|
ctx.JSON(http.StatusOK, gin.H{"file": file})
|
||||||
}
|
}
|
||||||
@@ -152,9 +144,6 @@ func (app *App) handleDirectBoxUpload(ctx *gin.Context) {
|
|||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if !app.enforceUploadRateLimit(ctx, file.Size) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
savedFile, err := boxstore.SaveUpload(boxID, file)
|
savedFile, err := boxstore.SaveUpload(boxID, file)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -194,9 +183,6 @@ func (app *App) handleLegacyUpload(ctx *gin.Context) {
|
|||||||
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
ctx.JSON(http.StatusBadRequest, gin.H{"error": err.Error()})
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if !app.enforceUploadRateLimit(ctx, totalSize) {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
boxID, err := boxstore.NewBoxID()
|
boxID, err := boxstore.NewBoxID()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -248,6 +234,7 @@ func (app *App) handleLegacyUpload(ctx *gin.Context) {
|
|||||||
|
|
||||||
savedFiles = append(savedFiles, savedFile)
|
savedFiles = append(savedFiles, savedFile)
|
||||||
}
|
}
|
||||||
|
app.indexBoxFromManifest(boxID)
|
||||||
|
|
||||||
ctx.JSON(http.StatusOK, gin.H{"box_id": boxID, "box_url": "/box/" + boxID, "files": savedFiles})
|
ctx.JSON(http.StatusOK, gin.H{"box_id": boxID, "box_url": "/box/" + boxID, "files": savedFiles})
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -3,7 +3,6 @@ package server
|
|||||||
import (
|
import (
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/http"
|
"net/http"
|
||||||
"strconv"
|
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
"github.com/gin-gonic/gin"
|
"github.com/gin-gonic/gin"
|
||||||
@@ -154,39 +153,3 @@ func (app *App) maxRequestBodyBytes() int64 {
|
|||||||
}
|
}
|
||||||
return limit + 10*1024*1024
|
return limit + 10*1024*1024
|
||||||
}
|
}
|
||||||
|
|
||||||
func (app *App) enforceUploadRateLimit(ctx *gin.Context, size int64) bool {
|
|
||||||
if !app.securityFeaturesEnabled() || app.securityGuard == nil {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
ip := app.clientIP(ctx)
|
|
||||||
if app.securityGuard.IsWhitelisted(ip) || app.securityGuard.IsAdminWhitelisted(ip) {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
allowed, requestCount, totalBytes := app.securityGuard.AllowUpload(
|
|
||||||
ip,
|
|
||||||
size,
|
|
||||||
app.config.SecurityUploadWindowSeconds,
|
|
||||||
app.config.SecurityUploadMaxRequests,
|
|
||||||
app.config.SecurityUploadMaxBytes,
|
|
||||||
)
|
|
||||||
if allowed {
|
|
||||||
return true
|
|
||||||
}
|
|
||||||
|
|
||||||
app.logActivity("security.upload_limit", "high", "Upload rate limit exceeded", ctx, map[string]string{
|
|
||||||
"requests": strconv.Itoa(requestCount),
|
|
||||||
"bytes": strconv.FormatInt(totalBytes, 10),
|
|
||||||
})
|
|
||||||
app.createAlert(
|
|
||||||
"Upload rate limit triggered",
|
|
||||||
"medium",
|
|
||||||
"security",
|
|
||||||
"430",
|
|
||||||
"security.upload.rate_limit",
|
|
||||||
"Per-IP upload rate limit blocked request.",
|
|
||||||
map[string]string{"ip": ip, "requests": strconv.Itoa(requestCount)},
|
|
||||||
)
|
|
||||||
ctx.JSON(http.StatusTooManyRequests, gin.H{"error": "Too many uploads from this IP. Try again later."})
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
|
|||||||
19
run.sh
19
run.sh
@@ -15,9 +15,9 @@ export WARPBOX_ONE_TIME_DOWNLOAD_EXPIRY_SECONDS="${WARPBOX_ONE_TIME_DOWNLOAD_EXP
|
|||||||
export WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE="${WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE:-false}"
|
export WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE="${WARPBOX_ONE_TIME_DOWNLOAD_RETRY_ON_FAILURE:-false}"
|
||||||
|
|
||||||
# Storage and expiry limits used by the upload UI and backend validators.
|
# Storage and expiry limits used by the upload UI and backend validators.
|
||||||
# Use decimal gigabytes here. Examples: 2, 4, 0.5
|
# Use megabytes here; WarpBox converts these to bytes internally.
|
||||||
export WARPBOX_GLOBAL_MAX_FILE_SIZE_GB="${WARPBOX_GLOBAL_MAX_FILE_SIZE_GB:-2}" # 2 GB
|
export WARPBOX_GLOBAL_MAX_FILE_SIZE_MB="${WARPBOX_GLOBAL_MAX_FILE_SIZE_MB:-2048}" # 2 GiB
|
||||||
export WARPBOX_GLOBAL_MAX_BOX_SIZE_GB="${WARPBOX_GLOBAL_MAX_BOX_SIZE_GB:-4}" # 4 GB
|
export WARPBOX_GLOBAL_MAX_BOX_SIZE_MB="${WARPBOX_GLOBAL_MAX_BOX_SIZE_MB:-4096}" # 4 GiB
|
||||||
export WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS="${WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS:-3600}" # 1 hour
|
export WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS="${WARPBOX_DEFAULT_GUEST_EXPIRY_SECONDS:-3600}" # 1 hour
|
||||||
export WARPBOX_MAX_GUEST_EXPIRY_SECONDS="${WARPBOX_MAX_GUEST_EXPIRY_SECONDS:-172800}" # 48 hours
|
export WARPBOX_MAX_GUEST_EXPIRY_SECONDS="${WARPBOX_MAX_GUEST_EXPIRY_SECONDS:-172800}" # 48 hours
|
||||||
|
|
||||||
@@ -25,19 +25,6 @@ export WARPBOX_MAX_GUEST_EXPIRY_SECONDS="${WARPBOX_MAX_GUEST_EXPIRY_SECONDS:-172
|
|||||||
export WARPBOX_BOX_POLL_INTERVAL_MS="${WARPBOX_BOX_POLL_INTERVAL_MS:-5000}"
|
export WARPBOX_BOX_POLL_INTERVAL_MS="${WARPBOX_BOX_POLL_INTERVAL_MS:-5000}"
|
||||||
export WARPBOX_THUMBNAIL_BATCH_SIZE="${WARPBOX_THUMBNAIL_BATCH_SIZE:-10}"
|
export WARPBOX_THUMBNAIL_BATCH_SIZE="${WARPBOX_THUMBNAIL_BATCH_SIZE:-10}"
|
||||||
export WARPBOX_THUMBNAIL_INTERVAL_SECONDS="${WARPBOX_THUMBNAIL_INTERVAL_SECONDS:-30}"
|
export WARPBOX_THUMBNAIL_INTERVAL_SECONDS="${WARPBOX_THUMBNAIL_INTERVAL_SECONDS:-30}"
|
||||||
export WARPBOX_ACTIVITY_RETENTION_SECONDS="${WARPBOX_ACTIVITY_RETENTION_SECONDS:-604800}"
|
|
||||||
export WARPBOX_SECURITY_ENABLED="${WARPBOX_SECURITY_ENABLED:-true}"
|
|
||||||
export WARPBOX_SECURITY_IP_WHITELIST="${WARPBOX_SECURITY_IP_WHITELIST:-}"
|
|
||||||
export WARPBOX_SECURITY_ADMIN_IP_WHITELIST="${WARPBOX_SECURITY_ADMIN_IP_WHITELIST:-}"
|
|
||||||
export WARPBOX_SECURITY_LOGIN_WINDOW_SECONDS="${WARPBOX_SECURITY_LOGIN_WINDOW_SECONDS:-600}"
|
|
||||||
export WARPBOX_SECURITY_LOGIN_MAX_ATTEMPTS="${WARPBOX_SECURITY_LOGIN_MAX_ATTEMPTS:-8}"
|
|
||||||
export WARPBOX_SECURITY_BAN_SECONDS="${WARPBOX_SECURITY_BAN_SECONDS:-1800}"
|
|
||||||
export WARPBOX_SECURITY_SCAN_WINDOW_SECONDS="${WARPBOX_SECURITY_SCAN_WINDOW_SECONDS:-300}"
|
|
||||||
export WARPBOX_SECURITY_SCAN_MAX_ATTEMPTS="${WARPBOX_SECURITY_SCAN_MAX_ATTEMPTS:-12}"
|
|
||||||
export WARPBOX_SECURITY_UPLOAD_WINDOW_SECONDS="${WARPBOX_SECURITY_UPLOAD_WINDOW_SECONDS:-60}"
|
|
||||||
export WARPBOX_SECURITY_UPLOAD_MAX_REQUESTS="${WARPBOX_SECURITY_UPLOAD_MAX_REQUESTS:-20}"
|
|
||||||
export WARPBOX_SECURITY_UPLOAD_MAX_GB="${WARPBOX_SECURITY_UPLOAD_MAX_GB:-10}"
|
|
||||||
export WARPBOX_EXPIRED_CLEANUP_INTERVAL_SECONDS="${WARPBOX_EXPIRED_CLEANUP_INTERVAL_SECONDS:-300}"
|
|
||||||
|
|
||||||
# Data location.
|
# Data location.
|
||||||
export WARPBOX_DATA_DIR="${WARPBOX_DATA_DIR:-./data}"
|
export WARPBOX_DATA_DIR="${WARPBOX_DATA_DIR:-./data}"
|
||||||
|
|||||||
1507
static/css/account.css
Normal file
1507
static/css/account.css
Normal file
File diff suppressed because it is too large
Load Diff
@@ -1,63 +0,0 @@
|
|||||||
.activity-page-body { display: grid; gap: 10px; }
|
|
||||||
.activity-panel {
|
|
||||||
min-height: 0;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
padding: 10px;
|
|
||||||
}
|
|
||||||
.activity-toolbar-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(220px, 1.2fr) minmax(130px, .4fr) minmax(150px, .5fr);
|
|
||||||
gap: 8px;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
.activity-input,
|
|
||||||
.activity-select {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 0;
|
|
||||||
height: 28px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
padding: 4px 6px;
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
.activity-table-wrap {
|
|
||||||
min-height: 420px;
|
|
||||||
height: 520px;
|
|
||||||
overflow: auto;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 2px solid #606060;
|
|
||||||
border-left: 2px solid #606060;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
}
|
|
||||||
.activity-table {
|
|
||||||
width: 100%;
|
|
||||||
border-collapse: collapse;
|
|
||||||
table-layout: fixed;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 14px;
|
|
||||||
}
|
|
||||||
.activity-table th,
|
|
||||||
.activity-table td {
|
|
||||||
padding: 6px;
|
|
||||||
border-bottom: 1px solid #e1e1e1;
|
|
||||||
text-align: left;
|
|
||||||
white-space: nowrap;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
}
|
|
||||||
.activity-table th {
|
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
background: #dfdfdf;
|
|
||||||
z-index: 2;
|
|
||||||
}
|
|
||||||
@@ -1,738 +1,132 @@
|
|||||||
/* ===========================
|
body {
|
||||||
Admin Shell / Frame
|
|
||||||
=========================== */
|
|
||||||
.admin-shell {
|
|
||||||
width: 100%;
|
|
||||||
min-height: 100vh;
|
min-height: 100vh;
|
||||||
|
}
|
||||||
|
|
||||||
|
.admin-window {
|
||||||
|
width: min(1120px, calc(100vw - 32px));
|
||||||
|
margin: 32px auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.admin-panel {
|
||||||
|
display: grid;
|
||||||
|
gap: 16px;
|
||||||
|
padding: 16px;
|
||||||
|
background-color: #ffffff;
|
||||||
|
background-image:
|
||||||
|
linear-gradient(180deg, rgba(255,255,255,.9), rgba(238,238,238,.58)),
|
||||||
|
repeating-linear-gradient(0deg, rgba(0,0,0,.025) 0 1px, transparent 1px 6px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.admin-nav {
|
||||||
display: flex;
|
display: flex;
|
||||||
justify-content: flex-start;
|
flex-wrap: wrap;
|
||||||
align-items: center;
|
|
||||||
flex-direction: column;
|
|
||||||
padding: 10px 16px 34px;
|
|
||||||
gap: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-frame {
|
|
||||||
width: min(var(--admin-frame-width, 1320px), 100%);
|
|
||||||
display: grid;
|
|
||||||
grid-template-rows: auto auto;
|
|
||||||
gap: 10px;
|
|
||||||
align-items: start;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Admin Taskbar (top nav)
|
|
||||||
=========================== */
|
|
||||||
.admin-taskbar {
|
|
||||||
width: 100%;
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: auto minmax(0, 1fr) auto;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
color: #000000;
|
|
||||||
background-color: var(--w98-gray);
|
|
||||||
background-image: linear-gradient(180deg, rgba(255,255,255,.36), rgba(0,0,0,.08)), repeating-linear-gradient(45deg, rgba(255,255,255,.12) 0 1px, transparent 1px 5px);
|
|
||||||
border-top: 2px solid #ffffff;
|
|
||||||
border-left: 2px solid #ffffff;
|
|
||||||
border-right: 2px solid #000000;
|
|
||||||
border-bottom: 2px solid #000000;
|
|
||||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf, 4px 4px 0 rgba(0,0,0,.45);
|
|
||||||
padding: 3px;
|
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
z-index: 50;
|
|
||||||
transition: box-shadow 120ms steps(2, end), filter 120ms steps(2, end);
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar.is-scrolled {
|
|
||||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf, 0 5px 0 rgba(0,0,0,.55), 0 11px 0 rgba(0,0,0,.18);
|
|
||||||
filter: brightness(1.02);
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar.is-scrolled::after {
|
|
||||||
content: "";
|
|
||||||
position: absolute;
|
|
||||||
left: 0;
|
|
||||||
right: 0;
|
|
||||||
bottom: -10px;
|
|
||||||
height: 10px;
|
|
||||||
pointer-events: none;
|
|
||||||
background: linear-gradient(to bottom, rgba(0,0,0,.46), rgba(0,0,0,0));
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Start Button
|
|
||||||
=========================== */
|
|
||||||
.admin-start-button {
|
|
||||||
min-width: 108px;
|
|
||||||
height: 24px;
|
|
||||||
display: inline-grid;
|
|
||||||
grid-template-columns: 18px 1fr;
|
|
||||||
align-items: center;
|
align-items: center;
|
||||||
gap: 5px;
|
|
||||||
padding: 0 8px;
|
|
||||||
color: #000000;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
border-top: 2px solid #ffffff;
|
|
||||||
border-left: 2px solid #ffffff;
|
|
||||||
border-right: 2px solid #000000;
|
|
||||||
border-bottom: 2px solid #000000;
|
|
||||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf;
|
|
||||||
font-weight: bold;
|
|
||||||
text-decoration: none;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-start-button:active {
|
.admin-spacer {
|
||||||
border-top-color: #000000;
|
flex: 1;
|
||||||
border-left-color: #000000;
|
|
||||||
border-right-color: #ffffff;
|
|
||||||
border-bottom-color: #ffffff;
|
|
||||||
box-shadow: inset -1px -1px 0 #dfdfdf, inset 1px 1px 0 #808080;
|
|
||||||
padding-top: 1px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-start-logo {
|
.admin-grid {
|
||||||
width: 16px;
|
|
||||||
height: 16px;
|
|
||||||
display: grid;
|
display: grid;
|
||||||
place-items: center;
|
grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
|
||||||
color: #ffffff;
|
gap: 12px;
|
||||||
background: #000078;
|
|
||||||
border: 1px solid #ffffff;
|
|
||||||
box-shadow: inset -5px 0 0 #0f80cd, inset 0 -5px 0 #4c1ca0;
|
|
||||||
font-size: 10px;
|
|
||||||
line-height: 10px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/* ===========================
|
.admin-link {
|
||||||
Taskbar Nav Buttons
|
min-height: 88px;
|
||||||
=========================== */
|
padding: 12px;
|
||||||
.admin-taskbar-nav {
|
color: inherit;
|
||||||
min-width: 0;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 4px;
|
|
||||||
overflow-x: auto;
|
|
||||||
scrollbar-width: thin;
|
|
||||||
padding-bottom: 1px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar-button {
|
|
||||||
height: 24px;
|
|
||||||
min-width: 76px;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
gap: 5px;
|
|
||||||
padding: 0 8px;
|
|
||||||
color: #000000;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
text-decoration: none;
|
text-decoration: none;
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar-button:active {
|
|
||||||
border-top-color: #000000;
|
|
||||||
border-left-color: #000000;
|
|
||||||
border-right-color: #ffffff;
|
|
||||||
border-bottom-color: #ffffff;
|
|
||||||
box-shadow: inset -1px -1px 0 #dfdfdf, inset 1px 1px 0 #808080;
|
|
||||||
padding-top: 1px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar-button.is-active {
|
|
||||||
color: #ffffff;
|
|
||||||
background: #000078;
|
|
||||||
border-top-color: #000000;
|
|
||||||
border-left-color: #000000;
|
|
||||||
border-right-color: #ffffff;
|
|
||||||
border-bottom-color: #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar-button:hover {
|
|
||||||
color: #ffffff;
|
|
||||||
background: #000078;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Taskbar Session Chips
|
|
||||||
=========================== */
|
|
||||||
.admin-taskbar-session {
|
|
||||||
min-width: 0;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: flex-end;
|
|
||||||
gap: 5px;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-session-chip,
|
|
||||||
.admin-alert-chip {
|
|
||||||
height: 24px;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 5px;
|
|
||||||
padding: 0 8px;
|
|
||||||
background: #dfdfdf;
|
background: #dfdfdf;
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
color: #000000;
|
|
||||||
text-decoration: none;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-alert-chip.is-ok { background: #e8ffe8; border-color: #008000 #ffffff #ffffff #008000; }
|
|
||||||
.admin-alert-chip.is-info { background: #d8e5f8; }
|
|
||||||
.admin-alert-chip.is-warning {
|
|
||||||
background: #ffffcc;
|
|
||||||
border: 3px solid transparent;
|
|
||||||
border-image: repeating-linear-gradient(45deg, #111111 0 8px, #ffcc00 8px 16px) 3;
|
|
||||||
}
|
|
||||||
.admin-alert-chip.is-danger {
|
|
||||||
color: #ffffff;
|
|
||||||
background: #800000;
|
|
||||||
border: 3px solid transparent;
|
|
||||||
border-image: repeating-linear-gradient(45deg, #ffcccc 0 8px, #300000 8px 16px) 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Dashboard Window
|
|
||||||
=========================== */
|
|
||||||
.admin-dashboard-window,
|
|
||||||
.admin-workspace-window {
|
|
||||||
width: 100%;
|
|
||||||
min-height: 0;
|
|
||||||
padding: 0;
|
|
||||||
overflow: visible;
|
|
||||||
color: #000000;
|
|
||||||
background-color: var(--w98-gray);
|
|
||||||
background-image: linear-gradient(180deg, rgba(255,255,255,.24), rgba(0,0,0,.06));
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-dashboard-window > .win98-titlebar,
|
|
||||||
.admin-workspace-window > .win98-titlebar {
|
|
||||||
margin: 2px 2px 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-dashboard-window > .menu-bar,
|
|
||||||
.admin-workspace-window > .menu-bar {
|
|
||||||
flex: 0 0 auto;
|
|
||||||
height: auto;
|
|
||||||
min-height: 24px;
|
|
||||||
margin: 0 2px;
|
|
||||||
padding: 1px 6px;
|
|
||||||
color: #000000;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
border-top: 1px solid #ffffff;
|
border-top: 1px solid #ffffff;
|
||||||
border-left: 1px solid #ffffff;
|
border-left: 1px solid #ffffff;
|
||||||
border-right: 1px solid #808080;
|
border-right: 1px solid #808080;
|
||||||
border-bottom: 1px solid #808080;
|
border-bottom: 1px solid #808080;
|
||||||
z-index: 30;
|
box-shadow: inset 1px 1px 0 #f7f7f7, inset -1px -1px 0 #b0b0b0;
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-dashboard-window > .menu-bar .menu-button,
|
.admin-link strong,
|
||||||
.admin-workspace-window > .menu-bar .menu-button {
|
.admin-link span {
|
||||||
color: #000000;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-dashboard-window > .dashboard-body,
|
|
||||||
.admin-workspace-window > .admin-workspace-body {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
margin-top: 0;
|
|
||||||
padding: 0 10px 10px;
|
|
||||||
background-color: var(--w98-gray);
|
|
||||||
background-image: linear-gradient(180deg, rgba(255,255,255,.18), rgba(0,0,0,.05));
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-dashboard-statusbar {
|
|
||||||
grid-template-columns: minmax(0, 1fr) 160px 210px;
|
|
||||||
height: 28px;
|
|
||||||
padding: 3px 4px 4px;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-dashboard-statusbar span {
|
|
||||||
min-height: 19px;
|
|
||||||
align-items: center;
|
|
||||||
padding: 1px 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Menu Bar (toolbar)
|
|
||||||
=========================== */
|
|
||||||
.admin-menu-bar {
|
|
||||||
position: relative;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 2px;
|
|
||||||
min-height: 24px;
|
|
||||||
padding: 1px 6px;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
z-index: 20;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-item {
|
|
||||||
position: relative;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-button {
|
|
||||||
height: 20px;
|
|
||||||
min-width: 54px;
|
|
||||||
padding: 0 8px;
|
|
||||||
color: #000000;
|
|
||||||
background: transparent;
|
|
||||||
border: 1px solid transparent;
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: 13px;
|
|
||||||
text-align: left;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-button:hover,
|
|
||||||
.admin-menu-button:focus-visible {
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
outline: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-popup {
|
|
||||||
position: absolute;
|
|
||||||
top: 22px;
|
|
||||||
left: 0;
|
|
||||||
min-width: 220px;
|
|
||||||
padding: 2px;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
border-top: 2px solid #ffffff;
|
|
||||||
border-left: 2px solid #ffffff;
|
|
||||||
border-right: 2px solid #000000;
|
|
||||||
border-bottom: 2px solid #000000;
|
|
||||||
box-shadow: 3px 3px 0 rgba(0,0,0,.35);
|
|
||||||
display: none;
|
|
||||||
z-index: 60;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-item.is-open .admin-menu-popup {
|
|
||||||
display: block;
|
display: block;
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-menu-action {
|
.admin-link span {
|
||||||
|
margin-top: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.admin-table {
|
||||||
width: 100%;
|
width: 100%;
|
||||||
min-height: 22px;
|
border-collapse: collapse;
|
||||||
display: grid;
|
background: #fff;
|
||||||
grid-template-columns: 20px minmax(0, 1fr) auto;
|
border-top: 2px solid #808080;
|
||||||
gap: 8px;
|
border-left: 2px solid #808080;
|
||||||
align-items: center;
|
border-right: 2px solid #ffffff;
|
||||||
padding: 2px 6px;
|
border-bottom: 2px solid #ffffff;
|
||||||
color: #000000;
|
}
|
||||||
background: transparent;
|
|
||||||
border: 0;
|
.admin-table th,
|
||||||
font-family: inherit;
|
.admin-table td {
|
||||||
font-size: 12px;
|
padding: 8px;
|
||||||
|
border: 1px solid #808080;
|
||||||
text-align: left;
|
text-align: left;
|
||||||
|
vertical-align: top;
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-menu-action:hover,
|
.admin-form {
|
||||||
.admin-menu-action:focus-visible {
|
|
||||||
color: #ffffff;
|
|
||||||
background: #000078;
|
|
||||||
outline: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-separator {
|
|
||||||
height: 1px;
|
|
||||||
margin: 3px 2px;
|
|
||||||
background: #808080;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-action .shortcut {
|
|
||||||
color: #555555;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-action:hover .shortcut {
|
|
||||||
color: #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Hero Section
|
|
||||||
=========================== */
|
|
||||||
.admin-hero {
|
|
||||||
display: grid;
|
display: grid;
|
||||||
grid-template-columns: minmax(0, 1fr) 330px;
|
|
||||||
gap: 10px;
|
gap: 10px;
|
||||||
padding: 9px;
|
|
||||||
align-items: stretch;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-hero-copy h2 {
|
.admin-form-row {
|
||||||
margin: 0 0 5px;
|
|
||||||
font-size: 22px;
|
|
||||||
line-height: 24px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-hero-copy p {
|
|
||||||
margin: 0;
|
|
||||||
color: #333333;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-hero-status {
|
|
||||||
display: grid;
|
display: grid;
|
||||||
gap: 4px;
|
gap: 4px;
|
||||||
align-content: center;
|
}
|
||||||
padding: 7px;
|
|
||||||
|
.admin-form-row input,
|
||||||
|
.admin-form-row textarea,
|
||||||
|
.admin-form-row select {
|
||||||
|
width: 100%;
|
||||||
|
min-height: 24px;
|
||||||
|
color: #000000;
|
||||||
background: #ffffff;
|
background: #ffffff;
|
||||||
border-top: 1px solid #808080;
|
border-top: 1px solid #808080;
|
||||||
border-left: 1px solid #808080;
|
border-left: 1px solid #808080;
|
||||||
border-right: 1px solid #ffffff;
|
border-right: 1px solid #ffffff;
|
||||||
border-bottom: 1px solid #ffffff;
|
border-bottom: 1px solid #ffffff;
|
||||||
font-size: 12px;
|
font-family: inherit;
|
||||||
line-height: 13px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-hero-status-row {
|
.admin-checks {
|
||||||
display: flex;
|
display: grid;
|
||||||
justify-content: space-between;
|
grid-template-columns: repeat(auto-fit, minmax(180px, 1fr));
|
||||||
gap: 8px;
|
gap: 8px;
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-status-ok { color: #008000; }
|
.admin-checks label {
|
||||||
.admin-status-warn { color: #8a6200; }
|
|
||||||
.admin-status-danger { color: #800000; }
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Stats Grid
|
|
||||||
=========================== */
|
|
||||||
.admin-stats-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
|
||||||
gap: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-stat-card {
|
|
||||||
position: relative;
|
|
||||||
min-height: 122px;
|
|
||||||
padding: 10px 11px 10px 14px;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Left accent bar */
|
|
||||||
.admin-stat-card::before {
|
|
||||||
content: "";
|
|
||||||
position: absolute;
|
|
||||||
inset: 0 auto 0 0;
|
|
||||||
width: 7px;
|
|
||||||
border-left: 7px solid #000078;
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Severity color states */
|
|
||||||
.admin-stat-card.is-ok { background: linear-gradient(180deg, #eeffee, #ffffff); }
|
|
||||||
.admin-stat-card.is-ok::before { border-left-color: #008000; }
|
|
||||||
.admin-stat-card.is-info { background: linear-gradient(180deg, #edf4ff, #ffffff); }
|
|
||||||
.admin-stat-card.is-info::before { border-left-color: #000078; }
|
|
||||||
.admin-stat-card.is-warning { background: linear-gradient(180deg, #ffffcc, #ffffff); }
|
|
||||||
.admin-stat-card.is-warning::before { border-left-color: #ffcc00; }
|
|
||||||
.admin-stat-card.is-danger {
|
|
||||||
color: #000000;
|
|
||||||
background: repeating-linear-gradient(45deg, #fff2f2 0 6px, #ffe1e1 6px 12px);
|
|
||||||
}
|
|
||||||
.admin-stat-card.is-danger::before { border-left-color: #800000; }
|
|
||||||
|
|
||||||
.admin-stat-label {
|
|
||||||
margin: 0 0 6px;
|
|
||||||
color: #333333;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-stat-value {
|
|
||||||
margin: 0 0 7px;
|
|
||||||
font-size: 32px;
|
|
||||||
line-height: 32px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-stat-note {
|
|
||||||
display: flex;
|
display: flex;
|
||||||
gap: 4px;
|
gap: 6px;
|
||||||
flex-wrap: wrap;
|
align-items: center;
|
||||||
margin: 0;
|
|
||||||
color: #222222;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 14px;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
.admin-stat-note-pill {
|
.admin-error {
|
||||||
display: inline-flex;
|
padding: 8px;
|
||||||
align-items: center;
|
border: 1px solid #800;
|
||||||
min-height: 18px;
|
background: #ffdede;
|
||||||
padding: 1px 6px;
|
}
|
||||||
|
|
||||||
|
.admin-summary {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.admin-summary span {
|
||||||
|
padding: 6px 8px;
|
||||||
background: #dfdfdf;
|
background: #dfdfdf;
|
||||||
border-top: 1px solid #ffffff;
|
border-top: 1px solid #ffffff;
|
||||||
border-left: 1px solid #ffffff;
|
border-left: 1px solid #ffffff;
|
||||||
border-right: 1px solid #808080;
|
border-right: 1px solid #808080;
|
||||||
border-bottom: 1px solid #808080;
|
border-bottom: 1px solid #808080;
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Main Grid / Section Windows
|
|
||||||
=========================== */
|
|
||||||
.admin-main-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
|
|
||||||
gap: 12px;
|
|
||||||
align-items: start;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-span-2 {
|
|
||||||
grid-column: 1 / -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-section-window {
|
|
||||||
min-height: 0;
|
|
||||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf, 3px 4px 0 rgba(0,0,0,.38);
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-section-body {
|
|
||||||
margin: 0 6px 6px;
|
|
||||||
padding: 8px;
|
|
||||||
min-height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Quick Actions
|
|
||||||
=========================== */
|
|
||||||
.admin-link-list {
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
list-style: none;
|
|
||||||
display: grid;
|
|
||||||
gap: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-link-list li {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: auto minmax(0, 1fr);
|
|
||||||
gap: 8px;
|
|
||||||
align-items: center;
|
|
||||||
color: #000000;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-link-button {
|
|
||||||
min-width: 112px;
|
|
||||||
height: 24px;
|
|
||||||
display: inline-grid;
|
|
||||||
place-items: center;
|
|
||||||
padding: 0 10px;
|
|
||||||
color: #000000;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #000000;
|
|
||||||
border-bottom: 1px solid #000000;
|
|
||||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-link-button:hover {
|
|
||||||
filter: brightness(1.06);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Titlebar action links (Show all) */
|
|
||||||
.titlebar-actions {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 2px;
|
|
||||||
margin-left: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.titlebar-link-button {
|
|
||||||
height: 18px;
|
|
||||||
min-width: 64px;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
padding: 0 7px;
|
|
||||||
color: #000000;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #000000;
|
|
||||||
border-bottom: 1px solid #000000;
|
|
||||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf;
|
|
||||||
text-decoration: none;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.titlebar-link-button:hover {
|
|
||||||
filter: brightness(1.08);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Compact Mode
|
|
||||||
=========================== */
|
|
||||||
body.is-compact .admin-dashboard-body {
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
body.is-compact .admin-section-body {
|
|
||||||
padding: 5px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Responsive: Medium (tablets)
|
|
||||||
=========================== */
|
|
||||||
@media (max-width: 1180px) {
|
|
||||||
.admin-taskbar {
|
|
||||||
grid-template-columns: auto minmax(0, 1fr);
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar-session {
|
|
||||||
grid-column: 1 / -1;
|
|
||||||
justify-content: flex-start;
|
|
||||||
overflow-x: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-stats-grid {
|
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-hero {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-main-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-span-2 {
|
|
||||||
grid-column: auto;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* ===========================
|
|
||||||
Responsive: Small (mobile)
|
|
||||||
=========================== */
|
|
||||||
@media (max-width: 760px) {
|
|
||||||
.admin-shell {
|
|
||||||
padding: 0 0 18px;
|
|
||||||
align-items: stretch;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-frame {
|
|
||||||
width: 100%;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
border-left: 0;
|
|
||||||
border-right: 0;
|
|
||||||
box-shadow: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-start-button {
|
|
||||||
width: 100%;
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar-nav {
|
|
||||||
width: 100%;
|
|
||||||
overflow-x: auto;
|
|
||||||
padding-bottom: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar-button {
|
|
||||||
min-width: 92px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-taskbar-session {
|
|
||||||
width: 100%;
|
|
||||||
overflow-x: auto;
|
|
||||||
padding-bottom: 3px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-session-chip,
|
|
||||||
.admin-alert-chip {
|
|
||||||
flex: 0 0 auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-dashboard-window,
|
|
||||||
.admin-workspace-window {
|
|
||||||
min-height: 100dvh;
|
|
||||||
border-left: 0;
|
|
||||||
border-right: 0;
|
|
||||||
box-shadow: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-dashboard-body {
|
|
||||||
padding: 6px;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-stats-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-stat-card {
|
|
||||||
min-height: 112px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-menu-popup {
|
|
||||||
position: fixed;
|
|
||||||
left: 6px;
|
|
||||||
right: 6px;
|
|
||||||
top: 74px;
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.titlebar-actions {
|
|
||||||
margin-left: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.titlebar-link-button {
|
|
||||||
min-width: 58px;
|
|
||||||
padding: 0 5px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.admin-dashboard-statusbar {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
height: auto;
|
|
||||||
min-height: 70px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.win98-titlebar h1,
|
|
||||||
.win98-titlebar h2 {
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.win98-window-controls {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Override global main layout on admin pages since admin uses its own shell */
|
|
||||||
body:has(.admin-shell) main {
|
|
||||||
display: contents;
|
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,394 +0,0 @@
|
|||||||
.alerts-page-body {
|
|
||||||
display: grid;
|
|
||||||
gap: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-summary-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-stat-card {
|
|
||||||
min-width: 0;
|
|
||||||
padding: 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
box-shadow: inset 1px 1px 0 #f7f7f7, inset -1px -1px 0 #b0b0b0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-stat-card.is-danger { background: linear-gradient(180deg, #ffd8d8, #f1b3b3); }
|
|
||||||
.alerts-stat-card.is-warning { background: linear-gradient(180deg, #fff1c9, #ffe39f); }
|
|
||||||
.alerts-stat-card.is-info { background: linear-gradient(180deg, #d7e6fb, #bfd7f8); }
|
|
||||||
|
|
||||||
.alerts-stat-label {
|
|
||||||
margin: 0 0 4px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
text-transform: uppercase;
|
|
||||||
color: #333333;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-stat-value {
|
|
||||||
margin: 0;
|
|
||||||
font-size: 24px;
|
|
||||||
line-height: 24px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-stat-note {
|
|
||||||
margin: 6px 0 0;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 0 6px;
|
|
||||||
color: #222222;
|
|
||||||
background: rgba(255,255,255,.65);
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #a0a0a0;
|
|
||||||
border-bottom: 1px solid #a0a0a0;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-content-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(0, 1.3fr) minmax(320px, .7fr);
|
|
||||||
gap: 10px;
|
|
||||||
min-height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-column {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 10px;
|
|
||||||
min-height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-list-panel {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 520px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-actions-panel {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 220px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-panel {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
min-height: 0;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
box-shadow: inset 1px 1px 0 rgba(255,255,255,.7), inset -1px -1px 0 rgba(0,0,0,.08);
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-panel-header {
|
|
||||||
flex: 0 0 auto;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 8px;
|
|
||||||
min-height: 34px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
box-shadow: inset 1px 1px 0 #f7f7f7;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-panel-title {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
min-width: 0;
|
|
||||||
min-height: 22px;
|
|
||||||
font-weight: bold;
|
|
||||||
font-size: 15px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-panel-sub {
|
|
||||||
color: #444444;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
font-weight: normal;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-panel-tools {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-panel-body {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 0;
|
|
||||||
padding: 10px;
|
|
||||||
overflow: auto;
|
|
||||||
background-color: #ffffff;
|
|
||||||
background-image: linear-gradient(180deg, rgba(255,255,255,.9), rgba(238,238,238,.58));
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-tool-button,
|
|
||||||
.alerts-row-button,
|
|
||||||
.alerts-footer-button {
|
|
||||||
min-width: 64px;
|
|
||||||
height: 24px;
|
|
||||||
padding: 0 8px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-action-button {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-toolbar-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(180px, 1.2fr) repeat(4, minmax(110px, .6fr));
|
|
||||||
gap: 8px;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-input,
|
|
||||||
.alerts-select,
|
|
||||||
.alerts-textarea {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 0;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
padding: 4px 6px;
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-input,
|
|
||||||
.alerts-select {
|
|
||||||
height: 28px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-table-wrap {
|
|
||||||
height: 430px;
|
|
||||||
overflow: auto;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 2px solid #606060;
|
|
||||||
border-left: 2px solid #606060;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-table {
|
|
||||||
width: 100%;
|
|
||||||
border-collapse: collapse;
|
|
||||||
table-layout: fixed;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 14px;
|
|
||||||
color: #000000;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-table thead th {
|
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
z-index: 2;
|
|
||||||
padding: 6px;
|
|
||||||
text-align: left;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
box-shadow: inset 0 1px 0 #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-table tbody tr:nth-child(odd) { background: rgba(255,255,255,.96); }
|
|
||||||
.alerts-table tbody tr:nth-child(even) { background: rgba(240,244,255,.9); }
|
|
||||||
.alerts-table tbody tr:hover { background: #d8e5f8; }
|
|
||||||
.alerts-table tbody tr.is-selected { background: #c5dcff; }
|
|
||||||
|
|
||||||
.alerts-table td {
|
|
||||||
padding: 6px;
|
|
||||||
border-bottom: 1px solid #e1e1e1;
|
|
||||||
vertical-align: middle;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-col-check { width: 34px; }
|
|
||||||
.alerts-col-severity { width: 76px; }
|
|
||||||
.alerts-col-status { width: 82px; }
|
|
||||||
.alerts-col-code { width: 70px; }
|
|
||||||
.alerts-col-time { width: 110px; }
|
|
||||||
.alerts-col-actions { width: 88px; }
|
|
||||||
|
|
||||||
.alerts-pill {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 0 6px;
|
|
||||||
color: #222222;
|
|
||||||
background: #f1f1f1;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #b0b0b0;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-pill.low { background: #deebff; }
|
|
||||||
.alerts-pill.medium { background: #fff2c8; }
|
|
||||||
.alerts-pill.high { background: #ffdcdc; }
|
|
||||||
.alerts-pill.open { background: #f2e1ff; }
|
|
||||||
.alerts-pill.acked { background: #e2f0e2; }
|
|
||||||
.alerts-pill.closed { background: #ececec; }
|
|
||||||
|
|
||||||
.alerts-info-list {
|
|
||||||
display: grid;
|
|
||||||
gap: 6px;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
list-style: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-info-item {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 110px minmax(0, 1fr);
|
|
||||||
gap: 8px;
|
|
||||||
align-items: start;
|
|
||||||
padding: 6px 8px;
|
|
||||||
background: #f5f5f5;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #c0c0c0;
|
|
||||||
border-bottom: 1px solid #c0c0c0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-info-item strong {
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-info-item span {
|
|
||||||
min-width: 0;
|
|
||||||
color: #222222;
|
|
||||||
word-break: break-word;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-json-box {
|
|
||||||
max-height: 180px;
|
|
||||||
overflow: auto;
|
|
||||||
margin: 0;
|
|
||||||
padding: 8px;
|
|
||||||
color: #b7ffc8;
|
|
||||||
background: #050505;
|
|
||||||
border-top: 2px solid #808080;
|
|
||||||
border-left: 2px solid #808080;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
font-family: "MonoCraft", "Courier New", monospace;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 15px;
|
|
||||||
white-space: pre-wrap;
|
|
||||||
word-break: break-word;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-mini-note {
|
|
||||||
margin-top: 8px;
|
|
||||||
padding: 8px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffcc;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #a08000;
|
|
||||||
border-bottom: 1px solid #a08000;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-action-stack {
|
|
||||||
display: grid;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-footerbar {
|
|
||||||
flex: 0 0 auto;
|
|
||||||
min-height: 42px;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 10px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
background: #dfdfdf;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-footer-left,
|
|
||||||
.alerts-footer-right {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-status-pill {
|
|
||||||
min-height: 24px;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
padding: 0 8px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 1120px) {
|
|
||||||
.alerts-summary-grid {
|
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-content-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-toolbar-grid {
|
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 760px) {
|
|
||||||
.alerts-summary-grid,
|
|
||||||
.alerts-toolbar-grid {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-table-wrap {
|
|
||||||
height: 360px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-panel-header,
|
|
||||||
.alerts-footerbar {
|
|
||||||
align-items: flex-start;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
|
|
||||||
.alerts-info-item {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,501 +0,0 @@
|
|||||||
.boxes-page-body {
|
|
||||||
display: grid;
|
|
||||||
gap: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-summary-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-stat-card {
|
|
||||||
min-width: 0;
|
|
||||||
padding: 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
box-shadow: inset 1px 1px 0 #f7f7f7, inset -1px -1px 0 #b0b0b0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-stat-card.is-info { background: linear-gradient(180deg, #d7e6fb, #bfd7f8); }
|
|
||||||
.boxes-stat-card.is-ok { background: linear-gradient(180deg, #dbf4dc, #c3ebc5); }
|
|
||||||
.boxes-stat-card.is-warning { background: linear-gradient(180deg, #fff1c9, #ffe39f); }
|
|
||||||
.boxes-stat-card.is-danger { background: linear-gradient(180deg, #ffd8d8, #f1b3b3); }
|
|
||||||
|
|
||||||
.boxes-stat-label {
|
|
||||||
margin: 0 0 4px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
text-transform: uppercase;
|
|
||||||
color: #333333;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-stat-value {
|
|
||||||
margin: 0;
|
|
||||||
font-size: 24px;
|
|
||||||
line-height: 24px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-stat-note {
|
|
||||||
margin: 6px 0 0;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 0 6px;
|
|
||||||
color: #222222;
|
|
||||||
background: rgba(255,255,255,.65);
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #a0a0a0;
|
|
||||||
border-bottom: 1px solid #a0a0a0;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-hero-note {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 12px;
|
|
||||||
padding: 8px 10px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffcc;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #a08000;
|
|
||||||
border-bottom: 1px solid #a08000;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-hero-note strong {
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-hero-note span {
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-hero-tags {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-hero-tag,
|
|
||||||
.boxes-flag {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 0 6px;
|
|
||||||
color: #222222;
|
|
||||||
background: #f1f1f1;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #b0b0b0;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-content-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(0, 1.45fr) minmax(320px, .75fr);
|
|
||||||
gap: 10px;
|
|
||||||
min-height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-column {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
gap: 10px;
|
|
||||||
min-height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-panel {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
min-height: 0;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
box-shadow: inset 1px 1px 0 rgba(255,255,255,.7), inset -1px -1px 0 rgba(0,0,0,.08);
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-files-panel {
|
|
||||||
min-height: 300px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-panel-header {
|
|
||||||
flex: 0 0 auto;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 8px;
|
|
||||||
min-height: 34px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
box-shadow: inset 1px 1px 0 #f7f7f7;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-panel-title {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
min-width: 0;
|
|
||||||
min-height: 22px;
|
|
||||||
font-weight: bold;
|
|
||||||
font-size: 15px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-panel-sub {
|
|
||||||
color: #444444;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
font-weight: normal;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-panel-tools {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-panel-body {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 0;
|
|
||||||
padding: 10px;
|
|
||||||
overflow: hidden;
|
|
||||||
background-color: #ffffff;
|
|
||||||
background-image: linear-gradient(180deg, rgba(255,255,255,.9), rgba(238,238,238,.58));
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-tool-button,
|
|
||||||
.boxes-page-button,
|
|
||||||
.boxes-action-button,
|
|
||||||
.boxes-row-button {
|
|
||||||
min-width: 62px;
|
|
||||||
height: 24px;
|
|
||||||
padding: 0 8px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-tool-button.is-danger,
|
|
||||||
.boxes-action-button.is-danger {
|
|
||||||
color: #ffffff;
|
|
||||||
background: #800000;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-toolbar-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(200px, 1.3fr) repeat(4, minmax(110px, .55fr));
|
|
||||||
gap: 8px;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-input,
|
|
||||||
.boxes-select {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 0;
|
|
||||||
height: 28px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
padding: 4px 6px;
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-table-wrap {
|
|
||||||
width: 100%;
|
|
||||||
min-height: 0;
|
|
||||||
height: 460px;
|
|
||||||
overflow-y: auto;
|
|
||||||
overflow-x: hidden;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 2px solid #606060;
|
|
||||||
border-left: 2px solid #606060;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-table {
|
|
||||||
width: 100%;
|
|
||||||
border-collapse: collapse;
|
|
||||||
table-layout: fixed;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 14px;
|
|
||||||
color: #000000;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-table thead th {
|
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
z-index: 2;
|
|
||||||
padding: 6px;
|
|
||||||
text-align: left;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
box-shadow: inset 0 1px 0 #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-table tbody tr:nth-child(odd) { background: rgba(255,255,255,.96); }
|
|
||||||
.boxes-table tbody tr:nth-child(even) { background: rgba(240,244,255,.9); }
|
|
||||||
.boxes-table tbody tr:hover { background: #d8e5f8; }
|
|
||||||
.boxes-table tbody tr.is-selected { background: #c5dcff; }
|
|
||||||
|
|
||||||
.boxes-table td {
|
|
||||||
padding: 6px;
|
|
||||||
border-bottom: 1px solid #e1e1e1;
|
|
||||||
vertical-align: middle;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-col-check { width: 34px; }
|
|
||||||
.boxes-col-id { width: 190px; }
|
|
||||||
.boxes-col-status { width: 84px; }
|
|
||||||
.boxes-col-files { width: 58px; }
|
|
||||||
.boxes-col-size { width: 76px; }
|
|
||||||
.boxes-col-retention { width: 96px; }
|
|
||||||
.boxes-col-expires { width: 126px; }
|
|
||||||
.boxes-col-actions { width: 98px; }
|
|
||||||
|
|
||||||
.boxes-status-pill {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 0 6px;
|
|
||||||
color: #222222;
|
|
||||||
background: #f1f1f1;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #b0b0b0;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-status-pill.ready { background: #def2e0; }
|
|
||||||
.boxes-status-pill.uploading { background: #fff1c9; }
|
|
||||||
.boxes-status-pill.attention { background: #ffe2bf; }
|
|
||||||
.boxes-status-pill.expired { background: #ffdcdc; }
|
|
||||||
.boxes-status-pill.consumed { background: #ead7ff; }
|
|
||||||
.boxes-status-pill.legacy { background: #ececec; }
|
|
||||||
|
|
||||||
.boxes-flags-cell,
|
|
||||||
.boxes-action-cell {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 4px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
min-width: 0;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-action-cell a {
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-empty-state {
|
|
||||||
padding: 24px 12px;
|
|
||||||
text-align: center;
|
|
||||||
color: #444444;
|
|
||||||
background: linear-gradient(180deg, rgba(255,255,255,.95), rgba(242,242,242,.95));
|
|
||||||
font-size: 14px;
|
|
||||||
line-height: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-footer-bar {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 8px;
|
|
||||||
margin-top: 8px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-pagination {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-detail-body {
|
|
||||||
display: grid;
|
|
||||||
gap: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-info-list {
|
|
||||||
display: grid;
|
|
||||||
gap: 6px;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
list-style: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-info-item {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 84px minmax(0, 1fr);
|
|
||||||
gap: 8px;
|
|
||||||
align-items: start;
|
|
||||||
padding: 6px 8px;
|
|
||||||
background: #f5f5f5;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #c0c0c0;
|
|
||||||
border-bottom: 1px solid #c0c0c0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-info-item strong {
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-info-item span {
|
|
||||||
min-width: 0;
|
|
||||||
color: #222222;
|
|
||||||
word-break: break-word;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-action-stack {
|
|
||||||
display: grid;
|
|
||||||
gap: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-action-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
|
||||||
gap: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-action-button {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 0;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-file-list {
|
|
||||||
display: grid;
|
|
||||||
gap: 8px;
|
|
||||||
min-height: 0;
|
|
||||||
height: 320px;
|
|
||||||
overflow-y: auto;
|
|
||||||
overflow-x: hidden;
|
|
||||||
padding-right: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-column:first-child > .boxes-panel {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-column:first-child > .boxes-panel > .boxes-panel-body {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-column:first-child .boxes-table-wrap {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
height: auto;
|
|
||||||
min-height: 560px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-file-card {
|
|
||||||
display: grid;
|
|
||||||
gap: 6px;
|
|
||||||
padding: 8px;
|
|
||||||
background: #f8f8f8;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #c0c0c0;
|
|
||||||
border-bottom: 1px solid #c0c0c0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-file-row {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-file-name {
|
|
||||||
min-width: 0;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
white-space: nowrap;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-file-meta {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
color: #333333;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-file-link {
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 1100px) {
|
|
||||||
.boxes-summary-grid,
|
|
||||||
.boxes-content-grid {
|
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-column-side {
|
|
||||||
grid-column: 1 / -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-toolbar-grid {
|
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 720px) {
|
|
||||||
.boxes-summary-grid,
|
|
||||||
.boxes-content-grid,
|
|
||||||
.boxes-toolbar-grid,
|
|
||||||
.boxes-action-grid {
|
|
||||||
grid-template-columns: minmax(0, 1fr);
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-hero-note,
|
|
||||||
.boxes-footer-bar {
|
|
||||||
align-items: flex-start;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-table-wrap {
|
|
||||||
height: 420px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.boxes-column:first-child .boxes-table-wrap {
|
|
||||||
min-height: 420px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,289 +0,0 @@
|
|||||||
/* ==============================================
|
|
||||||
Dashboard-specific styles (shared with admin)
|
|
||||||
Reusable across account dashboard pages
|
|
||||||
============================================== */
|
|
||||||
|
|
||||||
/* Hero section */
|
|
||||||
.dashboard-hero {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(0, 1fr) 330px;
|
|
||||||
gap: 10px;
|
|
||||||
padding: 9px;
|
|
||||||
align-items: stretch;
|
|
||||||
}
|
|
||||||
|
|
||||||
.hero-copy h2 { margin: 0 0 5px; font-size: 22px; line-height: 24px; }
|
|
||||||
.hero-copy p { margin: 0; color: #333; font-size: 13px; line-height: 15px; }
|
|
||||||
|
|
||||||
.hero-status {
|
|
||||||
display: grid;
|
|
||||||
gap: 4px;
|
|
||||||
align-content: center;
|
|
||||||
padding: 7px;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.hero-status-row { display: flex; justify-content: space-between; gap: 8px; }
|
|
||||||
.status-ok { color: #008000; }
|
|
||||||
.status-warn { color: #8a6200; }
|
|
||||||
.status-danger { color: #800000; }
|
|
||||||
|
|
||||||
/* Stats grid */
|
|
||||||
.stats-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
|
||||||
gap: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.stat-card {
|
|
||||||
position: relative;
|
|
||||||
min-height: 122px;
|
|
||||||
padding: 10px 11px 10px 14px;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.stat-card::before {
|
|
||||||
content: "";
|
|
||||||
position: absolute;
|
|
||||||
inset: 0 auto 0 0;
|
|
||||||
width: 7px;
|
|
||||||
border-left: 7px solid #000078;
|
|
||||||
pointer-events: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.stat-card.is-ok { background: linear-gradient(180deg, #eeffee, #ffffff); }
|
|
||||||
.stat-card.is-ok::before { border-left-color: #008000; }
|
|
||||||
.stat-card.is-info { background: linear-gradient(180deg, #edf4ff, #ffffff); }
|
|
||||||
.stat-card.is-info::before { border-left-color: #000078; }
|
|
||||||
.stat-card.is-warning { background: linear-gradient(180deg, #ffffcc, #ffffff); }
|
|
||||||
.stat-card.is-warning::before { border-left-color: #ffcc00; }
|
|
||||||
.stat-card.is-danger {
|
|
||||||
color: #000;
|
|
||||||
background: repeating-linear-gradient(45deg, #fff2f2 0 6px, #ffe1e1 6px 12px);
|
|
||||||
}
|
|
||||||
.stat-card.is-danger::before { border-left-color: #800000; }
|
|
||||||
|
|
||||||
.stat-label { margin: 0 0 6px; color: #333; font-size: 13px; line-height: 13px; font-weight: bold; }
|
|
||||||
.stat-value { margin: 0 0 7px; font-size: 32px; line-height: 32px; font-weight: bold; }
|
|
||||||
.stat-note { display: flex; gap: 4px; flex-wrap: wrap; margin: 0; color: #222; font-size: 12px; line-height: 14px; }
|
|
||||||
.stat-note-pill {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 1px 6px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Main two-column grid */
|
|
||||||
.dashboard-main-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(0, 1fr) minmax(0, 1fr);
|
|
||||||
gap: 12px;
|
|
||||||
align-items: start;
|
|
||||||
}
|
|
||||||
|
|
||||||
.dashboard-span-2 { grid-column: 1 / -1; }
|
|
||||||
|
|
||||||
/* Dashboard body */
|
|
||||||
.dashboard-body {
|
|
||||||
display: grid;
|
|
||||||
gap: 12px;
|
|
||||||
padding: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Section windows */
|
|
||||||
.section-window { min-height: 0; box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf, 3px 4px 0 rgba(0,0,0,.38); }
|
|
||||||
.section-body { margin: 0 6px 6px; padding: 8px; min-height: 0; }
|
|
||||||
|
|
||||||
/* Scroll panels */
|
|
||||||
.scroll-panel { overflow: auto; background: #ffffff; border-top: 2px solid #606060; border-left: 2px solid #606060; border-right: 2px solid #ffffff; border-bottom: 2px solid #ffffff; }
|
|
||||||
.alerts-scroll { height: 326px; }
|
|
||||||
.boxes-scroll { height: 352px; }
|
|
||||||
.activity-scroll { height: 326px; }
|
|
||||||
|
|
||||||
/* Alerts */
|
|
||||||
.alert-list { display: grid; min-width: 0; }
|
|
||||||
.alert-row {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 72px minmax(0, 1fr) auto;
|
|
||||||
gap: 8px;
|
|
||||||
align-items: start;
|
|
||||||
min-height: 74px;
|
|
||||||
padding: 7px;
|
|
||||||
color: #000;
|
|
||||||
border-bottom: 1px solid #dfdfdf;
|
|
||||||
background: #ffffff;
|
|
||||||
}
|
|
||||||
.alert-row:nth-child(even) { background: #f5f8ff; }
|
|
||||||
.alert-row.is-dismissed { display: none; }
|
|
||||||
.alert-severity {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: center;
|
|
||||||
min-width: 60px;
|
|
||||||
min-height: 20px;
|
|
||||||
padding: 2px 5px;
|
|
||||||
text-transform: uppercase;
|
|
||||||
font-weight: bold;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
}
|
|
||||||
.alert-row[data-severity="low"] .alert-severity { color: #000078; }
|
|
||||||
.alert-row[data-severity="medium"] .alert-severity { color: #8a6200; background: #ffffcc; }
|
|
||||||
.alert-row[data-severity="high"] .alert-severity { color: #ffffff; background: #800000; }
|
|
||||||
.alert-title { margin: 0 0 3px; font-weight: bold; font-size: 14px; line-height: 15px; }
|
|
||||||
.alert-desc { margin: 0 0 3px; color: #333; font-size: 12px; line-height: 14px; }
|
|
||||||
.alert-trace { margin: 0; color: #555; font-family: 'MonoCraft', 'Courier New', monospace; font-size: 10px; line-height: 13px; overflow-wrap: anywhere; }
|
|
||||||
.alert-actions { display: flex; gap: 5px; flex-wrap: wrap; justify-content: flex-end; }
|
|
||||||
|
|
||||||
/* Boxes table */
|
|
||||||
.box-table {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 900px;
|
|
||||||
border-collapse: collapse;
|
|
||||||
color: #000;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 14px;
|
|
||||||
}
|
|
||||||
.box-table th, .box-table td { padding: 6px 7px; border-bottom: 1px solid #dfdfdf; text-align: left; vertical-align: middle; }
|
|
||||||
.box-table th { position: sticky; top: 0; z-index: 5; background: #dfdfdf; border-bottom: 1px solid #808080; }
|
|
||||||
.box-table tr:nth-child(even) td { background: #f5f8ff; }
|
|
||||||
.box-actions { display: flex; gap: 5px; flex-wrap: nowrap; }
|
|
||||||
.box-action-button { min-width: 62px; height: 22px; padding: 0 6px; font-size: 12px; line-height: 12px; }
|
|
||||||
|
|
||||||
/* Activity */
|
|
||||||
.activity-list { display: grid; }
|
|
||||||
.activity-row {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 56px minmax(0, 1fr) auto;
|
|
||||||
gap: 9px;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 48px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
border-bottom: 1px solid #dfdfdf;
|
|
||||||
background: #ffffff;
|
|
||||||
color: #000;
|
|
||||||
}
|
|
||||||
.activity-row:nth-child(even) { background: #f5f8ff; }
|
|
||||||
.activity-time { font-weight: bold; color: #000078; }
|
|
||||||
.activity-title { margin: 0 0 2px; font-weight: bold; }
|
|
||||||
.activity-meta { margin: 0; color: #555; font-size: 12px; line-height: 13px; }
|
|
||||||
|
|
||||||
/* Modal / Popup */
|
|
||||||
.modal-backdrop {
|
|
||||||
position: fixed;
|
|
||||||
inset: 0;
|
|
||||||
display: none;
|
|
||||||
background: rgba(128, 128, 128, .42);
|
|
||||||
z-index: 70;
|
|
||||||
}
|
|
||||||
.modal-backdrop.is-visible { display: block; }
|
|
||||||
|
|
||||||
.popup-window {
|
|
||||||
position: fixed;
|
|
||||||
left: 50%;
|
|
||||||
top: 50%;
|
|
||||||
transform: translate(calc(-50% - 1px), -50%);
|
|
||||||
width: min(760px, calc(100vw - 24px));
|
|
||||||
max-height: min(760px, calc(100vh - 24px));
|
|
||||||
display: none;
|
|
||||||
z-index: 80;
|
|
||||||
}
|
|
||||||
.popup-window.is-visible { display: flex; animation: popup-open 160ms steps(5, end); }
|
|
||||||
@keyframes popup-open {
|
|
||||||
from { transform: translate(calc(-50% - 1px), calc(-50% + 10px)) scale(.97); opacity: .45; }
|
|
||||||
to { transform: translate(calc(-50% - 1px), -50%) scale(1); opacity: 1; }
|
|
||||||
}
|
|
||||||
.popup-body { margin: 0 6px 6px; padding: 10px; max-height: calc(100vh - 90px); overflow: auto; color: #000; }
|
|
||||||
.metadata-pre {
|
|
||||||
min-height: 240px;
|
|
||||||
margin: 0;
|
|
||||||
padding: 10px;
|
|
||||||
overflow: auto;
|
|
||||||
color: #b7ffc8;
|
|
||||||
background: #030403;
|
|
||||||
background-image: repeating-linear-gradient(transparent 0 4px, rgba(0,255,102,.018) 4px 6px);
|
|
||||||
font-family: 'MonoCraft', 'Courier New', monospace;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 16px;
|
|
||||||
white-space: pre-wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Tiny button (for alerts / boxes) */
|
|
||||||
.tiny-button {
|
|
||||||
min-width: 56px;
|
|
||||||
height: 22px;
|
|
||||||
display: inline-grid;
|
|
||||||
place-items: center;
|
|
||||||
padding: 0 7px;
|
|
||||||
color: #000;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #000000;
|
|
||||||
border-bottom: 1px solid #000000;
|
|
||||||
box-shadow: inset -1px -1px 0 #808080, inset 1px 1px 0 #dfdfdf;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
text-decoration: none;
|
|
||||||
}
|
|
||||||
.tiny-button:hover { filter: brightness(1.06); }
|
|
||||||
|
|
||||||
/* Compact mode */
|
|
||||||
body.is-compact .dashboard-body { gap: 8px; }
|
|
||||||
body.is-compact .section-body { padding: 5px; }
|
|
||||||
body.is-compact .alerts-scroll,
|
|
||||||
body.is-compact .boxes-scroll { height: 280px; }
|
|
||||||
body.is-compact .activity-scroll { height: 280px; }
|
|
||||||
body.is-compact .alert-row { min-height: 62px; }
|
|
||||||
body.is-compact .activity-row { min-height: 42px; }
|
|
||||||
|
|
||||||
/* Responsive: medium */
|
|
||||||
@media (max-width: 1180px) {
|
|
||||||
.stats-grid { grid-template-columns: repeat(2, minmax(0, 1fr)); }
|
|
||||||
.dashboard-hero { grid-template-columns: 1fr; }
|
|
||||||
.dashboard-main-grid { grid-template-columns: 1fr; }
|
|
||||||
.dashboard-span-2 { grid-column: auto; }
|
|
||||||
.alerts-scroll, .boxes-scroll { height: 310px; }
|
|
||||||
.activity-scroll { height: 310px; }
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Responsive: small (mobile) */
|
|
||||||
@media (max-width: 760px) {
|
|
||||||
.dashboard-body { padding: 6px; gap: 8px; }
|
|
||||||
.stats-grid { grid-template-columns: 1fr; }
|
|
||||||
.stat-card { min-height: 112px; }
|
|
||||||
.alert-row { grid-template-columns: 1fr; min-height: 0; }
|
|
||||||
.alert-actions { justify-content: flex-start; }
|
|
||||||
.alerts-scroll, .boxes-scroll, .activity-scroll { height: 320px; }
|
|
||||||
.boxes-scroll { overflow-x: auto; }
|
|
||||||
.activity-row { grid-template-columns: 48px minmax(0, 1fr); }
|
|
||||||
.activity-row .tag { grid-column: 2; justify-self: start; }
|
|
||||||
.popup-window {
|
|
||||||
left: 0;
|
|
||||||
top: 0;
|
|
||||||
transform: none;
|
|
||||||
width: 100vw;
|
|
||||||
height: 100dvh;
|
|
||||||
max-height: none;
|
|
||||||
border: 0;
|
|
||||||
box-shadow: none;
|
|
||||||
}
|
|
||||||
.popup-window.is-visible { animation: popup-open-mobile 150ms steps(5, end); }
|
|
||||||
@keyframes popup-open-mobile { from { transform: translateY(10px); opacity: .35; } to { transform: translateY(0); opacity: 1; } }
|
|
||||||
.popup-body { max-height: calc(100dvh - 40px); }
|
|
||||||
}
|
|
||||||
@@ -1,100 +0,0 @@
|
|||||||
.security-page-body { display: grid; gap: 10px; }
|
|
||||||
.security-grid { display: grid; grid-template-columns: minmax(260px, .65fr) minmax(0, 1.35fr); gap: 10px; }
|
|
||||||
.security-panel {
|
|
||||||
min-height: 0;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
}
|
|
||||||
.security-panel-header {
|
|
||||||
min-height: 34px;
|
|
||||||
display: flex;
|
|
||||||
justify-content: space-between;
|
|
||||||
align-items: center;
|
|
||||||
gap: 8px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
}
|
|
||||||
.security-panel-header span { color: #444444; font-size: 12px; }
|
|
||||||
.security-panel-body { flex: 1 1 auto; min-height: 0; padding: 10px; overflow: auto; }
|
|
||||||
.security-field { display: grid; gap: 4px; font-size: 12px; }
|
|
||||||
.security-input {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 0;
|
|
||||||
height: 28px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
padding: 4px 6px;
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
.security-button { margin-top: 8px; min-width: 100px; height: 24px; padding: 0 8px; font-size: 12px; line-height: 12px; }
|
|
||||||
.security-danger { color: #7a0000; }
|
|
||||||
.security-note {
|
|
||||||
margin-top: 8px;
|
|
||||||
padding: 8px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffcc;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #a08000;
|
|
||||||
border-bottom: 1px solid #a08000;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
.security-list { margin: 0; padding-left: 16px; display: grid; gap: 6px; font-size: 12px; }
|
|
||||||
.security-ban-grid { display: grid; grid-template-columns: minmax(0, 1.1fr) minmax(260px, .9fr); gap: 10px; }
|
|
||||||
.security-table-toolbar { display: grid; grid-template-columns: 1fr 180px; gap: 8px; margin-bottom: 8px; }
|
|
||||||
.security-bans-wrap { height: 260px; min-height: 260px; }
|
|
||||||
.security-ip-detail {
|
|
||||||
min-height: 0;
|
|
||||||
padding: 10px;
|
|
||||||
background: #f5f5f5;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #b0b0b0;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
}
|
|
||||||
.security-ip-detail h3 { margin: 0 0 8px; font-size: 16px; line-height: 16px; }
|
|
||||||
.security-ip-detail ul { margin: 0; padding: 0; list-style: none; display: grid; gap: 6px; font-size: 12px; }
|
|
||||||
.security-bans-body-row.is-selected { background: #c5dcff; }
|
|
||||||
.security-table-wrap {
|
|
||||||
min-height: 280px;
|
|
||||||
height: 320px;
|
|
||||||
overflow: auto;
|
|
||||||
border-top: 2px solid #606060;
|
|
||||||
border-left: 2px solid #606060;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
}
|
|
||||||
.security-table { width: 100%; border-collapse: collapse; table-layout: fixed; font-size: 12px; line-height: 14px; }
|
|
||||||
.security-table th,
|
|
||||||
.security-table td { padding: 6px; border-bottom: 1px solid #e1e1e1; text-align: left; white-space: nowrap; overflow: hidden; text-overflow: ellipsis; }
|
|
||||||
.security-table th { position: sticky; top: 0; background: #dfdfdf; z-index: 2; }
|
|
||||||
.security-docs h4 { margin: 4px 0; font-size: 13px; }
|
|
||||||
.security-docs p { margin: 0 0 8px; font-size: 12px; line-height: 1.4; }
|
|
||||||
.security-docs pre {
|
|
||||||
margin: 0 0 10px;
|
|
||||||
padding: 8px;
|
|
||||||
background: #f2f2f2;
|
|
||||||
border: 1px solid #c0c0c0;
|
|
||||||
overflow: auto;
|
|
||||||
font-size: 11px;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 980px) {
|
|
||||||
.security-grid,
|
|
||||||
.security-ban-grid,
|
|
||||||
.security-table-toolbar {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,516 +0,0 @@
|
|||||||
.settings-page-body {
|
|
||||||
display: grid;
|
|
||||||
gap: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-summary-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-stat-card {
|
|
||||||
min-width: 0;
|
|
||||||
padding: 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
box-shadow: inset 1px 1px 0 #f7f7f7, inset -1px -1px 0 #b0b0b0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-stat-card.is-info { background: linear-gradient(180deg, #d7e6fb, #bfd7f8); }
|
|
||||||
.settings-stat-card.is-ok { background: linear-gradient(180deg, #dbf4dc, #c3ebc5); }
|
|
||||||
.settings-stat-card.is-warning { background: linear-gradient(180deg, #fff1c9, #ffe39f); }
|
|
||||||
.settings-stat-card.is-danger { background: linear-gradient(180deg, #ffd8d8, #f1b3b3); }
|
|
||||||
|
|
||||||
.settings-stat-label {
|
|
||||||
margin: 0 0 4px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
text-transform: uppercase;
|
|
||||||
color: #333333;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-stat-value {
|
|
||||||
margin: 0;
|
|
||||||
font-size: 24px;
|
|
||||||
line-height: 24px;
|
|
||||||
font-weight: bold;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-stat-note {
|
|
||||||
margin: 6px 0 0;
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 0 6px;
|
|
||||||
color: #222222;
|
|
||||||
background: rgba(255,255,255,.65);
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #a0a0a0;
|
|
||||||
border-bottom: 1px solid #a0a0a0;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-main-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 238px minmax(0, 1fr);
|
|
||||||
gap: 10px;
|
|
||||||
min-height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-sidebar-panel {
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-sidebar {
|
|
||||||
position: sticky;
|
|
||||||
top: 48px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-workbench {
|
|
||||||
display: grid;
|
|
||||||
gap: 10px;
|
|
||||||
min-width: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-panel,
|
|
||||||
.settings-hero-panel {
|
|
||||||
min-width: 0;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
box-shadow: inset 1px 1px 0 rgba(255,255,255,.7), inset -1px -1px 0 rgba(0,0,0,.08);
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-panel {
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-panel-header {
|
|
||||||
flex: 0 0 auto;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 8px;
|
|
||||||
min-height: 34px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
box-shadow: inset 1px 1px 0 #f7f7f7;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-panel-title {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
min-width: 0;
|
|
||||||
min-height: 22px;
|
|
||||||
font-weight: bold;
|
|
||||||
font-size: 15px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-panel-sub {
|
|
||||||
color: #444444;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
font-weight: normal;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-panel-tools {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-panel-body {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 0;
|
|
||||||
padding: 10px;
|
|
||||||
overflow: hidden;
|
|
||||||
background-color: #ffffff;
|
|
||||||
background-image: linear-gradient(180deg, rgba(255,255,255,.9), rgba(238,238,238,.58));
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-hero-panel {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(0, 1.2fr) minmax(280px, .8fr);
|
|
||||||
gap: 10px;
|
|
||||||
padding: 10px;
|
|
||||||
background-image: linear-gradient(180deg, rgba(255,255,255,.92), rgba(238,238,238,.58));
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-hero-copy h2 {
|
|
||||||
margin: 0 0 6px;
|
|
||||||
font-size: 18px;
|
|
||||||
line-height: 18px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-hero-copy p {
|
|
||||||
margin: 0;
|
|
||||||
color: #222222;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-hero-legend {
|
|
||||||
display: grid;
|
|
||||||
gap: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-legend-row {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
color: #222222;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-search {
|
|
||||||
display: grid;
|
|
||||||
gap: 6px;
|
|
||||||
margin-bottom: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-search label {
|
|
||||||
font-weight: bold;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-input,
|
|
||||||
.settings-select {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 0;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
padding: 4px 6px;
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-input,
|
|
||||||
.settings-select {
|
|
||||||
height: 28px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-category-list {
|
|
||||||
display: grid;
|
|
||||||
gap: 4px;
|
|
||||||
margin: 0;
|
|
||||||
padding: 0;
|
|
||||||
list-style: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-category-button {
|
|
||||||
width: 100%;
|
|
||||||
min-height: 30px;
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 24px minmax(0, 1fr) auto;
|
|
||||||
align-items: center;
|
|
||||||
gap: 7px;
|
|
||||||
padding: 4px 6px;
|
|
||||||
color: #000000;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
font-family: inherit;
|
|
||||||
text-align: left;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-category-button.is-active {
|
|
||||||
color: #ffffff;
|
|
||||||
background: #000078;
|
|
||||||
border-top-color: #000000;
|
|
||||||
border-left-color: #000000;
|
|
||||||
border-right-color: #ffffff;
|
|
||||||
border-bottom-color: #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-category-count,
|
|
||||||
.settings-dirty-chip,
|
|
||||||
.settings-badge {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 0 6px;
|
|
||||||
color: #222222;
|
|
||||||
background: #f1f1f1;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #b0b0b0;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-category-button.is-active .settings-category-count {
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffcc;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-dirty-chip {
|
|
||||||
min-width: 78px;
|
|
||||||
justify-content: center;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-dirty-chip.is-dirty {
|
|
||||||
background: #ffffcc;
|
|
||||||
border: 3px solid transparent;
|
|
||||||
border-image: repeating-linear-gradient(45deg, #111111 0 8px, #ffcc00 8px 16px) 3;
|
|
||||||
}
|
|
||||||
|
|
||||||
.badge-default { background: #ececec; }
|
|
||||||
.badge-env { background: #c7d8f2; }
|
|
||||||
.badge-db { background: #d2efcf; }
|
|
||||||
.badge-hard { background: #ffd9d9; }
|
|
||||||
|
|
||||||
.settings-tool-button,
|
|
||||||
.settings-mini-button,
|
|
||||||
.settings-popup-close {
|
|
||||||
min-width: 64px;
|
|
||||||
height: 24px;
|
|
||||||
padding: 0 8px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-action-summary {
|
|
||||||
margin-bottom: 8px;
|
|
||||||
padding: 8px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffcc;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #a08000;
|
|
||||||
border-bottom: 1px solid #a08000;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-groups {
|
|
||||||
display: grid;
|
|
||||||
gap: 10px;
|
|
||||||
min-height: 0;
|
|
||||||
max-height: 700px;
|
|
||||||
overflow-y: auto;
|
|
||||||
overflow-x: hidden;
|
|
||||||
padding-right: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-group {
|
|
||||||
display: grid;
|
|
||||||
gap: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-group[hidden] {
|
|
||||||
display: none;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-group-title {
|
|
||||||
min-height: 28px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
color: #000000;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
font-weight: bold;
|
|
||||||
font-size: 14px;
|
|
||||||
line-height: 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-table-wrap {
|
|
||||||
border-top: 2px solid #606060;
|
|
||||||
border-left: 2px solid #606060;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
background: #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-table {
|
|
||||||
width: 100%;
|
|
||||||
border-collapse: collapse;
|
|
||||||
table-layout: fixed;
|
|
||||||
color: #000000;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-table th,
|
|
||||||
.settings-table td {
|
|
||||||
padding: 6px;
|
|
||||||
text-align: left;
|
|
||||||
vertical-align: top;
|
|
||||||
border-bottom: 1px solid #e1e1e1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-table th {
|
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
z-index: 2;
|
|
||||||
background: #dfdfdf;
|
|
||||||
box-shadow: inset 0 1px 0 #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-table tbody tr:nth-child(odd) { background: rgba(255,255,255,.96); }
|
|
||||||
.settings-table tbody tr:nth-child(even) { background: rgba(240,244,255,.9); }
|
|
||||||
.setting-row.is-locked { color: #555555; background: #efefef; }
|
|
||||||
.setting-row.is-hidden { display: none; }
|
|
||||||
.setting-row.is-invalid { background: #fff1c9; }
|
|
||||||
|
|
||||||
.setting-meta {
|
|
||||||
display: grid;
|
|
||||||
gap: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setting-meta strong {
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setting-meta code {
|
|
||||||
color: #1b325f;
|
|
||||||
font-size: 11px;
|
|
||||||
line-height: 12px;
|
|
||||||
word-break: break-word;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setting-control {
|
|
||||||
display: grid;
|
|
||||||
gap: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setting-input-row {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setting-hint {
|
|
||||||
color: #444444;
|
|
||||||
font-size: 11px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.setting-actions {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-modal-backdrop {
|
|
||||||
position: fixed;
|
|
||||||
inset: 0;
|
|
||||||
display: none;
|
|
||||||
background: rgba(0,0,0,.35);
|
|
||||||
z-index: 90;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-modal-backdrop.is-visible {
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-popup {
|
|
||||||
position: fixed;
|
|
||||||
left: 50%;
|
|
||||||
top: 50%;
|
|
||||||
width: min(520px, calc(100vw - 24px));
|
|
||||||
display: none;
|
|
||||||
transform: translate(-50%, -50%);
|
|
||||||
color: #000000;
|
|
||||||
background: var(--w98-gray);
|
|
||||||
border-top: 2px solid #ffffff;
|
|
||||||
border-left: 2px solid #ffffff;
|
|
||||||
border-right: 2px solid #000000;
|
|
||||||
border-bottom: 2px solid #000000;
|
|
||||||
box-shadow: 6px 6px 0 rgba(0,0,0,.35);
|
|
||||||
z-index: 95;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-popup.is-visible {
|
|
||||||
display: block;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-popup-titlebar {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 8px;
|
|
||||||
min-height: 28px;
|
|
||||||
padding: 4px 6px;
|
|
||||||
color: #ffffff;
|
|
||||||
background: #000078;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-popup-body {
|
|
||||||
padding: 10px;
|
|
||||||
background: #f5f5f5;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-popup-body p,
|
|
||||||
.settings-popup-body ul {
|
|
||||||
margin: 0 0 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 1100px) {
|
|
||||||
.settings-summary-grid,
|
|
||||||
.settings-main-grid,
|
|
||||||
.settings-hero-panel {
|
|
||||||
grid-template-columns: repeat(2, minmax(0, 1fr));
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-sidebar-panel,
|
|
||||||
.settings-workbench {
|
|
||||||
grid-column: 1 / -1;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-sidebar {
|
|
||||||
position: static;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 720px) {
|
|
||||||
.settings-summary-grid,
|
|
||||||
.settings-main-grid,
|
|
||||||
.settings-hero-panel {
|
|
||||||
grid-template-columns: minmax(0, 1fr);
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-panel-header {
|
|
||||||
align-items: flex-start;
|
|
||||||
flex-direction: column;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-category-list {
|
|
||||||
grid-template-columns: minmax(0, 1fr);
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-table-wrap {
|
|
||||||
overflow-x: auto;
|
|
||||||
}
|
|
||||||
|
|
||||||
.settings-table {
|
|
||||||
min-width: 760px;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -1,309 +0,0 @@
|
|||||||
.users-page-body {
|
|
||||||
display: grid;
|
|
||||||
gap: 10px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-hero {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(0, 1.1fr) minmax(300px, .9fr);
|
|
||||||
gap: 10px;
|
|
||||||
padding: 10px;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-hero h2 {
|
|
||||||
margin: 0 0 6px;
|
|
||||||
font-size: 24px;
|
|
||||||
line-height: 24px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-hero p {
|
|
||||||
margin: 0;
|
|
||||||
color: #333333;
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 16px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-hero-actions {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr;
|
|
||||||
gap: 8px;
|
|
||||||
align-content: start;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-summary-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: repeat(4, minmax(0, 1fr));
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-stat-card {
|
|
||||||
padding: 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-stat-card p {
|
|
||||||
margin: 0 0 6px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
text-transform: uppercase;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-stat-card strong {
|
|
||||||
font-size: 24px;
|
|
||||||
line-height: 24px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-stat-card.is-info { background: linear-gradient(180deg, #d7e6fb, #bfd7f8); }
|
|
||||||
.users-stat-card.is-ok { background: linear-gradient(180deg, #dbf4dc, #c3ebc5); }
|
|
||||||
.users-stat-card.is-warning { background: linear-gradient(180deg, #fff1c9, #ffe39f); }
|
|
||||||
.users-stat-card.is-danger { background: linear-gradient(180deg, #ffd8d8, #f1b3b3); }
|
|
||||||
|
|
||||||
.users-main-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(320px, .65fr) minmax(0, 1.35fr);
|
|
||||||
gap: 10px;
|
|
||||||
min-height: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-panel {
|
|
||||||
min-height: 0;
|
|
||||||
display: flex;
|
|
||||||
flex-direction: column;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-panel-header {
|
|
||||||
flex: 0 0 auto;
|
|
||||||
min-height: 34px;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 8px;
|
|
||||||
padding: 6px 8px;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-panel-title {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
min-width: 0;
|
|
||||||
font-weight: bold;
|
|
||||||
font-size: 15px;
|
|
||||||
line-height: 15px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-panel-title span {
|
|
||||||
font-weight: normal;
|
|
||||||
color: #444444;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-panel-tools {
|
|
||||||
display: flex;
|
|
||||||
gap: 6px;
|
|
||||||
flex-wrap: wrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-panel-body {
|
|
||||||
flex: 1 1 auto;
|
|
||||||
min-height: 0;
|
|
||||||
padding: 10px;
|
|
||||||
background: linear-gradient(180deg, rgba(255,255,255,.9), rgba(238,238,238,.58));
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-list-body {
|
|
||||||
display: grid;
|
|
||||||
grid-template-rows: auto minmax(0, 1fr) auto;
|
|
||||||
gap: 8px;
|
|
||||||
overflow: hidden;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-form-grid {
|
|
||||||
display: grid;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-row-two {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: 1fr 1fr;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-field {
|
|
||||||
display: grid;
|
|
||||||
gap: 4px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-input,
|
|
||||||
.users-select {
|
|
||||||
width: 100%;
|
|
||||||
min-width: 0;
|
|
||||||
height: 28px;
|
|
||||||
color: #000000;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 1px solid #808080;
|
|
||||||
border-left: 1px solid #808080;
|
|
||||||
border-right: 1px solid #ffffff;
|
|
||||||
border-bottom: 1px solid #ffffff;
|
|
||||||
padding: 4px 6px;
|
|
||||||
font-family: inherit;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-check {
|
|
||||||
min-height: 20px;
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
gap: 6px;
|
|
||||||
font-size: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-form-actions {
|
|
||||||
display: flex;
|
|
||||||
justify-content: flex-end;
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-action-button,
|
|
||||||
.users-tool-button,
|
|
||||||
.users-page-button {
|
|
||||||
min-width: 70px;
|
|
||||||
height: 24px;
|
|
||||||
padding: 0 8px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-toolbar-grid {
|
|
||||||
display: grid;
|
|
||||||
grid-template-columns: minmax(220px, 1.2fr) repeat(4, minmax(100px, .6fr));
|
|
||||||
gap: 8px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-table-wrap {
|
|
||||||
min-height: 420px;
|
|
||||||
height: 420px;
|
|
||||||
overflow: auto;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 2px solid #606060;
|
|
||||||
border-left: 2px solid #606060;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-table {
|
|
||||||
width: 100%;
|
|
||||||
border-collapse: collapse;
|
|
||||||
table-layout: fixed;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 14px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-table th,
|
|
||||||
.users-table td {
|
|
||||||
padding: 6px;
|
|
||||||
border-bottom: 1px solid #e1e1e1;
|
|
||||||
vertical-align: middle;
|
|
||||||
overflow: hidden;
|
|
||||||
text-overflow: ellipsis;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-table th {
|
|
||||||
position: sticky;
|
|
||||||
top: 0;
|
|
||||||
z-index: 2;
|
|
||||||
text-align: left;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-table tbody tr:nth-child(odd) { background: rgba(255,255,255,.96); }
|
|
||||||
.users-table tbody tr:nth-child(even) { background: rgba(240,244,255,.9); }
|
|
||||||
.users-table tbody tr:hover { background: #d8e5f8; }
|
|
||||||
|
|
||||||
.users-col-check { width: 30px; }
|
|
||||||
.users-col-actions { width: 136px; }
|
|
||||||
|
|
||||||
.users-username {
|
|
||||||
display: grid;
|
|
||||||
gap: 2px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-username strong {
|
|
||||||
font-size: 13px;
|
|
||||||
line-height: 13px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-muted {
|
|
||||||
color: #555555;
|
|
||||||
font-size: 11px;
|
|
||||||
line-height: 11px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-pill {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 18px;
|
|
||||||
padding: 0 6px;
|
|
||||||
color: #222222;
|
|
||||||
background: #f1f1f1;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #b0b0b0;
|
|
||||||
border-bottom: 1px solid #b0b0b0;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-pill.active { background: #def2e0; }
|
|
||||||
.users-pill.pending { background: #fff1c9; }
|
|
||||||
.users-pill.disabled { background: #ffdcdc; }
|
|
||||||
|
|
||||||
.users-row-actions {
|
|
||||||
display: flex;
|
|
||||||
justify-content: flex-end;
|
|
||||||
gap: 4px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-row-button {
|
|
||||||
min-width: 60px;
|
|
||||||
height: 22px;
|
|
||||||
padding: 0 6px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
.users-pagination {
|
|
||||||
display: flex;
|
|
||||||
align-items: center;
|
|
||||||
justify-content: space-between;
|
|
||||||
gap: 8px;
|
|
||||||
font-size: 12px;
|
|
||||||
line-height: 12px;
|
|
||||||
}
|
|
||||||
|
|
||||||
@media (max-width: 1024px) {
|
|
||||||
.users-main-grid,
|
|
||||||
.users-hero {
|
|
||||||
grid-template-columns: 1fr;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
@@ -128,81 +128,3 @@
|
|||||||
font-size: 13px;
|
font-size: 13px;
|
||||||
line-height: 13px;
|
line-height: 13px;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Raised panel - appears to sit above the surface */
|
|
||||||
.raised-panel {
|
|
||||||
background: #dfdfdf;
|
|
||||||
border-top: 1px solid #ffffff;
|
|
||||||
border-left: 1px solid #ffffff;
|
|
||||||
border-right: 1px solid #808080;
|
|
||||||
border-bottom: 1px solid #808080;
|
|
||||||
box-shadow: inset 1px 1px 0 #f7f7f7, inset -1px -1px 0 #b0b0b0;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Sunken panel - appears to be inset into the surface */
|
|
||||||
.sunken-panel {
|
|
||||||
background-color: #ffffff;
|
|
||||||
background-image:
|
|
||||||
linear-gradient(180deg, rgba(255,255,255,.9), rgba(238,238,238,.58)),
|
|
||||||
repeating-linear-gradient(0deg, rgba(0,0,0,.025) 0 1px, transparent 1px 6px);
|
|
||||||
border-top: 2px solid #606060;
|
|
||||||
border-left: 2px solid #606060;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Scroll panel - used for scrollable content areas within windows */
|
|
||||||
.scroll-panel {
|
|
||||||
overflow: auto;
|
|
||||||
background: #ffffff;
|
|
||||||
border-top: 2px solid #606060;
|
|
||||||
border-left: 2px solid #606060;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Meter track for progress bars */
|
|
||||||
.meter-track {
|
|
||||||
display: block;
|
|
||||||
height: 14px;
|
|
||||||
margin-top: 9px;
|
|
||||||
background-color: #ffffff;
|
|
||||||
background-image: repeating-linear-gradient(to right, rgba(0,0,0,.06) 0 1px, transparent 1px 18px);
|
|
||||||
border-top: 2px solid #808080;
|
|
||||||
border-left: 2px solid #808080;
|
|
||||||
border-right: 2px solid #ffffff;
|
|
||||||
border-bottom: 2px solid #ffffff;
|
|
||||||
}
|
|
||||||
|
|
||||||
.meter-bar {
|
|
||||||
display: block;
|
|
||||||
height: 100%;
|
|
||||||
width: var(--meter, 0%);
|
|
||||||
background-color: #000078;
|
|
||||||
background-image: repeating-linear-gradient(to right, rgba(255,255,255,.13) 0 1px, transparent 1px 18px);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Tag styles for status indicators */
|
|
||||||
.tag {
|
|
||||||
display: inline-flex;
|
|
||||||
align-items: center;
|
|
||||||
min-height: 17px;
|
|
||||||
margin: 1px 2px 1px 0;
|
|
||||||
padding: 1px 5px;
|
|
||||||
color: #000000;
|
|
||||||
background: #dfdfdf;
|
|
||||||
border: 1px solid #808080;
|
|
||||||
box-shadow: inset 1px 1px 0 #ffffff;
|
|
||||||
white-space: nowrap;
|
|
||||||
}
|
|
||||||
|
|
||||||
.tag.ok { color: #008000; background: #eeffee; }
|
|
||||||
.tag.info { color: #000078; background: #edf4ff; }
|
|
||||||
.tag.warn { color: #8a6200; background: #ffffcc; }
|
|
||||||
.tag.danger { color: #ffffff; background: #800000; }
|
|
||||||
|
|
||||||
/* Titlebar animation - gradient drift */
|
|
||||||
@keyframes titlebar-drift {
|
|
||||||
from { background-position: 0% 50%; }
|
|
||||||
to { background-position: 100% 50%; }
|
|
||||||
}
|
|
||||||
|
|||||||
16
static/js/account-alerts.js
Normal file
16
static/js/account-alerts.js
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
document.addEventListener("DOMContentLoaded", () => {
|
||||||
|
const title = document.querySelector("[data-alert-detail-title]");
|
||||||
|
const description = document.querySelector("[data-alert-detail-description]");
|
||||||
|
const metadata = document.querySelector("[data-alert-detail-metadata]");
|
||||||
|
|
||||||
|
document.querySelectorAll("[data-alert-row]").forEach((row) => {
|
||||||
|
row.addEventListener("click", (event) => {
|
||||||
|
if (event.target.closest("button, input, a")) return;
|
||||||
|
document.querySelectorAll("[data-alert-row].is-selected").forEach((item) => item.classList.remove("is-selected"));
|
||||||
|
row.classList.add("is-selected");
|
||||||
|
if (title) title.textContent = row.dataset.alertTitle || "";
|
||||||
|
if (description) description.textContent = row.dataset.alertDescription || "";
|
||||||
|
if (metadata) metadata.textContent = row.dataset.alertMetadata || "{}";
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
39
static/js/account-settings.js
Normal file
39
static/js/account-settings.js
Normal file
@@ -0,0 +1,39 @@
|
|||||||
|
document.addEventListener("DOMContentLoaded", () => {
|
||||||
|
const panel = document.querySelector("[data-settings-import-panel]");
|
||||||
|
const toggle = document.querySelector("[data-settings-import-toggle]");
|
||||||
|
const submit = document.querySelector("[data-settings-import-submit]");
|
||||||
|
const input = document.querySelector("[data-settings-import-json]");
|
||||||
|
const csrf = document.querySelector('input[name="csrf_token"]')?.value || "";
|
||||||
|
|
||||||
|
toggle?.addEventListener("click", () => {
|
||||||
|
if (!panel) return;
|
||||||
|
panel.hidden = !panel.hidden;
|
||||||
|
if (!panel.hidden) input?.focus();
|
||||||
|
});
|
||||||
|
|
||||||
|
submit?.addEventListener("click", async () => {
|
||||||
|
const body = input?.value.trim() || "";
|
||||||
|
if (!body) {
|
||||||
|
window.WarpBoxAccountUI.toast("Paste settings JSON first.", "warning");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch("/account/settings/import.json", {
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
"X-CSRF-Token": csrf,
|
||||||
|
},
|
||||||
|
body,
|
||||||
|
});
|
||||||
|
|
||||||
|
const payload = await response.json().catch(() => ({}));
|
||||||
|
if (!response.ok) {
|
||||||
|
window.WarpBoxAccountUI.toast(payload.error || "Settings import failed.", "error");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
window.WarpBoxAccountUI.toast(`Imported ${payload.applied || 0} settings.`, "success");
|
||||||
|
window.setTimeout(() => window.location.reload(), 700);
|
||||||
|
});
|
||||||
|
});
|
||||||
258
static/js/account-ui.js
Normal file
258
static/js/account-ui.js
Normal file
@@ -0,0 +1,258 @@
|
|||||||
|
window.WarpBoxAccountUI = (() => {
|
||||||
|
let toastTimer = null;
|
||||||
|
let activeConfirmResolve = null;
|
||||||
|
|
||||||
|
function initStickyTaskbar(options = {}) {
|
||||||
|
const taskbar = options.taskbar || document.querySelector(".top-taskbar");
|
||||||
|
if (!taskbar) return;
|
||||||
|
|
||||||
|
const update = () => {
|
||||||
|
taskbar.classList.toggle("is-scrolled", window.scrollY > 2);
|
||||||
|
};
|
||||||
|
|
||||||
|
update();
|
||||||
|
window.addEventListener("scroll", update, { passive: true });
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeMenus(root = document) {
|
||||||
|
root.querySelectorAll(".menu-item.is-open").forEach((item) => {
|
||||||
|
item.classList.remove("is-open");
|
||||||
|
item.querySelector(".menu-button")?.setAttribute("aria-expanded", "false");
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function openMenu(item) {
|
||||||
|
if (!item) return;
|
||||||
|
closeMenus(item.closest(".menu-bar") || document);
|
||||||
|
item.classList.add("is-open");
|
||||||
|
item.querySelector(".menu-button")?.setAttribute("aria-expanded", "true");
|
||||||
|
}
|
||||||
|
|
||||||
|
function initMenus(options = {}) {
|
||||||
|
const root = options.root || document;
|
||||||
|
root.addEventListener("click", (event) => {
|
||||||
|
const button = event.target.closest(".menu-button");
|
||||||
|
if (button) {
|
||||||
|
const item = button.closest(".menu-item");
|
||||||
|
const isOpen = item?.classList.contains("is-open");
|
||||||
|
closeMenus(root);
|
||||||
|
if (!isOpen) openMenu(item);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!event.target.closest(".menu-item")) {
|
||||||
|
closeMenus(root);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
root.querySelectorAll(".menu-item").forEach((item) => {
|
||||||
|
item.addEventListener("mouseenter", () => {
|
||||||
|
if (!root.querySelector(".menu-item.is-open")) return;
|
||||||
|
openMenu(item);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
document.addEventListener("keydown", (event) => {
|
||||||
|
if (event.key === "Escape") closeMenus(root);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function toast(message, type = "info", options = {}) {
|
||||||
|
if (window.WarpBoxUI?.toast && !options.forceAccountToast) {
|
||||||
|
window.WarpBoxUI.toast(message, type, options);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const target = options.target || document.querySelector("#account-toast") || document.querySelector("#toast");
|
||||||
|
if (!target) return;
|
||||||
|
|
||||||
|
target.textContent = message;
|
||||||
|
target.classList.remove("toast-info", "toast-success", "toast-warning", "toast-error", "is-visible");
|
||||||
|
target.classList.add(`toast-${type}`, "is-visible");
|
||||||
|
clearTimeout(toastTimer);
|
||||||
|
toastTimer = setTimeout(() => target.classList.remove("is-visible"), options.duration || 2600);
|
||||||
|
}
|
||||||
|
|
||||||
|
function modalElements(options = {}) {
|
||||||
|
return {
|
||||||
|
modal: options.modal || document.querySelector("#account-modal"),
|
||||||
|
title: options.title || document.querySelector("#account-modal-title"),
|
||||||
|
body: options.body || document.querySelector("#account-modal-body"),
|
||||||
|
backdrop: options.backdrop || document.querySelector("#account-modal-backdrop") || document.querySelector("#modal-backdrop"),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
function openModal(titleText, html, options = {}) {
|
||||||
|
const parts = modalElements(options);
|
||||||
|
if (!parts.modal || !parts.title || !parts.body) {
|
||||||
|
if (window.WarpBoxUI?.openPopup) {
|
||||||
|
window.WarpBoxUI.openPopup(titleText, html, options);
|
||||||
|
}
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
parts.title.textContent = titleText;
|
||||||
|
if (options.text) {
|
||||||
|
parts.body.textContent = html;
|
||||||
|
} else {
|
||||||
|
parts.body.innerHTML = html;
|
||||||
|
}
|
||||||
|
parts.modal.classList.add("is-visible");
|
||||||
|
parts.backdrop?.classList.add("is-visible");
|
||||||
|
parts.modal.querySelector("[data-modal-close]")?.focus();
|
||||||
|
}
|
||||||
|
|
||||||
|
function closeModal(options = {}) {
|
||||||
|
const parts = modalElements(options);
|
||||||
|
parts.modal?.classList.remove("is-visible");
|
||||||
|
parts.backdrop?.classList.remove("is-visible");
|
||||||
|
if (window.WarpBoxUI?.closePopup && !parts.modal) {
|
||||||
|
window.WarpBoxUI.closePopup(options);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function confirm(message, options = {}) {
|
||||||
|
const title = options.title || "Confirm action";
|
||||||
|
const confirmLabel = options.confirmLabel || "OK";
|
||||||
|
const cancelLabel = options.cancelLabel || "Cancel";
|
||||||
|
const html = `
|
||||||
|
<p>${htmlEscape(message)}</p>
|
||||||
|
<div class="modal-actions">
|
||||||
|
<button class="win98-button" type="button" data-confirm-cancel>${htmlEscape(cancelLabel)}</button>
|
||||||
|
<button class="win98-button" type="button" data-confirm-ok>${htmlEscape(confirmLabel)}</button>
|
||||||
|
</div>
|
||||||
|
`;
|
||||||
|
|
||||||
|
const parts = modalElements(options);
|
||||||
|
if (!parts.modal) {
|
||||||
|
return Promise.resolve(window.confirm(message));
|
||||||
|
}
|
||||||
|
|
||||||
|
openModal(title, html, options);
|
||||||
|
return new Promise((resolve) => {
|
||||||
|
activeConfirmResolve = resolve;
|
||||||
|
parts.modal.querySelector("[data-confirm-ok]")?.focus();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function finishConfirm(result) {
|
||||||
|
if (activeConfirmResolve) {
|
||||||
|
activeConfirmResolve(result);
|
||||||
|
activeConfirmResolve = null;
|
||||||
|
}
|
||||||
|
closeModal();
|
||||||
|
}
|
||||||
|
|
||||||
|
function setDirtyState(isDirty, options = {}) {
|
||||||
|
const target = options.target || document.querySelector("[data-dirty-chip]");
|
||||||
|
if (!target) return;
|
||||||
|
target.classList.toggle("is-dirty", Boolean(isDirty));
|
||||||
|
target.textContent = isDirty ? (options.dirtyText || "unsaved changes") : (options.cleanText || "");
|
||||||
|
}
|
||||||
|
|
||||||
|
function bindFormDirtyState(form, options = {}) {
|
||||||
|
const targetForm = typeof form === "string" ? document.querySelector(form) : form;
|
||||||
|
if (!targetForm) return;
|
||||||
|
|
||||||
|
let baseline = new FormData(targetForm);
|
||||||
|
const serialize = () => new URLSearchParams(new FormData(targetForm)).toString();
|
||||||
|
let baselineValue = new URLSearchParams(baseline).toString();
|
||||||
|
|
||||||
|
const update = () => setDirtyState(serialize() !== baselineValue, options);
|
||||||
|
targetForm.addEventListener("input", update);
|
||||||
|
targetForm.addEventListener("change", update);
|
||||||
|
targetForm.addEventListener("submit", () => {
|
||||||
|
baseline = new FormData(targetForm);
|
||||||
|
baselineValue = new URLSearchParams(baseline).toString();
|
||||||
|
setDirtyState(false, options);
|
||||||
|
});
|
||||||
|
update();
|
||||||
|
}
|
||||||
|
|
||||||
|
function bindConfirmActions(root = document) {
|
||||||
|
root.addEventListener("click", async (event) => {
|
||||||
|
const ok = event.target.closest("[data-confirm-ok]");
|
||||||
|
if (ok) {
|
||||||
|
finishConfirm(true);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const cancel = event.target.closest("[data-confirm-cancel], [data-modal-close]");
|
||||||
|
if (cancel) {
|
||||||
|
finishConfirm(false);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const action = event.target.closest("[data-confirm]");
|
||||||
|
if (!action) return;
|
||||||
|
if (action.dataset.confirmAccepted === "true") {
|
||||||
|
delete action.dataset.confirmAccepted;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const message = action.getAttribute("data-confirm");
|
||||||
|
if (!message) return;
|
||||||
|
|
||||||
|
event.preventDefault();
|
||||||
|
event.stopPropagation();
|
||||||
|
|
||||||
|
const accepted = await confirm(message, {
|
||||||
|
title: action.getAttribute("data-confirm-title") || "Confirm action",
|
||||||
|
confirmLabel: action.getAttribute("data-confirm-label") || "OK",
|
||||||
|
cancelLabel: action.getAttribute("data-cancel-label") || "Cancel",
|
||||||
|
});
|
||||||
|
if (!accepted) return;
|
||||||
|
|
||||||
|
if (action instanceof HTMLAnchorElement && action.href) {
|
||||||
|
window.location.href = action.href;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const form = action.closest("form");
|
||||||
|
const type = (action.getAttribute("type") || "").toLowerCase();
|
||||||
|
if (form && (type === "submit" || type === "")) {
|
||||||
|
form.requestSubmit(action);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
action.dataset.confirmAccepted = "true";
|
||||||
|
action.click();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function htmlEscape(value) {
|
||||||
|
return String(value || "")
|
||||||
|
.replaceAll("&", "&")
|
||||||
|
.replaceAll("<", "<")
|
||||||
|
.replaceAll(">", ">")
|
||||||
|
.replaceAll('"', """)
|
||||||
|
.replaceAll("'", "'");
|
||||||
|
}
|
||||||
|
|
||||||
|
function init(root = document) {
|
||||||
|
initStickyTaskbar();
|
||||||
|
initMenus({ root });
|
||||||
|
bindConfirmActions(root);
|
||||||
|
document.querySelector("#account-modal-backdrop")?.addEventListener("click", () => closeModal());
|
||||||
|
document.addEventListener("keydown", (event) => {
|
||||||
|
if (event.key === "Escape") closeModal();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
init,
|
||||||
|
initStickyTaskbar,
|
||||||
|
initMenus,
|
||||||
|
toast,
|
||||||
|
confirm,
|
||||||
|
openModal,
|
||||||
|
closeModal,
|
||||||
|
setDirtyState,
|
||||||
|
bindFormDirtyState,
|
||||||
|
closeMenus,
|
||||||
|
};
|
||||||
|
})();
|
||||||
|
|
||||||
|
document.addEventListener("DOMContentLoaded", () => {
|
||||||
|
window.WarpBoxAccountUI.init();
|
||||||
|
});
|
||||||
@@ -1,106 +0,0 @@
|
|||||||
(() => {
|
|
||||||
const menuController = window.WarpBoxUI?.bindMenuBar?.() || { close() {} };
|
|
||||||
const dataNode = document.getElementById("activity-data");
|
|
||||||
const body = document.getElementById("activity-body");
|
|
||||||
const searchInput = document.getElementById("activity-search");
|
|
||||||
const severityFilter = document.getElementById("activity-severity");
|
|
||||||
const kindFilter = document.getElementById("activity-kind");
|
|
||||||
const statusLeft = document.getElementById("activity-status-left");
|
|
||||||
const toast = document.getElementById("toast");
|
|
||||||
|
|
||||||
if (!dataNode || !body || !searchInput) return;
|
|
||||||
const events = parseData();
|
|
||||||
const initialQuery = new URLSearchParams(window.location.search).get("q");
|
|
||||||
if (initialQuery) searchInput.value = initialQuery;
|
|
||||||
|
|
||||||
function parseData() {
|
|
||||||
try {
|
|
||||||
return JSON.parse(dataNode.textContent || "[]");
|
|
||||||
} catch (_) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function showToast(message, type = "info", duration = 1800) {
|
|
||||||
window.WarpBoxUI?.toast?.(message, type, { target: toast, duration });
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderKindFilter() {
|
|
||||||
const kinds = new Set(events.map((event) => event.kind || "unknown"));
|
|
||||||
Array.from(kinds).sort().forEach((kind) => {
|
|
||||||
const option = document.createElement("option");
|
|
||||||
option.value = kind;
|
|
||||||
option.textContent = kind;
|
|
||||||
kindFilter.appendChild(option);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function createdLabel(value) {
|
|
||||||
const parsed = new Date(value);
|
|
||||||
if (Number.isNaN(parsed.getTime())) return "-";
|
|
||||||
return parsed.toISOString().replace("T", " ").slice(0, 16) + " UTC";
|
|
||||||
}
|
|
||||||
|
|
||||||
function filtered() {
|
|
||||||
const query = searchInput.value.trim().toLowerCase();
|
|
||||||
const severity = severityFilter.value;
|
|
||||||
const kind = kindFilter.value;
|
|
||||||
return events.filter((event) => {
|
|
||||||
const haystack = [event.kind, event.message, event.ip, event.path, event.method].join(" ").toLowerCase();
|
|
||||||
const matchesQuery = !query || haystack.includes(query);
|
|
||||||
const matchesSeverity = severity === "all" || event.severity === severity;
|
|
||||||
const matchesKind = kind === "all" || event.kind === kind;
|
|
||||||
return matchesQuery && matchesSeverity && matchesKind;
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function render() {
|
|
||||||
const rows = filtered();
|
|
||||||
body.innerHTML = "";
|
|
||||||
rows.forEach((event) => {
|
|
||||||
const row = document.createElement("tr");
|
|
||||||
row.innerHTML = `
|
|
||||||
<td>${createdLabel(event.created_at)}</td>
|
|
||||||
<td>${escapeHtml(event.kind || "-")}</td>
|
|
||||||
<td>${escapeHtml(event.severity || "-")}</td>
|
|
||||||
<td>${escapeHtml(event.ip || "-")}</td>
|
|
||||||
<td>${escapeHtml(event.method || "-")}</td>
|
|
||||||
<td title="${escapeHtml(event.path || "-")}">${escapeHtml(event.path || "-")}</td>
|
|
||||||
<td title="${escapeHtml(event.message || "-")}">${escapeHtml(event.message || "-")}</td>
|
|
||||||
`;
|
|
||||||
body.appendChild(row);
|
|
||||||
});
|
|
||||||
statusLeft.textContent = `${rows.length} activity event(s) visible`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function escapeHtml(value) {
|
|
||||||
return window.WarpBoxUI?.htmlEscape?.(value) || String(value ?? "");
|
|
||||||
}
|
|
||||||
|
|
||||||
[searchInput, severityFilter, kindFilter].forEach((element) => {
|
|
||||||
element.addEventListener(element.tagName === "INPUT" ? "input" : "change", render);
|
|
||||||
});
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-command]").forEach((button) => {
|
|
||||||
button.addEventListener("click", () => {
|
|
||||||
menuController.close();
|
|
||||||
if (button.dataset.command === "refresh") {
|
|
||||||
window.location.reload();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (button.dataset.command === "export") {
|
|
||||||
const blob = new Blob([JSON.stringify(filtered(), null, 2)], { type: "application/json;charset=utf-8" });
|
|
||||||
const url = URL.createObjectURL(blob);
|
|
||||||
const anchor = document.createElement("a");
|
|
||||||
anchor.href = url;
|
|
||||||
anchor.download = `warpbox-activity-${new Date().toISOString().replaceAll(":", "-")}.json`;
|
|
||||||
anchor.click();
|
|
||||||
URL.revokeObjectURL(url);
|
|
||||||
showToast("Visible activity exported");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
renderKindFilter();
|
|
||||||
render();
|
|
||||||
})();
|
|
||||||
@@ -1,267 +0,0 @@
|
|||||||
(() => {
|
|
||||||
const menuController = window.WarpBoxUI?.bindMenuBar?.() || { close() {} };
|
|
||||||
const dataNode = document.getElementById("alerts-data");
|
|
||||||
const alertsBody = document.getElementById("alerts-body");
|
|
||||||
const searchInput = document.getElementById("search-input");
|
|
||||||
const severityFilter = document.getElementById("severity-filter");
|
|
||||||
const statusFilter = document.getElementById("status-filter");
|
|
||||||
const sourceFilter = document.getElementById("source-filter");
|
|
||||||
const sortFilter = document.getElementById("sort-filter");
|
|
||||||
const selectAll = document.getElementById("select-all");
|
|
||||||
const selectedCountEl = document.getElementById("selected-count");
|
|
||||||
const totalPill = document.getElementById("alerts-total-pill");
|
|
||||||
const toast = document.getElementById("toast");
|
|
||||||
|
|
||||||
const detailEls = {
|
|
||||||
title: document.getElementById("detail-title"),
|
|
||||||
severity: document.getElementById("detail-severity"),
|
|
||||||
status: document.getElementById("detail-status"),
|
|
||||||
code: document.getElementById("detail-code"),
|
|
||||||
trace: document.getElementById("detail-trace"),
|
|
||||||
time: document.getElementById("detail-time"),
|
|
||||||
description: document.getElementById("detail-description"),
|
|
||||||
metadata: document.getElementById("detail-metadata")
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!dataNode || !alertsBody) return;
|
|
||||||
|
|
||||||
const state = {
|
|
||||||
alerts: parseData(),
|
|
||||||
selected: new Set(),
|
|
||||||
activeID: null
|
|
||||||
};
|
|
||||||
const initialQuery = new URLSearchParams(window.location.search).get("q");
|
|
||||||
if (initialQuery) searchInput.value = initialQuery;
|
|
||||||
|
|
||||||
function parseData() {
|
|
||||||
try {
|
|
||||||
return JSON.parse(dataNode.textContent || "[]");
|
|
||||||
} catch (_) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function showToast(message, type = "info", duration = 1800) {
|
|
||||||
window.WarpBoxUI?.toast?.(message, type, { target: toast, duration });
|
|
||||||
}
|
|
||||||
|
|
||||||
function createdLabel(value) {
|
|
||||||
const parsed = new Date(value);
|
|
||||||
if (Number.isNaN(parsed.getTime())) return "-";
|
|
||||||
return parsed.toISOString().replace("T", " ").slice(0, 16) + " UTC";
|
|
||||||
}
|
|
||||||
|
|
||||||
function allAlerts() {
|
|
||||||
return state.alerts.slice();
|
|
||||||
}
|
|
||||||
|
|
||||||
function filteredAlerts() {
|
|
||||||
const query = searchInput.value.trim().toLowerCase();
|
|
||||||
const severity = severityFilter.value;
|
|
||||||
const status = statusFilter.value;
|
|
||||||
const group = sourceFilter.value;
|
|
||||||
const rows = allAlerts().filter((alert) => {
|
|
||||||
const haystack = [
|
|
||||||
alert.title,
|
|
||||||
alert.message,
|
|
||||||
alert.code,
|
|
||||||
alert.trace,
|
|
||||||
alert.group
|
|
||||||
].join(" ").toLowerCase();
|
|
||||||
const matchesSearch = !query || haystack.includes(query);
|
|
||||||
const matchesSeverity = severity === "all" || alert.severity === severity;
|
|
||||||
const matchesStatus = status === "all" || alert.status === status;
|
|
||||||
const matchesGroup = group === "all" || alert.group === group;
|
|
||||||
return matchesSearch && matchesSeverity && matchesStatus && matchesGroup;
|
|
||||||
});
|
|
||||||
const order = { high: 3, medium: 2, low: 1 };
|
|
||||||
rows.sort((a, b) => {
|
|
||||||
if (sortFilter.value === "severity") return (order[b.severity] || 0) - (order[a.severity] || 0);
|
|
||||||
if (sortFilter.value === "oldest") return String(a.created_at).localeCompare(String(b.created_at));
|
|
||||||
return String(b.created_at).localeCompare(String(a.created_at));
|
|
||||||
});
|
|
||||||
return rows;
|
|
||||||
}
|
|
||||||
|
|
||||||
function ensureActive(rows) {
|
|
||||||
if (rows.length === 0) {
|
|
||||||
state.activeID = null;
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
const found = rows.find((item) => item.id === state.activeID);
|
|
||||||
if (found) return found;
|
|
||||||
state.activeID = rows[0].id;
|
|
||||||
return rows[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
function render() {
|
|
||||||
const rows = filteredAlerts();
|
|
||||||
alertsBody.innerHTML = "";
|
|
||||||
rows.forEach((alert) => alertsBody.appendChild(buildRow(alert)));
|
|
||||||
const active = ensureActive(rows);
|
|
||||||
if (active) renderDetails(active);
|
|
||||||
renderSummary(rows);
|
|
||||||
syncSelected();
|
|
||||||
syncSelectAll(rows);
|
|
||||||
}
|
|
||||||
|
|
||||||
function buildRow(alert) {
|
|
||||||
const row = document.createElement("tr");
|
|
||||||
if (state.activeID === alert.id) row.classList.add("is-selected");
|
|
||||||
row.innerHTML = `
|
|
||||||
<td><input type="checkbox" class="row-check"${state.selected.has(alert.id) ? " checked" : ""}></td>
|
|
||||||
<td>${escapeHtml(alert.title || "-")}</td>
|
|
||||||
<td><span class="alerts-pill ${escapeHtml(alert.severity || "low")}">${escapeHtml(alert.severity || "low")}</span></td>
|
|
||||||
<td><span class="alerts-pill ${escapeHtml(alert.status || "open")}">${escapeHtml(alert.status || "open")}</span></td>
|
|
||||||
<td>${escapeHtml(alert.code || "-")}</td>
|
|
||||||
<td>${escapeHtml(alert.trace || "-")}</td>
|
|
||||||
<td>${createdLabel(alert.created_at)}</td>
|
|
||||||
<td><button class="win98-button alerts-row-button row-open" type="button">Open</button></td>
|
|
||||||
`;
|
|
||||||
row.addEventListener("click", (event) => {
|
|
||||||
if (event.target.closest("button") || event.target.closest("input")) return;
|
|
||||||
state.activeID = alert.id;
|
|
||||||
render();
|
|
||||||
});
|
|
||||||
row.querySelector(".row-open")?.addEventListener("click", () => {
|
|
||||||
state.activeID = alert.id;
|
|
||||||
render();
|
|
||||||
});
|
|
||||||
row.querySelector(".row-check")?.addEventListener("change", (event) => {
|
|
||||||
if (event.target.checked) state.selected.add(alert.id);
|
|
||||||
else state.selected.delete(alert.id);
|
|
||||||
syncSelected();
|
|
||||||
syncSelectAll(filteredAlerts());
|
|
||||||
});
|
|
||||||
return row;
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderDetails(alert) {
|
|
||||||
detailEls.title.textContent = alert.title || "";
|
|
||||||
detailEls.severity.textContent = alert.severity || "";
|
|
||||||
detailEls.status.textContent = alert.status || "";
|
|
||||||
detailEls.code.textContent = alert.code || "";
|
|
||||||
detailEls.trace.textContent = alert.trace || "";
|
|
||||||
detailEls.time.textContent = createdLabel(alert.created_at);
|
|
||||||
detailEls.description.textContent = alert.message || "";
|
|
||||||
detailEls.metadata.textContent = JSON.stringify(alert.meta || {}, null, 2);
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderSummary(rows) {
|
|
||||||
const open = rows.filter((item) => item.status === "open").length;
|
|
||||||
const high = rows.filter((item) => item.severity === "high" && item.status !== "closed").length;
|
|
||||||
const ack = rows.filter((item) => item.status === "acked").length;
|
|
||||||
const closed = rows.filter((item) => item.status === "closed").length;
|
|
||||||
document.querySelector("[data-open-count]").textContent = String(open);
|
|
||||||
document.querySelector("[data-high-count]").textContent = String(high);
|
|
||||||
document.querySelector("[data-ack-count]").textContent = String(ack);
|
|
||||||
document.querySelector("[data-closed-count]").textContent = String(closed);
|
|
||||||
totalPill.textContent = `${rows.length} alerts`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function syncSelected() {
|
|
||||||
selectedCountEl.textContent = `Selected: ${state.selected.size}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function syncSelectAll(rows) {
|
|
||||||
selectAll.checked = rows.length > 0 && rows.every((alert) => state.selected.has(alert.id));
|
|
||||||
}
|
|
||||||
|
|
||||||
async function postAction(action, ids) {
|
|
||||||
const response = await fetch("/admin/alerts/actions", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({ action, ids })
|
|
||||||
});
|
|
||||||
const payload = await response.json().catch(() => ({}));
|
|
||||||
if (!response.ok) throw new Error(payload.error || "Request failed");
|
|
||||||
state.alerts = payload.alerts || [];
|
|
||||||
}
|
|
||||||
|
|
||||||
async function runAction(action) {
|
|
||||||
const ids = Array.from(state.selected);
|
|
||||||
if (!ids.length && (action === "ack" || action === "close" || action === "delete")) {
|
|
||||||
showToast("Select one or more alerts first", "warning");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (action === "open-only") {
|
|
||||||
statusFilter.value = "open";
|
|
||||||
render();
|
|
||||||
showToast("Showing open alerts only");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (action === "refresh") {
|
|
||||||
window.location.reload();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (action === "copy-meta") {
|
|
||||||
const active = allAlerts().find((item) => item.id === state.activeID);
|
|
||||||
if (active) {
|
|
||||||
navigator.clipboard?.writeText(JSON.stringify(active.meta || {}, null, 2)).catch(() => {});
|
|
||||||
}
|
|
||||||
showToast("Metadata copied");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (action === "export") {
|
|
||||||
const blob = new Blob([JSON.stringify(filteredAlerts(), null, 2)], { type: "application/json;charset=utf-8" });
|
|
||||||
const url = URL.createObjectURL(blob);
|
|
||||||
const anchor = document.createElement("a");
|
|
||||||
anchor.href = url;
|
|
||||||
anchor.download = `warpbox-alerts-${new Date().toISOString().replaceAll(":", "-")}.json`;
|
|
||||||
anchor.click();
|
|
||||||
URL.revokeObjectURL(url);
|
|
||||||
showToast("Visible alerts exported");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (action === "help-codes") {
|
|
||||||
showToast("Codes map to internal security and service traces.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (action === "help-meta") {
|
|
||||||
showToast("Metadata shows extra context for each alert.");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
await postAction(action, ids);
|
|
||||||
state.selected.clear();
|
|
||||||
render();
|
|
||||||
showToast(`Action complete: ${action}`, "success");
|
|
||||||
}
|
|
||||||
|
|
||||||
function escapeHtml(value) {
|
|
||||||
return window.WarpBoxUI?.htmlEscape?.(value) || String(value ?? "");
|
|
||||||
}
|
|
||||||
|
|
||||||
[searchInput, severityFilter, statusFilter, sourceFilter, sortFilter].forEach((control) => {
|
|
||||||
control.addEventListener(control.tagName === "INPUT" ? "input" : "change", render);
|
|
||||||
});
|
|
||||||
|
|
||||||
selectAll?.addEventListener("change", () => {
|
|
||||||
const rows = filteredAlerts();
|
|
||||||
rows.forEach((alert) => {
|
|
||||||
if (selectAll.checked) state.selected.add(alert.id);
|
|
||||||
else state.selected.delete(alert.id);
|
|
||||||
});
|
|
||||||
render();
|
|
||||||
});
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-command]").forEach((button) => {
|
|
||||||
button.addEventListener("click", async () => {
|
|
||||||
menuController.close();
|
|
||||||
try {
|
|
||||||
await runAction(button.dataset.command);
|
|
||||||
} catch (error) {
|
|
||||||
showToast(error.message, "error", 3200);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.addEventListener("keydown", async (event) => {
|
|
||||||
if (event.key === "Escape") menuController.close();
|
|
||||||
if (event.key === "F5") {
|
|
||||||
event.preventDefault();
|
|
||||||
await runAction("refresh");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
render();
|
|
||||||
})();
|
|
||||||
@@ -1,559 +0,0 @@
|
|||||||
(() => {
|
|
||||||
const menuController = window.WarpBoxUI?.bindMenuBar?.() || {
|
|
||||||
close() {
|
|
||||||
document.querySelectorAll(".menu-item.is-open").forEach((item) => {
|
|
||||||
item.classList.remove("is-open");
|
|
||||||
item.querySelector(".menu-button")?.setAttribute("aria-expanded", "false");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
const toastTarget = document.getElementById("toast");
|
|
||||||
const dataNode = document.getElementById("boxes-data");
|
|
||||||
const tableBody = document.getElementById("boxes-table-body");
|
|
||||||
const emptyState = document.getElementById("boxes-empty-state");
|
|
||||||
const searchInput = document.getElementById("boxes-search");
|
|
||||||
const statusFilter = document.getElementById("boxes-status-filter");
|
|
||||||
const flagFilter = document.getElementById("boxes-flag-filter");
|
|
||||||
const sortFilter = document.getElementById("boxes-sort");
|
|
||||||
const pageSizeFilter = document.getElementById("boxes-page-size");
|
|
||||||
const selectAll = document.getElementById("boxes-select-all");
|
|
||||||
const prevPageButton = document.getElementById("boxes-prev-page");
|
|
||||||
const nextPageButton = document.getElementById("boxes-next-page");
|
|
||||||
const pageLabel = document.getElementById("boxes-page-label");
|
|
||||||
const rangeLabel = document.getElementById("boxes-range-label");
|
|
||||||
const selectedLabel = document.getElementById("boxes-selected-label");
|
|
||||||
const footerSummary = document.getElementById("boxes-footer-summary");
|
|
||||||
const detailFileList = document.getElementById("detail-file-list");
|
|
||||||
|
|
||||||
if (!dataNode || !tableBody || !searchInput || !detailFileList) return;
|
|
||||||
|
|
||||||
const statEls = {
|
|
||||||
total: document.querySelector("[data-stat-total]"),
|
|
||||||
ready: document.querySelector("[data-stat-ready]"),
|
|
||||||
uploading: document.querySelector("[data-stat-uploading]"),
|
|
||||||
expired: document.querySelector("[data-stat-expired]")
|
|
||||||
};
|
|
||||||
|
|
||||||
const detailEls = {
|
|
||||||
boxId: document.getElementById("detail-box-id"),
|
|
||||||
status: document.getElementById("detail-status"),
|
|
||||||
created: document.getElementById("detail-created"),
|
|
||||||
expires: document.getElementById("detail-expires"),
|
|
||||||
retention: document.getElementById("detail-retention"),
|
|
||||||
files: document.getElementById("detail-files"),
|
|
||||||
size: document.getElementById("detail-size"),
|
|
||||||
flags: document.getElementById("detail-flags"),
|
|
||||||
open: document.getElementById("detail-open"),
|
|
||||||
zip: document.getElementById("detail-zip")
|
|
||||||
};
|
|
||||||
|
|
||||||
function showToast(message, type = "info", duration = 2200) {
|
|
||||||
if (window.WarpBoxUI) {
|
|
||||||
window.WarpBoxUI.toast(message, type, { target: toastTarget, duration });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!toastTarget) return;
|
|
||||||
toastTarget.textContent = message;
|
|
||||||
toastTarget.classList.add("is-visible");
|
|
||||||
window.setTimeout(() => toastTarget.classList.remove("is-visible"), duration);
|
|
||||||
}
|
|
||||||
|
|
||||||
function parseData() {
|
|
||||||
try {
|
|
||||||
return JSON.parse(dataNode.textContent || "[]");
|
|
||||||
} catch (_) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const state = {
|
|
||||||
boxes: parseData(),
|
|
||||||
selected: new Set(),
|
|
||||||
activeId: null,
|
|
||||||
page: 1
|
|
||||||
};
|
|
||||||
|
|
||||||
function pageSize() {
|
|
||||||
return Number(pageSizeFilter.value || 10);
|
|
||||||
}
|
|
||||||
|
|
||||||
function allBoxes() {
|
|
||||||
return state.boxes.slice();
|
|
||||||
}
|
|
||||||
|
|
||||||
function sortBoxes(boxes) {
|
|
||||||
const sorted = boxes.slice();
|
|
||||||
switch (sortFilter.value) {
|
|
||||||
case "name":
|
|
||||||
sorted.sort((a, b) => a.id.localeCompare(b.id));
|
|
||||||
break;
|
|
||||||
case "largest":
|
|
||||||
sorted.sort((a, b) => compareSizeLabel(a.total_size_label, b.total_size_label));
|
|
||||||
break;
|
|
||||||
case "expires":
|
|
||||||
sorted.sort((a, b) => compareExpiry(a.expires_at_iso, b.expires_at_iso));
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
sorted.sort((a, b) => (b.created_at_iso || "").localeCompare(a.created_at_iso || ""));
|
|
||||||
}
|
|
||||||
return sorted;
|
|
||||||
}
|
|
||||||
|
|
||||||
function compareSizeLabel(left, right) {
|
|
||||||
return sizeLabelToBytes(right) - sizeLabelToBytes(left);
|
|
||||||
}
|
|
||||||
|
|
||||||
function sizeLabelToBytes(label) {
|
|
||||||
const match = String(label || "").trim().match(/^([\d.]+)\s*([KMGT]?i?B|B)$/i);
|
|
||||||
if (!match) return 0;
|
|
||||||
const value = Number(match[1]);
|
|
||||||
const unit = match[2].toUpperCase();
|
|
||||||
const map = { B: 1, KIB: 1024, MIB: 1024 ** 2, GIB: 1024 ** 3, TIB: 1024 ** 4 };
|
|
||||||
return value * (map[unit] || 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
function compareExpiry(left, right) {
|
|
||||||
if (!left && !right) return 0;
|
|
||||||
if (!left) return 1;
|
|
||||||
if (!right) return -1;
|
|
||||||
return left.localeCompare(right);
|
|
||||||
}
|
|
||||||
|
|
||||||
function filteredBoxes() {
|
|
||||||
const query = searchInput.value.trim().toLowerCase();
|
|
||||||
const status = statusFilter.value;
|
|
||||||
const flag = flagFilter.value;
|
|
||||||
|
|
||||||
return sortBoxes(allBoxes().filter((box) => {
|
|
||||||
const matchesSearch = !query || String(box.search_text || "").includes(query);
|
|
||||||
const matchesStatus = status === "all" || box.status === status;
|
|
||||||
const matchesFlag = flag === "all" || (box.flags || []).includes(flag);
|
|
||||||
return matchesSearch && matchesStatus && matchesFlag;
|
|
||||||
}));
|
|
||||||
}
|
|
||||||
|
|
||||||
function pagedBoxes(boxes) {
|
|
||||||
const size = pageSize();
|
|
||||||
const pages = Math.max(1, Math.ceil(boxes.length / size));
|
|
||||||
if (state.page > pages) state.page = pages;
|
|
||||||
if (state.page < 1) state.page = 1;
|
|
||||||
const start = (state.page - 1) * size;
|
|
||||||
return {
|
|
||||||
items: boxes.slice(start, start + size),
|
|
||||||
start,
|
|
||||||
pages
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
function selectedBoxes() {
|
|
||||||
return allBoxes().filter((box) => state.selected.has(box.id));
|
|
||||||
}
|
|
||||||
|
|
||||||
function currentActiveBox() {
|
|
||||||
const boxes = allBoxes();
|
|
||||||
return boxes.find((box) => box.id === state.activeId) || null;
|
|
||||||
}
|
|
||||||
|
|
||||||
function ensureActiveBox(filtered) {
|
|
||||||
if (filtered.length === 0) {
|
|
||||||
state.activeId = null;
|
|
||||||
return null;
|
|
||||||
}
|
|
||||||
if (!filtered.some((box) => box.id === state.activeId)) {
|
|
||||||
state.activeId = filtered[0].id;
|
|
||||||
}
|
|
||||||
return filtered.find((box) => box.id === state.activeId) || filtered[0];
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderSummary(filtered) {
|
|
||||||
const total = filtered.length;
|
|
||||||
const ready = filtered.filter((box) => box.status === "ready").length;
|
|
||||||
const uploading = filtered.filter((box) => box.status === "uploading").length;
|
|
||||||
const expired = filtered.filter((box) => box.status === "expired" || box.status === "consumed").length;
|
|
||||||
statEls.total.textContent = String(total);
|
|
||||||
statEls.ready.textContent = String(ready);
|
|
||||||
statEls.uploading.textContent = String(uploading);
|
|
||||||
statEls.expired.textContent = String(expired);
|
|
||||||
footerSummary.textContent = `${allBoxes().length} boxes loaded`;
|
|
||||||
selectedLabel.textContent = `Selected: ${state.selected.size}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderTable() {
|
|
||||||
const filtered = filteredBoxes();
|
|
||||||
const active = ensureActiveBox(filtered);
|
|
||||||
const page = pagedBoxes(filtered);
|
|
||||||
|
|
||||||
tableBody.innerHTML = "";
|
|
||||||
page.items.forEach((box) => tableBody.appendChild(buildRow(box)));
|
|
||||||
emptyState.hidden = page.items.length !== 0;
|
|
||||||
|
|
||||||
const startIndex = filtered.length ? page.start + 1 : 0;
|
|
||||||
const endIndex = page.start + page.items.length;
|
|
||||||
rangeLabel.textContent = `Showing ${startIndex}-${endIndex} of ${filtered.length}`;
|
|
||||||
pageLabel.textContent = `Page ${state.page} / ${page.pages}`;
|
|
||||||
prevPageButton.disabled = state.page <= 1;
|
|
||||||
nextPageButton.disabled = state.page >= page.pages;
|
|
||||||
selectAll.checked = page.items.length > 0 && page.items.every((box) => state.selected.has(box.id));
|
|
||||||
|
|
||||||
renderSummary(filtered);
|
|
||||||
renderDetails(active);
|
|
||||||
}
|
|
||||||
|
|
||||||
function buildRow(box) {
|
|
||||||
const row = document.createElement("tr");
|
|
||||||
if (box.id === state.activeId) row.classList.add("is-selected");
|
|
||||||
|
|
||||||
row.innerHTML = `
|
|
||||||
<td><input type="checkbox" class="boxes-row-check"${state.selected.has(box.id) ? " checked" : ""}></td>
|
|
||||||
<td title="${escapeAttr(box.id)}">${box.id}</td>
|
|
||||||
<td><span class="boxes-status-pill ${box.status}">${box.status_label}</span></td>
|
|
||||||
<td>${box.complete_files}/${box.file_count}</td>
|
|
||||||
<td>${box.total_size_label}</td>
|
|
||||||
<td>${box.retention_label || "Not set"}</td>
|
|
||||||
<td>${box.expires_at_label || "Not set"}</td>
|
|
||||||
<td><div class="boxes-flags-cell">${renderFlags(box.flags)}</div></td>
|
|
||||||
<td><div class="boxes-action-cell">${renderRowActions(box)}</div></td>
|
|
||||||
`;
|
|
||||||
|
|
||||||
row.addEventListener("click", (event) => {
|
|
||||||
if (event.target.closest("button") || event.target.closest("a") || event.target.closest("input")) return;
|
|
||||||
state.activeId = box.id;
|
|
||||||
renderTable();
|
|
||||||
});
|
|
||||||
|
|
||||||
row.querySelector(".boxes-row-check")?.addEventListener("change", (event) => {
|
|
||||||
if (event.target.checked) {
|
|
||||||
state.selected.add(box.id);
|
|
||||||
} else {
|
|
||||||
state.selected.delete(box.id);
|
|
||||||
}
|
|
||||||
selectedLabel.textContent = `Selected: ${state.selected.size}`;
|
|
||||||
syncSelectAllForPage();
|
|
||||||
});
|
|
||||||
|
|
||||||
row.querySelector('[data-row-action="focus"]')?.addEventListener("click", () => {
|
|
||||||
state.activeId = box.id;
|
|
||||||
renderTable();
|
|
||||||
});
|
|
||||||
|
|
||||||
return row;
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderFlags(flags) {
|
|
||||||
if (!flags || !flags.length) return '<span class="boxes-flag">none</span>';
|
|
||||||
return flags.map((flag) => `<span class="boxes-flag">${escapeHtml(flag)}</span>`).join("");
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderRowActions(box) {
|
|
||||||
const parts = [
|
|
||||||
`<a class="win98-button boxes-row-button" href="${escapeAttr(box.open_url)}" target="_blank" rel="noreferrer">Open</a>`,
|
|
||||||
`<button class="win98-button boxes-row-button" type="button" data-row-action="focus">View</button>`
|
|
||||||
];
|
|
||||||
if (box.zip_available && box.zip_url) {
|
|
||||||
parts.push(`<a class="win98-button boxes-row-button" href="${escapeAttr(box.zip_url)}" target="_blank" rel="noreferrer">ZIP</a>`);
|
|
||||||
}
|
|
||||||
return parts.join("");
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderDetails(box) {
|
|
||||||
if (!box) {
|
|
||||||
detailEls.boxId.textContent = "-";
|
|
||||||
detailEls.status.textContent = "-";
|
|
||||||
detailEls.created.textContent = "-";
|
|
||||||
detailEls.expires.textContent = "-";
|
|
||||||
detailEls.retention.textContent = "-";
|
|
||||||
detailEls.files.textContent = "-";
|
|
||||||
detailEls.size.textContent = "-";
|
|
||||||
detailEls.flags.textContent = "-";
|
|
||||||
detailEls.open.href = "#";
|
|
||||||
detailEls.zip.href = "#";
|
|
||||||
detailEls.zip.setAttribute("aria-disabled", "true");
|
|
||||||
detailFileList.innerHTML = '<div class="boxes-file-card">No box selected.</div>';
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
detailEls.boxId.textContent = box.id;
|
|
||||||
detailEls.status.textContent = box.status_label;
|
|
||||||
detailEls.created.textContent = box.created_at_label || "Not set";
|
|
||||||
detailEls.expires.textContent = box.expires_at_label || "Not set";
|
|
||||||
detailEls.retention.textContent = box.retention_label || "Not set";
|
|
||||||
detailEls.files.textContent = `${box.complete_files}/${box.file_count} complete`;
|
|
||||||
detailEls.size.textContent = box.total_size_label;
|
|
||||||
detailEls.flags.textContent = (box.flags || []).join(", ") || "none";
|
|
||||||
detailEls.open.href = box.open_url || "#";
|
|
||||||
|
|
||||||
if (box.zip_available && box.zip_url) {
|
|
||||||
detailEls.zip.href = box.zip_url;
|
|
||||||
detailEls.zip.removeAttribute("aria-disabled");
|
|
||||||
detailEls.zip.style.pointerEvents = "";
|
|
||||||
detailEls.zip.style.opacity = "";
|
|
||||||
} else {
|
|
||||||
detailEls.zip.href = "#";
|
|
||||||
detailEls.zip.setAttribute("aria-disabled", "true");
|
|
||||||
detailEls.zip.style.pointerEvents = "none";
|
|
||||||
detailEls.zip.style.opacity = ".55";
|
|
||||||
}
|
|
||||||
|
|
||||||
renderFiles(box.files || []);
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderFiles(files) {
|
|
||||||
if (!files.length) {
|
|
||||||
detailFileList.innerHTML = '<div class="boxes-file-card">No file inventory available for this box.</div>';
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
detailFileList.innerHTML = files.map((file) => `
|
|
||||||
<div class="boxes-file-card">
|
|
||||||
<div class="boxes-file-row">
|
|
||||||
<div class="boxes-file-name" title="${escapeAttr(file.name)}">${escapeHtml(file.name)}</div>
|
|
||||||
<span class="boxes-status-pill ${escapeAttr(file.status || "legacy")}">${escapeHtml(file.status_label || file.status || "Unknown")}</span>
|
|
||||||
</div>
|
|
||||||
<div class="boxes-file-meta">
|
|
||||||
<span>${escapeHtml(file.size_label || "0 B")}</span>
|
|
||||||
<span>${escapeHtml(file.mime_type || "application/octet-stream")}</span>
|
|
||||||
${file.is_complete && file.download_path ? `<a class="boxes-file-link" href="${escapeAttr(file.download_path)}" target="_blank" rel="noreferrer">download</a>` : "<span>pending</span>"}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
`).join("");
|
|
||||||
}
|
|
||||||
|
|
||||||
function syncSelectAllForPage() {
|
|
||||||
const filtered = filteredBoxes();
|
|
||||||
const page = pagedBoxes(filtered);
|
|
||||||
selectAll.checked = page.items.length > 0 && page.items.every((box) => state.selected.has(box.id));
|
|
||||||
selectedLabel.textContent = `Selected: ${state.selected.size}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function clearFilters() {
|
|
||||||
searchInput.value = "";
|
|
||||||
statusFilter.value = "all";
|
|
||||||
flagFilter.value = "all";
|
|
||||||
sortFilter.value = "newest";
|
|
||||||
pageSizeFilter.value = "10";
|
|
||||||
state.page = 1;
|
|
||||||
renderTable();
|
|
||||||
}
|
|
||||||
|
|
||||||
async function runBulkAction(action, ids, deltaSeconds = 0) {
|
|
||||||
if (!ids.length) {
|
|
||||||
showToast("Select one or more boxes first", "warning");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const response = await fetch("/admin/boxes/actions", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({ action, box_ids: ids, delta_seconds: deltaSeconds })
|
|
||||||
});
|
|
||||||
const payload = await response.json();
|
|
||||||
if (!response.ok) {
|
|
||||||
const message = payload.error || payload.message || "Action failed";
|
|
||||||
const warning = Array.isArray(payload.warnings) && payload.warnings.length ? ` (${payload.warnings[0]})` : "";
|
|
||||||
showToast(`${message}${warning}`, "error", 3200);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
state.boxes = Array.isArray(payload.boxes) ? payload.boxes : state.boxes;
|
|
||||||
state.selected.clear();
|
|
||||||
if (state.activeId && !state.boxes.some((box) => box.id === state.activeId)) {
|
|
||||||
state.activeId = null;
|
|
||||||
}
|
|
||||||
renderTable();
|
|
||||||
|
|
||||||
let message = payload.message || "Action complete";
|
|
||||||
if (Array.isArray(payload.warnings) && payload.warnings.length) {
|
|
||||||
message += ` (${payload.warnings.length} warning${payload.warnings.length === 1 ? "" : "s"})`;
|
|
||||||
}
|
|
||||||
showToast(message, Array.isArray(payload.warnings) && payload.warnings.length ? "warning" : "success", 2800);
|
|
||||||
} catch (_) {
|
|
||||||
showToast("Network error while updating boxes", "error", 3200);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function runCleanupAction() {
|
|
||||||
try {
|
|
||||||
const response = await fetch("/admin/boxes/actions", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({ action: "cleanup_expired", box_ids: [] })
|
|
||||||
});
|
|
||||||
const payload = await response.json();
|
|
||||||
if (!response.ok) {
|
|
||||||
const message = payload.error || payload.message || "Cleanup failed";
|
|
||||||
showToast(message, "error", 3200);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
state.boxes = Array.isArray(payload.boxes) ? payload.boxes : state.boxes;
|
|
||||||
state.selected.clear();
|
|
||||||
if (state.activeId && !state.boxes.some((box) => box.id === state.activeId)) {
|
|
||||||
state.activeId = null;
|
|
||||||
}
|
|
||||||
renderTable();
|
|
||||||
let message = payload.message || "Expired cleanup completed";
|
|
||||||
if (Array.isArray(payload.warnings) && payload.warnings.length) {
|
|
||||||
message += ` (${payload.warnings.length} warning${payload.warnings.length === 1 ? "" : "s"})`;
|
|
||||||
}
|
|
||||||
showToast(message, Array.isArray(payload.warnings) && payload.warnings.length ? "warning" : "success", 3200);
|
|
||||||
} catch (_) {
|
|
||||||
showToast("Network error while running cleanup", "error", 3200);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function selectedIDsOrActive() {
|
|
||||||
if (state.selected.size) return Array.from(state.selected);
|
|
||||||
const active = currentActiveBox();
|
|
||||||
return active ? [active.id] : [];
|
|
||||||
}
|
|
||||||
|
|
||||||
async function runCommand(command) {
|
|
||||||
switch (command) {
|
|
||||||
case "refresh":
|
|
||||||
window.location.reload();
|
|
||||||
return;
|
|
||||||
case "export":
|
|
||||||
exportVisibleCSV();
|
|
||||||
showToast("Visible boxes exported");
|
|
||||||
return;
|
|
||||||
case "status-ready":
|
|
||||||
statusFilter.value = "ready";
|
|
||||||
state.page = 1;
|
|
||||||
renderTable();
|
|
||||||
return;
|
|
||||||
case "status-expired":
|
|
||||||
statusFilter.value = "expired";
|
|
||||||
state.page = 1;
|
|
||||||
renderTable();
|
|
||||||
return;
|
|
||||||
case "clear-filters":
|
|
||||||
clearFilters();
|
|
||||||
showToast("Filters cleared");
|
|
||||||
return;
|
|
||||||
case "expire":
|
|
||||||
case "active-expire":
|
|
||||||
await runBulkAction("expire", selectedIDsOrActive());
|
|
||||||
return;
|
|
||||||
case "extend-day":
|
|
||||||
case "active-extend-day":
|
|
||||||
await runBulkAction("bump", selectedIDsOrActive(), 24 * 60 * 60);
|
|
||||||
return;
|
|
||||||
case "extend-week":
|
|
||||||
case "active-extend-week":
|
|
||||||
await runBulkAction("bump", selectedIDsOrActive(), 7 * 24 * 60 * 60);
|
|
||||||
return;
|
|
||||||
case "delete":
|
|
||||||
case "active-delete":
|
|
||||||
if (!window.confirm("Delete selected boxes? This removes stored files.")) return;
|
|
||||||
await runBulkAction("delete", selectedIDsOrActive());
|
|
||||||
return;
|
|
||||||
case "cleanup-expired":
|
|
||||||
if (!window.confirm("Run cleanup for expired boxes now?")) return;
|
|
||||||
await runCleanupAction();
|
|
||||||
return;
|
|
||||||
case "help-scope":
|
|
||||||
showToast("Ownership filter waits for account + box owner data in backend", "info", 3400);
|
|
||||||
return;
|
|
||||||
case "help-flags":
|
|
||||||
showToast("Flags: protected, one-time, zip off, legacy, consumed", "info", 3200);
|
|
||||||
return;
|
|
||||||
default:
|
|
||||||
showToast(`Unknown command: ${command}`, "warning");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function exportVisibleCSV() {
|
|
||||||
const rows = filteredBoxes().map((box) => ([
|
|
||||||
box.id,
|
|
||||||
box.status_label,
|
|
||||||
box.file_count,
|
|
||||||
box.total_size_label,
|
|
||||||
box.retention_label,
|
|
||||||
box.expires_at_label,
|
|
||||||
(box.flags || []).join("|")
|
|
||||||
]));
|
|
||||||
const csv = [
|
|
||||||
["box_id", "status", "files", "size", "retention", "expires", "flags"],
|
|
||||||
...rows
|
|
||||||
].map((row) => row.map(csvCell).join(",")).join("\n");
|
|
||||||
|
|
||||||
const blob = new Blob([csv], { type: "text/csv;charset=utf-8" });
|
|
||||||
const url = URL.createObjectURL(blob);
|
|
||||||
const anchor = document.createElement("a");
|
|
||||||
anchor.href = url;
|
|
||||||
anchor.download = "warpbox-boxes.csv";
|
|
||||||
anchor.click();
|
|
||||||
URL.revokeObjectURL(url);
|
|
||||||
}
|
|
||||||
|
|
||||||
function csvCell(value) {
|
|
||||||
const text = String(value ?? "");
|
|
||||||
if (/[",\n]/.test(text)) return `"${text.replaceAll('"', '""')}"`;
|
|
||||||
return text;
|
|
||||||
}
|
|
||||||
|
|
||||||
function escapeHtml(value) {
|
|
||||||
return String(value ?? "")
|
|
||||||
.replaceAll("&", "&")
|
|
||||||
.replaceAll("<", "<")
|
|
||||||
.replaceAll(">", ">")
|
|
||||||
.replaceAll('"', """);
|
|
||||||
}
|
|
||||||
|
|
||||||
function escapeAttr(value) {
|
|
||||||
return escapeHtml(value).replaceAll("'", "'");
|
|
||||||
}
|
|
||||||
|
|
||||||
[searchInput, statusFilter, flagFilter, sortFilter].forEach((control) => {
|
|
||||||
control.addEventListener(control.tagName === "INPUT" ? "input" : "change", () => {
|
|
||||||
state.page = 1;
|
|
||||||
renderTable();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
pageSizeFilter.addEventListener("change", () => {
|
|
||||||
state.page = 1;
|
|
||||||
renderTable();
|
|
||||||
});
|
|
||||||
|
|
||||||
selectAll?.addEventListener("change", () => {
|
|
||||||
const filtered = filteredBoxes();
|
|
||||||
const page = pagedBoxes(filtered);
|
|
||||||
page.items.forEach((box) => {
|
|
||||||
if (selectAll.checked) state.selected.add(box.id);
|
|
||||||
else state.selected.delete(box.id);
|
|
||||||
});
|
|
||||||
renderTable();
|
|
||||||
});
|
|
||||||
|
|
||||||
prevPageButton?.addEventListener("click", () => {
|
|
||||||
state.page -= 1;
|
|
||||||
renderTable();
|
|
||||||
});
|
|
||||||
|
|
||||||
nextPageButton?.addEventListener("click", () => {
|
|
||||||
state.page += 1;
|
|
||||||
renderTable();
|
|
||||||
});
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-command]").forEach((button) => {
|
|
||||||
button.addEventListener("click", async () => {
|
|
||||||
menuController.close();
|
|
||||||
await runCommand(button.dataset.command);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.addEventListener("keydown", async (event) => {
|
|
||||||
if (event.key === "Escape") menuController.close();
|
|
||||||
if (event.key === "F5") {
|
|
||||||
event.preventDefault();
|
|
||||||
await runCommand("refresh");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (state.boxes.length > 0) {
|
|
||||||
state.activeId = state.boxes[0].id;
|
|
||||||
}
|
|
||||||
renderTable();
|
|
||||||
})();
|
|
||||||
@@ -1,201 +0,0 @@
|
|||||||
(() => {
|
|
||||||
const menuController = window.WarpBoxUI?.bindMenuBar?.() || {
|
|
||||||
close() {
|
|
||||||
document.querySelectorAll(".menu-item.is-open").forEach((item) => {
|
|
||||||
item.classList.remove("is-open");
|
|
||||||
item.querySelector(".menu-button")?.setAttribute("aria-expanded", "false");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
};
|
|
||||||
const toast = document.getElementById("toast");
|
|
||||||
const statusText = document.getElementById("statusText");
|
|
||||||
const modal = document.querySelector("[data-alert-modal]");
|
|
||||||
const backdrop = document.querySelector("[data-modal-backdrop]");
|
|
||||||
const modalTitle = document.getElementById("modalTitle");
|
|
||||||
const modalMeta = document.getElementById("modalMeta");
|
|
||||||
const alertCountValue = document.getElementById("alertCountValue");
|
|
||||||
const alertStatNote = document.getElementById("alertStatNote");
|
|
||||||
const alertsCard = document.getElementById("alertsCard");
|
|
||||||
const topAlertChip = document.getElementById("topAlertChip");
|
|
||||||
const topTaskbar = document.querySelector(".admin-taskbar");
|
|
||||||
|
|
||||||
if (!statusText || !alertsCard || !topAlertChip) return;
|
|
||||||
|
|
||||||
function showToast(message, type = "info") {
|
|
||||||
if (window.WarpBoxUI) {
|
|
||||||
window.WarpBoxUI.toast(message, type, { target: toast });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!toast) return;
|
|
||||||
toast.textContent = message;
|
|
||||||
toast.classList.add("is-visible");
|
|
||||||
window.clearTimeout(showToast.timer);
|
|
||||||
showToast.timer = window.setTimeout(() => toast.classList.remove("is-visible"), 2600);
|
|
||||||
}
|
|
||||||
|
|
||||||
function setStatus(message) {
|
|
||||||
statusText.textContent = message;
|
|
||||||
}
|
|
||||||
|
|
||||||
function openModal(title, meta) {
|
|
||||||
if (!modal || !backdrop || !modalTitle || !modalMeta) return;
|
|
||||||
modalTitle.textContent = title;
|
|
||||||
modalMeta.textContent = meta;
|
|
||||||
modal.classList.add("is-visible");
|
|
||||||
modal.setAttribute("aria-hidden", "false");
|
|
||||||
backdrop.classList.add("is-visible");
|
|
||||||
}
|
|
||||||
|
|
||||||
function closeModal() {
|
|
||||||
modal?.classList.remove("is-visible");
|
|
||||||
modal?.setAttribute("aria-hidden", "true");
|
|
||||||
backdrop?.classList.remove("is-visible");
|
|
||||||
}
|
|
||||||
|
|
||||||
function visibleAlertRows() {
|
|
||||||
return Array.from(document.querySelectorAll(".alert-row")).filter((row) => !row.classList.contains("is-dismissed"));
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateStickyHeader() {
|
|
||||||
topTaskbar?.classList.toggle("is-scrolled", window.scrollY > 4);
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateAlertSummary() {
|
|
||||||
const rows = visibleAlertRows();
|
|
||||||
const counts = rows.reduce((acc, row) => {
|
|
||||||
const severity = row.dataset.severity || "low";
|
|
||||||
acc[severity] = (acc[severity] || 0) + 1;
|
|
||||||
return acc;
|
|
||||||
}, { high: 0, medium: 0, low: 0 });
|
|
||||||
const score = counts.high * 5 + counts.medium * 2 + counts.low;
|
|
||||||
const total = rows.length;
|
|
||||||
const stateClass = counts.high > 0 || score >= 12 ? "is-danger" : counts.medium >= 2 || score >= 5 ? "is-warning" : total > 0 ? "is-info" : "is-ok";
|
|
||||||
|
|
||||||
alertsCard.classList.remove("is-ok", "is-info", "is-warning", "is-danger");
|
|
||||||
alertsCard.classList.add(stateClass);
|
|
||||||
topAlertChip.classList.remove("is-ok", "is-info", "is-warning", "is-danger");
|
|
||||||
topAlertChip.classList.add(stateClass);
|
|
||||||
if (alertCountValue) alertCountValue.textContent = String(total);
|
|
||||||
topAlertChip.textContent = total === 0 ? "OK no alerts" : `! ${total} alerts`;
|
|
||||||
if (alertStatNote) {
|
|
||||||
alertStatNote.innerHTML = total === 0
|
|
||||||
? '<span class="stat-note-pill">all clear</span>'
|
|
||||||
: `<span class="stat-note-pill">${counts.high} high</span><span class="stat-note-pill">${counts.medium} medium</span><span class="stat-note-pill">${counts.low} low</span>`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function scrollToSection(id) {
|
|
||||||
const target = document.getElementById(id);
|
|
||||||
if (!target) return;
|
|
||||||
target.scrollIntoView({ behavior: "smooth", block: "start" });
|
|
||||||
setStatus(`Focused ${id.replace("-", " ")}`);
|
|
||||||
}
|
|
||||||
|
|
||||||
const commandMessages = {
|
|
||||||
refresh: "CURRENTLY_MOCKED_LEAVE_AS_IS: dashboard refresh would re-fetch dashboard data.",
|
|
||||||
"dashboard-snapshot": "CURRENTLY_MOCKED_LEAVE_AS_IS: dashboard snapshot export would start here.",
|
|
||||||
logout: "CURRENTLY_MOCKED_LEAVE_AS_IS: logout would submit to the account logout route.",
|
|
||||||
"compact-mode": "Toggled compact density.",
|
|
||||||
"show-all-boxes": "TO-DO: navigate to the admin boxes view when that page exists.",
|
|
||||||
"show-all-alerts": "TO-DO: navigate to /admin/alerts.",
|
|
||||||
"export-boxes": "CURRENTLY_MOCKED_LEAVE_AS_IS: boxes CSV export would be requested.",
|
|
||||||
"export-alerts": "CURRENTLY_MOCKED_LEAVE_AS_IS: alerts JSON export would be requested.",
|
|
||||||
"cleanup-dry-run": "CURRENTLY_MOCKED_LEAVE_AS_IS: cleanup dry run would calculate affected boxes without deleting.",
|
|
||||||
"dismiss-low-alerts": "Closed visible low-severity alerts in this mock.",
|
|
||||||
"config-snapshot": "CURRENTLY_MOCKED_LEAVE_AS_IS: config snapshot would summarize runtime settings and sources.",
|
|
||||||
"support-summary": "CURRENTLY_MOCKED_LEAVE_AS_IS: support summary would collect safe diagnostic information.",
|
|
||||||
"thumbnail-rebuild": "CURRENTLY_MOCKED_LEAVE_AS_IS: thumbnail rebuild would enqueue preview regeneration.",
|
|
||||||
"open-users": "TO-DO: navigate to the admin users view when that page exists.",
|
|
||||||
"open-settings": "TO-DO: navigate to the admin settings view when that page exists.",
|
|
||||||
"alerts-help": "Alerts use title, description, severity, metadata JSON, trace identifier, and unique numeric code.",
|
|
||||||
shortcuts: "Shortcuts: F5 refresh, Alt+A alerts, Alt+B boxes, Alt+R activity, Esc close menus/modal.",
|
|
||||||
about: "WarpBox dashboard mock v5, single-window Win98 account dashboard."
|
|
||||||
};
|
|
||||||
|
|
||||||
function runCommand(command) {
|
|
||||||
if (command === "compact-mode") document.body.classList.toggle("is-compact");
|
|
||||||
if (command === "dismiss-low-alerts") {
|
|
||||||
document.querySelectorAll('.alert-row[data-severity="low"]').forEach((row) => row.classList.add("is-dismissed"));
|
|
||||||
updateAlertSummary();
|
|
||||||
}
|
|
||||||
if (command === "show-all-boxes") window.location.hash = "recent-boxes";
|
|
||||||
if (command === "show-all-alerts") window.location.hash = "alerts";
|
|
||||||
|
|
||||||
const message = commandMessages[command] || `Command: ${command}`;
|
|
||||||
showToast(message);
|
|
||||||
setStatus(message);
|
|
||||||
}
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-command]").forEach((button) => {
|
|
||||||
button.addEventListener("click", () => {
|
|
||||||
menuController.close();
|
|
||||||
runCommand(button.dataset.command);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-scroll-to]").forEach((button) => {
|
|
||||||
button.addEventListener("click", () => {
|
|
||||||
menuController.close();
|
|
||||||
scrollToSection(button.dataset.scrollTo);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-view-meta]").forEach((button) => {
|
|
||||||
button.addEventListener("click", () => {
|
|
||||||
const row = button.closest(".alert-row");
|
|
||||||
const title = row?.dataset.alertTitle || "Alert Metadata";
|
|
||||||
let meta = row?.dataset.alertMeta || "{}";
|
|
||||||
try {
|
|
||||||
meta = JSON.stringify(JSON.parse(meta), null, 2);
|
|
||||||
} catch (_) {
|
|
||||||
meta = row?.dataset.alertMeta || "{}";
|
|
||||||
}
|
|
||||||
openModal(`${title} (${row?.dataset.alertCode || "mock"})`, meta);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-dismiss-alert]").forEach((button) => {
|
|
||||||
button.addEventListener("click", () => {
|
|
||||||
const row = button.closest(".alert-row");
|
|
||||||
row?.classList.add("is-dismissed");
|
|
||||||
updateAlertSummary();
|
|
||||||
showToast(`Closed alert ${row?.dataset.alertCode || "mock"}.`);
|
|
||||||
setStatus(`Closed alert ${row?.dataset.alertCode || "mock"}`);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.querySelector("[data-close-modal]")?.addEventListener("click", closeModal);
|
|
||||||
backdrop?.addEventListener("click", closeModal);
|
|
||||||
topAlertChip.addEventListener("click", (event) => {
|
|
||||||
event.preventDefault();
|
|
||||||
scrollToSection("alerts");
|
|
||||||
});
|
|
||||||
|
|
||||||
window.addEventListener("scroll", updateStickyHeader, { passive: true });
|
|
||||||
|
|
||||||
document.addEventListener("keydown", (event) => {
|
|
||||||
if (event.key === "Escape") {
|
|
||||||
menuController.close();
|
|
||||||
closeModal();
|
|
||||||
}
|
|
||||||
if (event.key === "F5") {
|
|
||||||
event.preventDefault();
|
|
||||||
runCommand("refresh");
|
|
||||||
}
|
|
||||||
if (event.altKey && event.key.toLowerCase() === "a") {
|
|
||||||
event.preventDefault();
|
|
||||||
scrollToSection("alerts");
|
|
||||||
}
|
|
||||||
if (event.altKey && event.key.toLowerCase() === "b") {
|
|
||||||
event.preventDefault();
|
|
||||||
scrollToSection("recent-boxes");
|
|
||||||
}
|
|
||||||
if (event.altKey && event.key.toLowerCase() === "r") {
|
|
||||||
event.preventDefault();
|
|
||||||
scrollToSection("recent-activity");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
updateAlertSummary();
|
|
||||||
updateStickyHeader();
|
|
||||||
})();
|
|
||||||
@@ -1,314 +0,0 @@
|
|||||||
(() => {
|
|
||||||
const menuController = window.WarpBoxUI?.bindMenuBar?.() || { close() {} };
|
|
||||||
const eventsNode = document.getElementById("security-events-data");
|
|
||||||
const alertsNode = document.getElementById("security-alerts-data");
|
|
||||||
const bansNode = document.getElementById("security-bans-data");
|
|
||||||
const ipInput = document.getElementById("security-ip-input");
|
|
||||||
const banUntilInput = document.getElementById("security-ban-until");
|
|
||||||
const alertList = document.getElementById("security-alert-list");
|
|
||||||
const activityBody = document.getElementById("security-activity-body");
|
|
||||||
const bansBody = document.getElementById("security-bans-body");
|
|
||||||
const bansCount = document.getElementById("security-bans-count");
|
|
||||||
const filterInput = document.getElementById("security-ban-filter");
|
|
||||||
const sortSelect = document.getElementById("security-ban-sort");
|
|
||||||
const selectAll = document.getElementById("security-select-all");
|
|
||||||
const copyIPButton = document.getElementById("security-copy-ip");
|
|
||||||
const openActivityButton = document.getElementById("security-open-activity");
|
|
||||||
const openAlertsButton = document.getElementById("security-open-alerts");
|
|
||||||
const toast = document.getElementById("toast");
|
|
||||||
|
|
||||||
const detail = {
|
|
||||||
ip: document.getElementById("security-detail-ip"),
|
|
||||||
risk: document.getElementById("security-detail-risk"),
|
|
||||||
threat: document.getElementById("security-detail-threat"),
|
|
||||||
geo: document.getElementById("security-detail-geo"),
|
|
||||||
asn: document.getElementById("security-detail-asn"),
|
|
||||||
until: document.getElementById("security-detail-until"),
|
|
||||||
why: document.getElementById("security-detail-why")
|
|
||||||
};
|
|
||||||
|
|
||||||
if (!eventsNode || !alertsNode || !bansNode) return;
|
|
||||||
const state = {
|
|
||||||
events: parse(eventsNode),
|
|
||||||
alerts: parse(alertsNode),
|
|
||||||
bans: parse(bansNode),
|
|
||||||
selectedIP: "",
|
|
||||||
selectedIPs: new Set()
|
|
||||||
};
|
|
||||||
|
|
||||||
setDefaultBanUntil();
|
|
||||||
|
|
||||||
function parse(node) {
|
|
||||||
try { return JSON.parse(node.textContent || "[]"); } catch (_) { return []; }
|
|
||||||
}
|
|
||||||
|
|
||||||
function showToast(message, type = "info", duration = 1800) {
|
|
||||||
window.WarpBoxUI?.toast?.(message, type, { target: toast, duration });
|
|
||||||
}
|
|
||||||
|
|
||||||
function createdLabel(value) {
|
|
||||||
const parsed = new Date(value);
|
|
||||||
if (Number.isNaN(parsed.getTime())) return "-";
|
|
||||||
return parsed.toISOString().replace("T", " ").slice(0, 16) + " UTC";
|
|
||||||
}
|
|
||||||
|
|
||||||
function setDefaultBanUntil() {
|
|
||||||
const base = new Date(Date.now() + 30 * 60 * 1000);
|
|
||||||
const yyyy = String(base.getUTCFullYear());
|
|
||||||
const mm = String(base.getUTCMonth() + 1).padStart(2, "0");
|
|
||||||
const dd = String(base.getUTCDate()).padStart(2, "0");
|
|
||||||
const hh = String(base.getUTCHours()).padStart(2, "0");
|
|
||||||
const mi = String(base.getUTCMinutes()).padStart(2, "0");
|
|
||||||
banUntilInput.value = `${yyyy}-${mm}-${dd}T${hh}:${mi}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function toRFC3339FromLocalUTC(datetimeLocalValue) {
|
|
||||||
if (!datetimeLocalValue) return "";
|
|
||||||
const date = new Date(datetimeLocalValue + ":00Z");
|
|
||||||
if (Number.isNaN(date.getTime())) return "";
|
|
||||||
return date.toISOString();
|
|
||||||
}
|
|
||||||
|
|
||||||
function setSelectedIP(ip) {
|
|
||||||
state.selectedIP = ip || "";
|
|
||||||
if (state.selectedIP) ipInput.value = state.selectedIP;
|
|
||||||
renderBans();
|
|
||||||
renderIPDetails();
|
|
||||||
}
|
|
||||||
|
|
||||||
function render() {
|
|
||||||
renderAlerts();
|
|
||||||
renderActivity();
|
|
||||||
renderBans();
|
|
||||||
renderIPDetails();
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderAlerts() {
|
|
||||||
alertList.innerHTML = "";
|
|
||||||
state.alerts.slice(0, 12).forEach((alert) => {
|
|
||||||
const entry = document.createElement("li");
|
|
||||||
entry.textContent = `${createdLabel(alert.created_at)} | ${alert.severity || "low"} | ${alert.title || "-"}`;
|
|
||||||
alertList.appendChild(entry);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderActivity() {
|
|
||||||
activityBody.innerHTML = "";
|
|
||||||
state.events
|
|
||||||
.filter((event) => String(event.kind || "").startsWith("security") || String(event.kind || "").startsWith("auth.admin"))
|
|
||||||
.slice(0, 60)
|
|
||||||
.forEach((event) => {
|
|
||||||
const row = document.createElement("tr");
|
|
||||||
row.innerHTML = `
|
|
||||||
<td>${createdLabel(event.created_at)}</td>
|
|
||||||
<td>${escapeHtml(event.kind || "-")}</td>
|
|
||||||
<td>${escapeHtml(event.severity || "-")}</td>
|
|
||||||
<td>${escapeHtml(event.ip || "-")}</td>
|
|
||||||
<td>${escapeHtml(event.path || "-")}</td>
|
|
||||||
<td title="${escapeHtml(event.message || "-")}">${escapeHtml(event.message || "-")}</td>
|
|
||||||
`;
|
|
||||||
activityBody.appendChild(row);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function rowData() {
|
|
||||||
const banMap = new Map(state.bans.map((entry) => [entry.ip, entry]));
|
|
||||||
const filter = String(filterInput?.value || "").trim().toLowerCase();
|
|
||||||
let rows = state.bans.map((entry) => ({ ip: entry.ip, status: "banned", until: entry.until, ban: entry }));
|
|
||||||
if (filter) rows = rows.filter((row) => row.ip.toLowerCase().includes(filter));
|
|
||||||
const sort = sortSelect?.value || "expiry_asc";
|
|
||||||
rows.sort((a, b) => {
|
|
||||||
if (sort === "ip_asc") return a.ip.localeCompare(b.ip);
|
|
||||||
if (sort === "ip_desc") return b.ip.localeCompare(a.ip);
|
|
||||||
const av = new Date(a.until).getTime();
|
|
||||||
const bv = new Date(b.until).getTime();
|
|
||||||
return sort === "expiry_desc" ? bv - av : av - bv;
|
|
||||||
});
|
|
||||||
return { rows, banMap };
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderBans() {
|
|
||||||
bansBody.innerHTML = "";
|
|
||||||
const { rows } = rowData();
|
|
||||||
rows.forEach((rowData) => {
|
|
||||||
const row = document.createElement("tr");
|
|
||||||
row.className = "security-bans-body-row";
|
|
||||||
if (rowData.ip === state.selectedIP) row.classList.add("is-selected");
|
|
||||||
row.innerHTML = `
|
|
||||||
<td><input type="checkbox" class="security-row-select" data-ip="${escapeHtml(rowData.ip)}" ${state.selectedIPs.has(rowData.ip) ? "checked" : ""}></td>
|
|
||||||
<td>${escapeHtml(rowData.ip || "-")}</td>
|
|
||||||
<td>${rowData.status}</td>
|
|
||||||
<td>${createdLabel(rowData.until)}</td>
|
|
||||||
`;
|
|
||||||
row.addEventListener("click", (event) => {
|
|
||||||
if (event.target && event.target.classList.contains("security-row-select")) return;
|
|
||||||
setSelectedIP(rowData.ip);
|
|
||||||
});
|
|
||||||
bansBody.appendChild(row);
|
|
||||||
});
|
|
||||||
bansBody.querySelectorAll(".security-row-select").forEach((checkbox) => {
|
|
||||||
checkbox.addEventListener("change", () => {
|
|
||||||
const ip = checkbox.getAttribute("data-ip");
|
|
||||||
if (!ip) return;
|
|
||||||
if (checkbox.checked) state.selectedIPs.add(ip); else state.selectedIPs.delete(ip);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
bansCount.textContent = `${state.bans.length} active bans`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderIPDetails() {
|
|
||||||
const ip = state.selectedIP || String(ipInput.value || "").trim();
|
|
||||||
if (!ip) {
|
|
||||||
detail.ip.textContent = "No IP selected";
|
|
||||||
detail.risk.textContent = "-";
|
|
||||||
detail.threat.textContent = "-";
|
|
||||||
detail.geo.textContent = "GeoIP not enabled yet";
|
|
||||||
detail.asn.textContent = "GeoIP not enabled yet";
|
|
||||||
detail.until.textContent = "-";
|
|
||||||
detail.why.textContent = "-";
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
const ban = state.bans.find((entry) => entry.ip === ip);
|
|
||||||
const matchingEvents = state.events.filter((event) => String(event.ip || "") === ip);
|
|
||||||
const matchingAlerts = state.alerts.filter((alert) => String(alert?.meta?.ip || "") === ip);
|
|
||||||
const lastEvent = matchingEvents[0] || null;
|
|
||||||
detail.ip.textContent = ip;
|
|
||||||
detail.risk.textContent = ban ? "high" : "medium";
|
|
||||||
detail.threat.textContent = ban ? "Temporary banned source" : "Observed source";
|
|
||||||
detail.geo.textContent = "GeoIP not enabled yet";
|
|
||||||
detail.asn.textContent = "GeoIP not enabled yet";
|
|
||||||
detail.until.textContent = ban ? createdLabel(ban.until) : "Not banned";
|
|
||||||
detail.why.textContent = `${matchingEvents.length} events, ${matchingAlerts.length} alerts${lastEvent ? `, latest=${lastEvent.kind}` : ""}`;
|
|
||||||
if (ban && ban.until) {
|
|
||||||
const parsed = new Date(ban.until);
|
|
||||||
if (!Number.isNaN(parsed.getTime())) {
|
|
||||||
const yyyy = String(parsed.getUTCFullYear());
|
|
||||||
const mm = String(parsed.getUTCMonth() + 1).padStart(2, "0");
|
|
||||||
const dd = String(parsed.getUTCDate()).padStart(2, "0");
|
|
||||||
const hh = String(parsed.getUTCHours()).padStart(2, "0");
|
|
||||||
const mi = String(parsed.getUTCMinutes()).padStart(2, "0");
|
|
||||||
banUntilInput.value = `${yyyy}-${mm}-${dd}T${hh}:${mi}`;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function escapeHtml(value) {
|
|
||||||
return window.WarpBoxUI?.htmlEscape?.(value) || String(value ?? "");
|
|
||||||
}
|
|
||||||
|
|
||||||
async function postAction(action, payload = {}) {
|
|
||||||
const response = await fetch("/admin/security/actions", {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify({ action, ...payload })
|
|
||||||
});
|
|
||||||
const result = await response.json().catch(() => ({}));
|
|
||||||
if (!response.ok) throw new Error(result.error || "Request failed");
|
|
||||||
if (Array.isArray(result.bans)) state.bans = result.bans;
|
|
||||||
return result;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function banIP() {
|
|
||||||
const ip = String(ipInput.value || "").trim();
|
|
||||||
if (!ip) return showToast("Enter IP first", "warning");
|
|
||||||
const payload = await postAction("ban", { ip });
|
|
||||||
setSelectedIP(ip);
|
|
||||||
showToast(payload.message || "IP banned", "success");
|
|
||||||
}
|
|
||||||
|
|
||||||
async function banUntil() {
|
|
||||||
const ip = String(ipInput.value || "").trim();
|
|
||||||
if (!ip) return showToast("Enter IP first", "warning");
|
|
||||||
const banUntil = toRFC3339FromLocalUTC(banUntilInput.value);
|
|
||||||
if (!banUntil) return showToast("Set valid expiration date", "warning");
|
|
||||||
if (!window.confirm("Apply custom ban expiration?")) return;
|
|
||||||
const payload = await postAction("ban_until", { ip, ban_until: banUntil });
|
|
||||||
setSelectedIP(ip);
|
|
||||||
showToast(payload.message || "IP ban expiration updated", "success");
|
|
||||||
}
|
|
||||||
|
|
||||||
async function unbanIP() {
|
|
||||||
const ip = state.selectedIP || String(ipInput.value || "").trim();
|
|
||||||
if (!ip) return showToast("Select or enter IP first", "warning");
|
|
||||||
if (!window.confirm(`Unban ${ip}?`)) return;
|
|
||||||
const payload = await postAction("unban", { ip });
|
|
||||||
state.selectedIPs.delete(ip);
|
|
||||||
setSelectedIP("");
|
|
||||||
showToast(payload.message || "IP unbanned", "success");
|
|
||||||
}
|
|
||||||
|
|
||||||
async function bulkUnban() {
|
|
||||||
const ips = Array.from(state.selectedIPs);
|
|
||||||
if (ips.length === 0) return showToast("Select at least one banned IP", "warning");
|
|
||||||
if (!window.confirm(`Unban ${ips.length} selected IPs?`)) return;
|
|
||||||
const payload = await postAction("bulk_unban", { ips });
|
|
||||||
state.selectedIPs.clear();
|
|
||||||
setSelectedIP("");
|
|
||||||
showToast(payload.message || "Bulk unban complete", "success");
|
|
||||||
}
|
|
||||||
|
|
||||||
async function unbanAll() {
|
|
||||||
if (!window.confirm("Unban all active bans?")) return;
|
|
||||||
const payload = await postAction("unban_all");
|
|
||||||
state.selectedIPs.clear();
|
|
||||||
setSelectedIP("");
|
|
||||||
showToast(payload.message || "All bans cleared", "success");
|
|
||||||
}
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-command]").forEach((button) => {
|
|
||||||
button.addEventListener("click", async () => {
|
|
||||||
menuController.close();
|
|
||||||
try {
|
|
||||||
const command = button.dataset.command;
|
|
||||||
if (command === "refresh") return window.location.reload();
|
|
||||||
if (command === "ban-ip") return await banIP();
|
|
||||||
if (command === "ban-until") return await banUntil();
|
|
||||||
if (command === "unban-ip") return await unbanIP();
|
|
||||||
if (command === "bulk-unban") return await bulkUnban();
|
|
||||||
if (command === "unban-all") return await unbanAll();
|
|
||||||
} catch (error) {
|
|
||||||
showToast(error.message, "error", 3200);
|
|
||||||
} finally {
|
|
||||||
renderBans();
|
|
||||||
renderIPDetails();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
ipInput.addEventListener("input", () => renderIPDetails());
|
|
||||||
filterInput?.addEventListener("input", () => renderBans());
|
|
||||||
sortSelect?.addEventListener("change", () => renderBans());
|
|
||||||
selectAll?.addEventListener("change", () => {
|
|
||||||
if (selectAll.checked) state.bans.forEach((ban) => state.selectedIPs.add(ban.ip));
|
|
||||||
else state.selectedIPs.clear();
|
|
||||||
renderBans();
|
|
||||||
});
|
|
||||||
|
|
||||||
copyIPButton?.addEventListener("click", async () => {
|
|
||||||
const ip = state.selectedIP || String(ipInput.value || "").trim();
|
|
||||||
if (!ip) return showToast("No IP selected", "warning");
|
|
||||||
await navigator.clipboard.writeText(ip);
|
|
||||||
showToast("IP copied", "success");
|
|
||||||
});
|
|
||||||
openActivityButton?.addEventListener("click", () => {
|
|
||||||
const ip = state.selectedIP || String(ipInput.value || "").trim();
|
|
||||||
if (!ip) return showToast("No IP selected", "warning");
|
|
||||||
window.location.href = `/admin/activity?q=${encodeURIComponent(ip)}`;
|
|
||||||
});
|
|
||||||
openAlertsButton?.addEventListener("click", () => {
|
|
||||||
const ip = state.selectedIP || String(ipInput.value || "").trim();
|
|
||||||
if (!ip) return showToast("No IP selected", "warning");
|
|
||||||
window.location.href = `/admin/alerts?q=${encodeURIComponent(ip)}`;
|
|
||||||
});
|
|
||||||
|
|
||||||
document.addEventListener("keydown", (event) => {
|
|
||||||
if (event.key === "Escape") menuController.close();
|
|
||||||
if (event.key === "F5") {
|
|
||||||
event.preventDefault();
|
|
||||||
window.location.reload();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
if (state.bans.length > 0) setSelectedIP(state.bans[0].ip);
|
|
||||||
render();
|
|
||||||
})();
|
|
||||||
@@ -1,459 +0,0 @@
|
|||||||
(() => {
|
|
||||||
const menuController = window.WarpBoxUI?.bindMenuBar?.() || { close() {} };
|
|
||||||
const rowsNode = document.getElementById("settings-rows");
|
|
||||||
const searchInput = document.getElementById("settingsSearch");
|
|
||||||
const categoryButtons = Array.from(document.querySelectorAll(".settings-category-button"));
|
|
||||||
const groups = Array.from(document.querySelectorAll(".settings-group"));
|
|
||||||
const saveButton = document.getElementById("saveButton");
|
|
||||||
const exportButton = document.getElementById("exportButton");
|
|
||||||
const importButton = document.getElementById("importButton");
|
|
||||||
const resetButton = document.getElementById("resetButton");
|
|
||||||
const importInput = document.getElementById("settingsImportInput");
|
|
||||||
const dirtyChip = document.getElementById("dirtyChip");
|
|
||||||
const actionSummary = document.getElementById("actionSummary");
|
|
||||||
const visibleCount = document.getElementById("visibleCount");
|
|
||||||
const editableCount = document.getElementById("editableCount");
|
|
||||||
const unsavedCount = document.getElementById("unsavedCount");
|
|
||||||
const lockedCount = document.getElementById("lockedCount");
|
|
||||||
const statusLeft = document.getElementById("statusLeft");
|
|
||||||
const statusMiddle = document.getElementById("statusMiddle");
|
|
||||||
const statusRight = document.getElementById("statusRight");
|
|
||||||
const popupClose = document.getElementById("doc-popup-close");
|
|
||||||
const toastTarget = document.getElementById("toast");
|
|
||||||
|
|
||||||
if (!rowsNode || !searchInput || !saveButton) return;
|
|
||||||
|
|
||||||
const state = {
|
|
||||||
currentCategory: "all",
|
|
||||||
showChangedOnly: false,
|
|
||||||
showLockedOnly: false
|
|
||||||
};
|
|
||||||
|
|
||||||
function parseRows() {
|
|
||||||
try {
|
|
||||||
return JSON.parse(rowsNode.textContent || "[]");
|
|
||||||
} catch (_) {
|
|
||||||
return [];
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
const rowData = parseRows().reduce((map, row) => {
|
|
||||||
map[row.key] = row;
|
|
||||||
return map;
|
|
||||||
}, {});
|
|
||||||
|
|
||||||
const rows = Array.from(document.querySelectorAll(".setting-row")).map((row) => ({
|
|
||||||
element: row,
|
|
||||||
input: row.querySelector(".setting-input"),
|
|
||||||
hint: row.querySelector('[data-role="hint"]'),
|
|
||||||
badge: row.querySelector('[data-role="source-badge"]'),
|
|
||||||
key: row.dataset.key,
|
|
||||||
label: row.dataset.label,
|
|
||||||
category: row.dataset.category,
|
|
||||||
envName: row.dataset.envName,
|
|
||||||
type: row.dataset.type,
|
|
||||||
minimum: Number(row.dataset.minimum || 0),
|
|
||||||
locked: row.classList.contains("is-locked")
|
|
||||||
}));
|
|
||||||
|
|
||||||
function showToast(message, type = "info", duration = 2400) {
|
|
||||||
window.WarpBoxUI?.toast?.(message, type, { target: toastTarget, duration });
|
|
||||||
}
|
|
||||||
|
|
||||||
function escapeHtml(value) {
|
|
||||||
return window.WarpBoxUI?.htmlEscape?.(value) || String(value ?? "");
|
|
||||||
}
|
|
||||||
|
|
||||||
function currentValue(row) {
|
|
||||||
if (!row.input) return row.element.dataset.original || "";
|
|
||||||
return String(row.input.value ?? "").trim();
|
|
||||||
}
|
|
||||||
|
|
||||||
function isDirty(row) {
|
|
||||||
return !row.locked && currentValue(row) !== (row.element.dataset.original || "");
|
|
||||||
}
|
|
||||||
|
|
||||||
function validateRow(row) {
|
|
||||||
if (row.locked || !row.input) {
|
|
||||||
row.element.classList.remove("is-invalid");
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
|
|
||||||
const value = currentValue(row);
|
|
||||||
let valid = true;
|
|
||||||
|
|
||||||
if (row.type === "size_gb") {
|
|
||||||
if (!/^\d+(?:\.\d+)?$/.test(value)) valid = false;
|
|
||||||
else if (Number(value) < row.minimum) valid = false;
|
|
||||||
} else if (row.type === "int" || row.type === "int64") {
|
|
||||||
if (!/^\d+$/.test(value)) valid = false;
|
|
||||||
else if (Number(value) < row.minimum) valid = false;
|
|
||||||
} else if (row.type === "bool") {
|
|
||||||
valid = value === "true" || value === "false";
|
|
||||||
}
|
|
||||||
|
|
||||||
row.element.classList.toggle("is-invalid", !valid);
|
|
||||||
return valid;
|
|
||||||
}
|
|
||||||
|
|
||||||
function rowMatchesSearch(row) {
|
|
||||||
const query = searchInput.value.trim().toLowerCase();
|
|
||||||
if (!query) return true;
|
|
||||||
const data = [
|
|
||||||
row.label,
|
|
||||||
row.envName,
|
|
||||||
row.element.dataset.description,
|
|
||||||
row.key
|
|
||||||
].join(" ").toLowerCase();
|
|
||||||
return data.includes(query);
|
|
||||||
}
|
|
||||||
|
|
||||||
function applyFilters() {
|
|
||||||
let visible = 0;
|
|
||||||
|
|
||||||
groups.forEach((group) => {
|
|
||||||
let groupVisible = 0;
|
|
||||||
group.querySelectorAll(".setting-row").forEach((node) => {
|
|
||||||
const row = rows.find((item) => item.element === node);
|
|
||||||
const categoryMatch = state.currentCategory === "all" || row.category === state.currentCategory;
|
|
||||||
const searchMatch = rowMatchesSearch(row);
|
|
||||||
const changedMatch = !state.showChangedOnly || isDirty(row);
|
|
||||||
const lockedMatch = !state.showLockedOnly || row.locked;
|
|
||||||
const show = categoryMatch && searchMatch && changedMatch && lockedMatch;
|
|
||||||
node.classList.toggle("is-hidden", !show);
|
|
||||||
if (show) {
|
|
||||||
visible += 1;
|
|
||||||
groupVisible += 1;
|
|
||||||
}
|
|
||||||
});
|
|
||||||
group.hidden = groupVisible === 0;
|
|
||||||
});
|
|
||||||
|
|
||||||
visibleCount.textContent = String(visible);
|
|
||||||
statusMiddle.textContent = `category: ${state.currentCategory}`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateStats() {
|
|
||||||
let dirty = 0;
|
|
||||||
let editable = 0;
|
|
||||||
let locked = 0;
|
|
||||||
let invalid = 0;
|
|
||||||
|
|
||||||
rows.forEach((row) => {
|
|
||||||
if (row.locked) locked += 1;
|
|
||||||
else editable += 1;
|
|
||||||
if (isDirty(row)) dirty += 1;
|
|
||||||
if (!validateRow(row)) invalid += 1;
|
|
||||||
});
|
|
||||||
|
|
||||||
editableCount.textContent = String(editable);
|
|
||||||
lockedCount.textContent = String(locked);
|
|
||||||
unsavedCount.textContent = String(dirty);
|
|
||||||
dirtyChip.textContent = `${dirty} unsaved`;
|
|
||||||
dirtyChip.classList.toggle("is-dirty", dirty > 0);
|
|
||||||
saveButton.disabled = dirty === 0 || invalid > 0;
|
|
||||||
|
|
||||||
if (invalid > 0) {
|
|
||||||
actionSummary.textContent = `${invalid} invalid setting value(s) must be fixed before save.`;
|
|
||||||
statusLeft.textContent = "Invalid values";
|
|
||||||
statusRight.textContent = "fix before save";
|
|
||||||
} else if (dirty > 0) {
|
|
||||||
actionSummary.textContent = `${dirty} unsaved change(s) ready to save or export.`;
|
|
||||||
statusLeft.textContent = "Unsaved changes";
|
|
||||||
statusRight.textContent = "draft ready";
|
|
||||||
} else {
|
|
||||||
actionSummary.textContent = "No unsaved changes.";
|
|
||||||
statusLeft.textContent = "No unsaved changes";
|
|
||||||
statusRight.textContent = "admin only";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateView() {
|
|
||||||
updateStats();
|
|
||||||
applyFilters();
|
|
||||||
}
|
|
||||||
|
|
||||||
function setCategory(category) {
|
|
||||||
state.currentCategory = category;
|
|
||||||
categoryButtons.forEach((button) => button.classList.toggle("is-active", button.dataset.category === category));
|
|
||||||
applyFilters();
|
|
||||||
}
|
|
||||||
|
|
||||||
function draftValues() {
|
|
||||||
const values = {};
|
|
||||||
rows.forEach((row) => {
|
|
||||||
if (!row.locked && isDirty(row)) values[row.key] = currentValue(row);
|
|
||||||
});
|
|
||||||
return values;
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateRowFromPayload(payload) {
|
|
||||||
const row = rows.find((item) => item.key === payload.key);
|
|
||||||
if (!row) return;
|
|
||||||
|
|
||||||
row.element.dataset.original = payload.value;
|
|
||||||
row.element.dataset.default = payload.default_value || "";
|
|
||||||
row.element.dataset.source = payload.source || "default";
|
|
||||||
row.element.dataset.sourceBadge = payload.source_badge || payload.source || "default";
|
|
||||||
row.element.dataset.description = payload.description || "";
|
|
||||||
row.element.dataset.minimum = String(payload.minimum || 0);
|
|
||||||
row.element.classList.toggle("is-locked", Boolean(payload.locked));
|
|
||||||
row.locked = Boolean(payload.locked);
|
|
||||||
row.minimum = Number(payload.minimum || 0);
|
|
||||||
|
|
||||||
if (row.input) {
|
|
||||||
row.input.value = payload.value ?? "";
|
|
||||||
row.input.disabled = Boolean(payload.locked);
|
|
||||||
}
|
|
||||||
if (row.hint) {
|
|
||||||
row.hint.textContent = payload.locked
|
|
||||||
? "Locked by environment or hard runtime implication."
|
|
||||||
: payload.type === "size_gb"
|
|
||||||
? "Use GB values. Decimals allowed, for example `0.5`."
|
|
||||||
: (payload.default_value ? `Default: ${payload.default_value}` : "");
|
|
||||||
}
|
|
||||||
if (row.badge) {
|
|
||||||
row.badge.textContent = payload.source_badge || payload.source || "default";
|
|
||||||
row.badge.className = `settings-badge ${badgeClass(payload.source_badge || payload.source || "default")}`;
|
|
||||||
}
|
|
||||||
rowData[payload.key] = payload;
|
|
||||||
}
|
|
||||||
|
|
||||||
function badgeClass(source) {
|
|
||||||
if (source === "default") return "badge-default";
|
|
||||||
if (source === "environment") return "badge-env";
|
|
||||||
if (source === "db override") return "badge-db";
|
|
||||||
return "badge-hard";
|
|
||||||
}
|
|
||||||
|
|
||||||
function hydrateRows(payloadRows) {
|
|
||||||
if (!Array.isArray(payloadRows)) return;
|
|
||||||
payloadRows.forEach(updateRowFromPayload);
|
|
||||||
updateView();
|
|
||||||
}
|
|
||||||
|
|
||||||
async function postJSON(url, body) {
|
|
||||||
const response = await fetch(url, {
|
|
||||||
method: "POST",
|
|
||||||
headers: { "Content-Type": "application/json" },
|
|
||||||
body: JSON.stringify(body)
|
|
||||||
});
|
|
||||||
const payload = await response.json().catch(() => ({}));
|
|
||||||
if (!response.ok) {
|
|
||||||
throw new Error(payload.error || "Request failed");
|
|
||||||
}
|
|
||||||
return payload;
|
|
||||||
}
|
|
||||||
|
|
||||||
async function saveChanges() {
|
|
||||||
const values = draftValues();
|
|
||||||
if (Object.keys(values).length === 0) {
|
|
||||||
showToast("No changed settings to save", "info");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
try {
|
|
||||||
const payload = await postJSON("/admin/settings/save", { values });
|
|
||||||
hydrateRows(payload.rows);
|
|
||||||
showToast(payload.message || "Settings saved", payload.warnings?.length ? "warning" : "success");
|
|
||||||
} catch (error) {
|
|
||||||
showToast(error.message, "error", 3200);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function resetDefaults() {
|
|
||||||
if (!window.confirm("Reset all editable settings to built-in defaults?")) return;
|
|
||||||
try {
|
|
||||||
const payload = await postJSON("/admin/settings/reset", {});
|
|
||||||
hydrateRows(payload.rows);
|
|
||||||
showToast(payload.message || "Defaults restored", "success");
|
|
||||||
} catch (error) {
|
|
||||||
showToast(error.message, "error", 3200);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function resetSingleSetting(row) {
|
|
||||||
if (row.locked || !row.input) return;
|
|
||||||
|
|
||||||
if (isDirty(row)) {
|
|
||||||
row.input.value = row.element.dataset.original || "";
|
|
||||||
updateView();
|
|
||||||
showToast(`${row.label} draft cleared`);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
try {
|
|
||||||
const payload = await postJSON("/admin/settings/reset", { keys: [row.key] });
|
|
||||||
hydrateRows(payload.rows);
|
|
||||||
showToast(`${row.label} reset`, "success");
|
|
||||||
} catch (error) {
|
|
||||||
showToast(error.message, "error", 3200);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function exportSettings() {
|
|
||||||
try {
|
|
||||||
const response = await fetch("/admin/settings/export");
|
|
||||||
if (!response.ok) throw new Error("Could not export settings");
|
|
||||||
const payload = await response.json();
|
|
||||||
const blob = new Blob([JSON.stringify(payload, null, 2)], { type: "application/json;charset=utf-8" });
|
|
||||||
const url = URL.createObjectURL(blob);
|
|
||||||
const anchor = document.createElement("a");
|
|
||||||
anchor.href = url;
|
|
||||||
anchor.download = `warpbox-settings-${new Date().toISOString().replaceAll(":", "-")}.json`;
|
|
||||||
anchor.click();
|
|
||||||
URL.revokeObjectURL(url);
|
|
||||||
showToast("Settings JSON exported");
|
|
||||||
} catch (error) {
|
|
||||||
showToast(error.message, "error", 3200);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async function importSettingsFile(file) {
|
|
||||||
if (!file) return;
|
|
||||||
try {
|
|
||||||
const text = await file.text();
|
|
||||||
const payload = JSON.parse(text);
|
|
||||||
const result = await postJSON("/admin/settings/import", payload);
|
|
||||||
hydrateRows(result.rows);
|
|
||||||
showToast(result.message || "Settings imported", result.warnings?.length ? "warning" : "success", 3200);
|
|
||||||
} catch (error) {
|
|
||||||
showToast(error.message || "Could not import settings JSON", "error", 3200);
|
|
||||||
} finally {
|
|
||||||
importInput.value = "";
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function discardUnsaved() {
|
|
||||||
rows.forEach((row) => {
|
|
||||||
if (!row.input) return;
|
|
||||||
row.input.value = row.element.dataset.original || "";
|
|
||||||
});
|
|
||||||
updateView();
|
|
||||||
showToast("Unsaved changes discarded");
|
|
||||||
}
|
|
||||||
|
|
||||||
function explainSources() {
|
|
||||||
window.WarpBoxUI?.openPopup?.(
|
|
||||||
"Setting Sources",
|
|
||||||
`
|
|
||||||
<ul>
|
|
||||||
<li><strong>default</strong>: built-in application value.</li>
|
|
||||||
<li><strong>environment</strong>: loaded from an environment variable.</li>
|
|
||||||
<li><strong>db override</strong>: saved from the admin settings page.</li>
|
|
||||||
<li><strong>hard env</strong>: visible here, but locked for safety.</li>
|
|
||||||
</ul>
|
|
||||||
`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function explainReset() {
|
|
||||||
window.WarpBoxUI?.openPopup?.(
|
|
||||||
"Reset Behavior",
|
|
||||||
`
|
|
||||||
<p>Reset clears saved admin overrides.</p>
|
|
||||||
<p>After reset, environment values win again. If no environment value exists, built-in defaults apply.</p>
|
|
||||||
`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
function showRowInfo(row) {
|
|
||||||
window.WarpBoxUI?.openPopup?.(
|
|
||||||
row.label,
|
|
||||||
`
|
|
||||||
<p><strong>Environment variable:</strong> ${escapeHtml(row.envName || "n/a")}</p>
|
|
||||||
<p><strong>Current source:</strong> ${escapeHtml(row.badge?.textContent || row.element.dataset.sourceBadge || "default")}</p>
|
|
||||||
<p><strong>Description:</strong> ${escapeHtml(row.element.dataset.description || "No description available.")}</p>
|
|
||||||
${row.element.dataset.default ? `<p><strong>Default value:</strong> ${escapeHtml(row.element.dataset.default)}</p>` : ""}
|
|
||||||
`
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
async function runCommand(command) {
|
|
||||||
switch (command) {
|
|
||||||
case "save":
|
|
||||||
await saveChanges();
|
|
||||||
return;
|
|
||||||
case "export":
|
|
||||||
await exportSettings();
|
|
||||||
return;
|
|
||||||
case "import":
|
|
||||||
importInput.click();
|
|
||||||
return;
|
|
||||||
case "discard":
|
|
||||||
discardUnsaved();
|
|
||||||
return;
|
|
||||||
case "show-all":
|
|
||||||
state.showChangedOnly = false;
|
|
||||||
state.showLockedOnly = false;
|
|
||||||
applyFilters();
|
|
||||||
showToast("Showing all matching settings");
|
|
||||||
return;
|
|
||||||
case "show-changed":
|
|
||||||
state.showChangedOnly = !state.showChangedOnly;
|
|
||||||
if (state.showChangedOnly) state.showLockedOnly = false;
|
|
||||||
applyFilters();
|
|
||||||
showToast(state.showChangedOnly ? "Showing changed settings only" : "Showing all matching settings");
|
|
||||||
return;
|
|
||||||
case "show-locked":
|
|
||||||
state.showLockedOnly = !state.showLockedOnly;
|
|
||||||
if (state.showLockedOnly) state.showChangedOnly = false;
|
|
||||||
applyFilters();
|
|
||||||
showToast(state.showLockedOnly ? "Showing locked settings only" : "Showing all matching settings");
|
|
||||||
return;
|
|
||||||
case "reset-defaults":
|
|
||||||
await resetDefaults();
|
|
||||||
return;
|
|
||||||
case "reload":
|
|
||||||
window.location.reload();
|
|
||||||
return;
|
|
||||||
case "legend":
|
|
||||||
explainSources();
|
|
||||||
return;
|
|
||||||
case "reset-help":
|
|
||||||
explainReset();
|
|
||||||
return;
|
|
||||||
default:
|
|
||||||
showToast(`Unknown command: ${command}`, "warning");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
rows.forEach((row) => {
|
|
||||||
row.input?.addEventListener(row.input.tagName === "SELECT" ? "change" : "input", updateView);
|
|
||||||
row.element.querySelector(".row-reset")?.addEventListener("click", () => resetSingleSetting(row));
|
|
||||||
row.element.querySelector(".row-info")?.addEventListener("click", () => showRowInfo(row));
|
|
||||||
});
|
|
||||||
|
|
||||||
searchInput.addEventListener("input", applyFilters);
|
|
||||||
categoryButtons.forEach((button) => button.addEventListener("click", () => setCategory(button.dataset.category)));
|
|
||||||
saveButton.addEventListener("click", saveChanges);
|
|
||||||
exportButton.addEventListener("click", exportSettings);
|
|
||||||
importButton.addEventListener("click", () => importInput.click());
|
|
||||||
resetButton.addEventListener("click", resetDefaults);
|
|
||||||
importInput.addEventListener("change", (event) => importSettingsFile(event.target.files?.[0]));
|
|
||||||
popupClose?.addEventListener("click", () => window.WarpBoxUI?.closePopup?.());
|
|
||||||
document.getElementById("modal-backdrop")?.addEventListener("click", () => window.WarpBoxUI?.closePopup?.());
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-command]").forEach((button) => {
|
|
||||||
button.addEventListener("click", async () => {
|
|
||||||
menuController.close();
|
|
||||||
await runCommand(button.dataset.command);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.addEventListener("keydown", async (event) => {
|
|
||||||
if ((event.ctrlKey || event.metaKey) && event.key.toLowerCase() === "s") {
|
|
||||||
event.preventDefault();
|
|
||||||
await saveChanges();
|
|
||||||
}
|
|
||||||
if (event.key === "F5") {
|
|
||||||
event.preventDefault();
|
|
||||||
window.location.reload();
|
|
||||||
}
|
|
||||||
if (event.key === "Escape") {
|
|
||||||
menuController.close();
|
|
||||||
window.WarpBoxUI?.closePopup?.();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
updateView();
|
|
||||||
})();
|
|
||||||
@@ -1,304 +0,0 @@
|
|||||||
(() => {
|
|
||||||
const menuController = window.WarpBoxUI?.bindMenuBar?.() || { close() {} };
|
|
||||||
const toastTarget = document.getElementById("toast");
|
|
||||||
const body = document.getElementById("users-body");
|
|
||||||
const search = document.getElementById("users-search");
|
|
||||||
const status = document.getElementById("users-status");
|
|
||||||
const role = document.getElementById("users-role-filter");
|
|
||||||
const sort = document.getElementById("users-sort");
|
|
||||||
const size = document.getElementById("users-size");
|
|
||||||
const masterCheck = document.getElementById("users-master-check");
|
|
||||||
const pageInfo = document.getElementById("users-page-info");
|
|
||||||
const visiblePill = document.getElementById("visible-pill");
|
|
||||||
const selectedPill = document.getElementById("users-selected-pill");
|
|
||||||
const prevBtn = document.getElementById("users-prev");
|
|
||||||
const nextBtn = document.getElementById("users-next");
|
|
||||||
const selectVisible = document.getElementById("select-visible");
|
|
||||||
const form = document.getElementById("users-form");
|
|
||||||
const modeInput = document.getElementById("users-mode");
|
|
||||||
const usernameInput = document.getElementById("users-username");
|
|
||||||
const emailInput = document.getElementById("users-email");
|
|
||||||
const roleInput = document.getElementById("users-role");
|
|
||||||
const planInput = document.getElementById("users-plan");
|
|
||||||
const statusLeft = document.getElementById("users-status-left");
|
|
||||||
|
|
||||||
if (!body || !search || !status || !role || !sort || !size) return;
|
|
||||||
|
|
||||||
const users = [
|
|
||||||
{ id: "u_admin", username: "admin", email: "admin@warpbox.local", status: "active", role: "admin", plan: "unlimited", boxes: 18, created: "2026-04-12", lastSeen: "active now" },
|
|
||||||
{ id: "u_geo", username: "geo", email: "geo@example.test", status: "active", role: "uploader", plan: "trusted", boxes: 7, created: "2026-04-21", lastSeen: "today 12:10" },
|
|
||||||
{ id: "u_reo", username: "reo", email: "reo@example.test", status: "active", role: "uploader", plan: "standard", boxes: 3, created: "2026-04-20", lastSeen: "today 09:44" },
|
|
||||||
{ id: "u_teo", username: "teo", email: "teo@example.test", status: "active", role: "uploader", plan: "trusted", boxes: 5, created: "2026-04-19", lastSeen: "yesterday" },
|
|
||||||
{ id: "u_mara", username: "mara", email: "mara@example.test", status: "pending", role: "viewer", plan: "guest-like", boxes: 0, created: "2026-04-28", lastSeen: "never" },
|
|
||||||
{ id: "u_ion", username: "ion", email: "ion@example.test", status: "disabled", role: "uploader", plan: "standard", boxes: 2, created: "2026-04-01", lastSeen: "2026-04-15" },
|
|
||||||
{ id: "u_sara", username: "sara", email: "sara@example.test", status: "active", role: "operator", plan: "trusted", boxes: 12, created: "2026-03-30", lastSeen: "today 08:25" },
|
|
||||||
{ id: "u_vlad", username: "vlad", email: "vlad@example.test", status: "pending", role: "uploader", plan: "standard", boxes: 0, created: "2026-04-27", lastSeen: "never" },
|
|
||||||
{ id: "u_lina", username: "lina", email: "lina@example.test", status: "active", role: "viewer", plan: "guest-like", boxes: 1, created: "2026-03-22", lastSeen: "2026-04-29" },
|
|
||||||
{ id: "u_adi", username: "adi", email: "adi@example.test", status: "active", role: "uploader", plan: "standard", boxes: 4, created: "2026-02-18", lastSeen: "2026-04-26" },
|
|
||||||
{ id: "u_nora", username: "nora", email: "nora@example.test", status: "disabled", role: "viewer", plan: "guest-like", boxes: 0, created: "2026-01-14", lastSeen: "2026-03-02" },
|
|
||||||
{ id: "u_alex", username: "alex", email: "alex@example.test", status: "active", role: "uploader", plan: "trusted", boxes: 9, created: "2026-04-10", lastSeen: "2026-04-30" },
|
|
||||||
{ id: "u_rina", username: "rina", email: "rina@example.test", status: "pending", role: "uploader", plan: "standard", boxes: 0, created: "2026-04-29", lastSeen: "never" },
|
|
||||||
{ id: "u_mihai", username: "mihai", email: "mihai@example.test", status: "active", role: "operator", plan: "trusted", boxes: 6, created: "2026-02-08", lastSeen: "2026-04-22" }
|
|
||||||
];
|
|
||||||
|
|
||||||
const state = { page: 1, selected: new Set() };
|
|
||||||
|
|
||||||
function toast(message, type = "info") {
|
|
||||||
if (window.WarpBoxUI) {
|
|
||||||
window.WarpBoxUI.toast(message, type, { target: toastTarget, duration: 2200 });
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!toastTarget) return;
|
|
||||||
toastTarget.textContent = message;
|
|
||||||
toastTarget.classList.add("is-visible");
|
|
||||||
}
|
|
||||||
|
|
||||||
function filtered() {
|
|
||||||
const query = search.value.trim().toLowerCase();
|
|
||||||
const statusFilter = status.value;
|
|
||||||
const roleFilter = role.value;
|
|
||||||
const sortBy = sort.value;
|
|
||||||
const rows = users.filter((user) => {
|
|
||||||
const matchesQuery = !query || user.username.toLowerCase().includes(query) || user.email.toLowerCase().includes(query);
|
|
||||||
const matchesStatus = statusFilter === "all" || user.status === statusFilter;
|
|
||||||
const matchesRole = roleFilter === "all" || user.role === roleFilter;
|
|
||||||
return matchesQuery && matchesStatus && matchesRole;
|
|
||||||
});
|
|
||||||
|
|
||||||
rows.sort((a, b) => {
|
|
||||||
if (sortBy === "createdDesc") return b.created.localeCompare(a.created);
|
|
||||||
if (sortBy === "lastSeenDesc") return b.lastSeen.localeCompare(a.lastSeen);
|
|
||||||
if (sortBy === "boxesDesc") return b.boxes - a.boxes;
|
|
||||||
return a.username.localeCompare(b.username);
|
|
||||||
});
|
|
||||||
return rows;
|
|
||||||
}
|
|
||||||
|
|
||||||
function paged(rows) {
|
|
||||||
const perPage = Number(size.value || 12);
|
|
||||||
const pages = Math.max(1, Math.ceil(rows.length / perPage));
|
|
||||||
if (state.page > pages) state.page = pages;
|
|
||||||
if (state.page < 1) state.page = 1;
|
|
||||||
const start = (state.page - 1) * perPage;
|
|
||||||
return { rows: rows.slice(start, start + perPage), pages, start };
|
|
||||||
}
|
|
||||||
|
|
||||||
function statusPill(value) {
|
|
||||||
return `<span class="users-pill ${value}">${value}</span>`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderRow(user) {
|
|
||||||
const checked = state.selected.has(user.id) ? " checked" : "";
|
|
||||||
const row = document.createElement("tr");
|
|
||||||
row.innerHTML = `
|
|
||||||
<td><input type="checkbox" class="row-check"${checked}></td>
|
|
||||||
<td><div class="users-username"><strong>${user.username}</strong><span class="users-muted">${user.id}</span></div></td>
|
|
||||||
<td title="${user.email}">${user.email}</td>
|
|
||||||
<td>${statusPill(user.status)}</td>
|
|
||||||
<td>${user.role}</td>
|
|
||||||
<td>${user.plan}</td>
|
|
||||||
<td>${user.boxes}</td>
|
|
||||||
<td>${user.lastSeen}</td>
|
|
||||||
<td><div class="users-row-actions"><button class="win98-button users-row-button" type="button" data-action="open">Open</button></div></td>
|
|
||||||
`;
|
|
||||||
|
|
||||||
row.querySelector(".row-check")?.addEventListener("change", (event) => {
|
|
||||||
if (event.target.checked) state.selected.add(user.id);
|
|
||||||
else state.selected.delete(user.id);
|
|
||||||
syncSelected();
|
|
||||||
syncMasterCheck();
|
|
||||||
});
|
|
||||||
row.querySelector('[data-action="open"]')?.addEventListener("click", () => {
|
|
||||||
toast(`Mock user preview: ${user.username}`);
|
|
||||||
});
|
|
||||||
return row;
|
|
||||||
}
|
|
||||||
|
|
||||||
function syncSelected() {
|
|
||||||
selectedPill.textContent = `${state.selected.size} selected`;
|
|
||||||
}
|
|
||||||
|
|
||||||
function syncMasterCheck() {
|
|
||||||
const checks = Array.from(body.querySelectorAll(".row-check"));
|
|
||||||
masterCheck.checked = checks.length > 0 && checks.every((item) => item.checked);
|
|
||||||
}
|
|
||||||
|
|
||||||
function renderStats() {
|
|
||||||
document.getElementById("stat-total").textContent = String(users.length);
|
|
||||||
document.getElementById("stat-active").textContent = String(users.filter((u) => u.status === "active").length);
|
|
||||||
document.getElementById("stat-pending").textContent = String(users.filter((u) => u.status === "pending").length);
|
|
||||||
document.getElementById("stat-disabled").textContent = String(users.filter((u) => u.status === "disabled").length);
|
|
||||||
}
|
|
||||||
|
|
||||||
function render() {
|
|
||||||
const rows = filtered();
|
|
||||||
const page = paged(rows);
|
|
||||||
body.innerHTML = "";
|
|
||||||
page.rows.forEach((user) => body.appendChild(renderRow(user)));
|
|
||||||
|
|
||||||
visiblePill.textContent = `${rows.length} visible`;
|
|
||||||
pageInfo.textContent = `Page ${state.page} / ${page.pages}`;
|
|
||||||
prevBtn.disabled = state.page <= 1;
|
|
||||||
nextBtn.disabled = state.page >= page.pages;
|
|
||||||
statusLeft.textContent = `Ready. ${rows.length} user rows in current filter.`;
|
|
||||||
syncSelected();
|
|
||||||
syncMasterCheck();
|
|
||||||
}
|
|
||||||
|
|
||||||
function clearFilters() {
|
|
||||||
search.value = "";
|
|
||||||
status.value = "all";
|
|
||||||
role.value = "all";
|
|
||||||
sort.value = "username";
|
|
||||||
state.page = 1;
|
|
||||||
render();
|
|
||||||
}
|
|
||||||
|
|
||||||
function applyBulk(nextStatus) {
|
|
||||||
const selected = users.filter((user) => state.selected.has(user.id));
|
|
||||||
if (!selected.length) {
|
|
||||||
toast("Select one or more users first", "warning");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
selected.forEach((user) => { user.status = nextStatus; });
|
|
||||||
toast(`Updated ${selected.length} user(s) to ${nextStatus}`);
|
|
||||||
renderStats();
|
|
||||||
render();
|
|
||||||
}
|
|
||||||
|
|
||||||
function runCommand(command) {
|
|
||||||
switch (command) {
|
|
||||||
case "invite":
|
|
||||||
modeInput.value = "invite";
|
|
||||||
toast("Invite mode selected");
|
|
||||||
break;
|
|
||||||
case "create":
|
|
||||||
modeInput.value = "create";
|
|
||||||
toast("Create mode selected");
|
|
||||||
break;
|
|
||||||
case "export":
|
|
||||||
toast("Mock CSV export complete");
|
|
||||||
break;
|
|
||||||
case "bulk-disable":
|
|
||||||
applyBulk("disabled");
|
|
||||||
break;
|
|
||||||
case "bulk-enable":
|
|
||||||
applyBulk("active");
|
|
||||||
break;
|
|
||||||
case "bulk-revoke":
|
|
||||||
toast("Mock session revocation queued");
|
|
||||||
break;
|
|
||||||
case "refresh":
|
|
||||||
toast("Users list refreshed");
|
|
||||||
render();
|
|
||||||
break;
|
|
||||||
case "pending-only":
|
|
||||||
status.value = "pending";
|
|
||||||
state.page = 1;
|
|
||||||
render();
|
|
||||||
break;
|
|
||||||
case "clear-filters":
|
|
||||||
clearFilters();
|
|
||||||
break;
|
|
||||||
case "policy-help":
|
|
||||||
toast("Policy editor will be added in user details later.");
|
|
||||||
break;
|
|
||||||
case "mock-note":
|
|
||||||
toast("Mock-only page: no backend writes yet.");
|
|
||||||
break;
|
|
||||||
default:
|
|
||||||
toast(`Mock action: ${command}`);
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
[search, status, role, sort, size].forEach((el) => {
|
|
||||||
el.addEventListener(el.tagName === "INPUT" ? "input" : "change", () => {
|
|
||||||
state.page = 1;
|
|
||||||
render();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
prevBtn.addEventListener("click", () => {
|
|
||||||
state.page -= 1;
|
|
||||||
render();
|
|
||||||
});
|
|
||||||
|
|
||||||
nextBtn.addEventListener("click", () => {
|
|
||||||
state.page += 1;
|
|
||||||
render();
|
|
||||||
});
|
|
||||||
|
|
||||||
masterCheck.addEventListener("change", () => {
|
|
||||||
Array.from(body.querySelectorAll("tr")).forEach((row) => {
|
|
||||||
const checkbox = row.querySelector(".row-check");
|
|
||||||
if (!checkbox) return;
|
|
||||||
checkbox.checked = masterCheck.checked;
|
|
||||||
const userID = row.querySelector(".users-muted")?.textContent || "";
|
|
||||||
if (masterCheck.checked) state.selected.add(userID);
|
|
||||||
else state.selected.delete(userID);
|
|
||||||
});
|
|
||||||
syncSelected();
|
|
||||||
});
|
|
||||||
|
|
||||||
selectVisible.addEventListener("click", () => {
|
|
||||||
Array.from(body.querySelectorAll("tr")).forEach((row) => {
|
|
||||||
const checkbox = row.querySelector(".row-check");
|
|
||||||
const userID = row.querySelector(".users-muted")?.textContent || "";
|
|
||||||
if (!checkbox) return;
|
|
||||||
checkbox.checked = true;
|
|
||||||
state.selected.add(userID);
|
|
||||||
});
|
|
||||||
syncSelected();
|
|
||||||
syncMasterCheck();
|
|
||||||
});
|
|
||||||
|
|
||||||
form.addEventListener("submit", (event) => {
|
|
||||||
event.preventDefault();
|
|
||||||
const username = usernameInput.value.trim();
|
|
||||||
const email = emailInput.value.trim();
|
|
||||||
const mode = modeInput.value;
|
|
||||||
if (!username || !email) {
|
|
||||||
toast("Username and email are required", "warning");
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
users.unshift({
|
|
||||||
id: `u_${username.toLowerCase().replaceAll(/[^a-z0-9]+/g, "_")}`,
|
|
||||||
username,
|
|
||||||
email,
|
|
||||||
status: mode === "invite" ? "pending" : "active",
|
|
||||||
role: roleInput.value,
|
|
||||||
plan: planInput.value,
|
|
||||||
boxes: 0,
|
|
||||||
created: new Date().toISOString().slice(0, 10),
|
|
||||||
lastSeen: "never"
|
|
||||||
});
|
|
||||||
form.reset();
|
|
||||||
modeInput.value = "invite";
|
|
||||||
renderStats();
|
|
||||||
render();
|
|
||||||
toast(mode === "invite" ? "Mock invite created" : "Mock user created");
|
|
||||||
});
|
|
||||||
|
|
||||||
document.querySelectorAll("[data-command]").forEach((button) => {
|
|
||||||
button.addEventListener("click", () => {
|
|
||||||
menuController.close();
|
|
||||||
runCommand(button.dataset.command);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.addEventListener("keydown", (event) => {
|
|
||||||
if (event.key === "Escape") menuController.close();
|
|
||||||
if (event.key === "F5") {
|
|
||||||
event.preventDefault();
|
|
||||||
runCommand("refresh");
|
|
||||||
}
|
|
||||||
if ((event.ctrlKey || event.metaKey) && event.key.toLowerCase() === "i") {
|
|
||||||
event.preventDefault();
|
|
||||||
runCommand("invite");
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
renderStats();
|
|
||||||
render();
|
|
||||||
})();
|
|
||||||
@@ -53,46 +53,5 @@ function renderTemplate(template, data = {}) {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
function bindMenuBar(options = {}) {
|
return { toast, openPopup, closePopup, htmlEscape, renderTemplate };
|
||||||
const root = options.root || document;
|
|
||||||
const itemSelector = options.itemSelector || ".menu-item";
|
|
||||||
const buttonSelector = options.buttonSelector || ".menu-button";
|
|
||||||
const items = Array.from(root.querySelectorAll(itemSelector));
|
|
||||||
|
|
||||||
function close() {
|
|
||||||
items.forEach((item) => {
|
|
||||||
item.classList.remove("is-open");
|
|
||||||
item.querySelector(buttonSelector)?.setAttribute("aria-expanded", "false");
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function open(item) {
|
|
||||||
close();
|
|
||||||
item.classList.add("is-open");
|
|
||||||
item.querySelector(buttonSelector)?.setAttribute("aria-expanded", "true");
|
|
||||||
}
|
|
||||||
|
|
||||||
items.forEach((item) => {
|
|
||||||
const button = item.querySelector(buttonSelector);
|
|
||||||
button?.addEventListener("click", (event) => {
|
|
||||||
event.stopPropagation();
|
|
||||||
const wasOpen = item.classList.contains("is-open");
|
|
||||||
close();
|
|
||||||
if (!wasOpen) open(item);
|
|
||||||
});
|
|
||||||
|
|
||||||
item.addEventListener("mouseenter", () => {
|
|
||||||
if (!root.querySelector(`${itemSelector}.is-open`)) return;
|
|
||||||
open(item);
|
|
||||||
});
|
|
||||||
});
|
|
||||||
|
|
||||||
document.addEventListener("click", (event) => {
|
|
||||||
if (!event.target.closest(itemSelector)) close();
|
|
||||||
});
|
|
||||||
|
|
||||||
return { close, open };
|
|
||||||
}
|
|
||||||
|
|
||||||
return { toast, openPopup, closePopup, htmlEscape, renderTemplate, bindMenuBar };
|
|
||||||
})();
|
})();
|
||||||
|
|||||||
182
templates/account_alerts.html
Normal file
182
templates/account_alerts.html
Normal file
@@ -0,0 +1,182 @@
|
|||||||
|
{{ template "account_shell_start" . }}
|
||||||
|
<main class="account-window" aria-labelledby="account-alerts-title">
|
||||||
|
{{ template "account_window_titlebar" . }}
|
||||||
|
|
||||||
|
<nav class="menu-bar" aria-label="Alerts toolbar">
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">File</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
<a class="menu-action" href="/account/alerts"><span>R</span><span>Refresh alerts</span><span></span></a>
|
||||||
|
<a class="menu-action" href="/account/alerts/export.json"><span>E</span><span>Export JSON</span><span></span></a>
|
||||||
|
<div class="menu-separator"></div>
|
||||||
|
<form action="/account/logout" method="post">
|
||||||
|
{{ template "account_csrf_field" . }}
|
||||||
|
<button class="menu-action" type="submit"><span>Q</span><span>Log out</span><span></span></button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">View</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
<a class="menu-action" href="/account/alerts?status=open"><span>O</span><span>Open</span><span></span></a>
|
||||||
|
<a class="menu-action" href="/account/alerts?severity=high"><span>H</span><span>High severity</span><span></span></a>
|
||||||
|
<a class="menu-action" href="/account/alerts?sort=severity"><span>S</span><span>Sort by severity</span><span></span></a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<div class="alerts-layout account-body-content">
|
||||||
|
<section class="stats-grid" aria-label="Alert statistics">
|
||||||
|
<article class="stat-card sunken-panel is-warning">
|
||||||
|
<p class="stat-label">Open</p>
|
||||||
|
<p class="stat-value">{{ .Stats.Open }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">needs attention</span></p>
|
||||||
|
</article>
|
||||||
|
<article class="stat-card sunken-panel is-info">
|
||||||
|
<p class="stat-label">Acknowledged</p>
|
||||||
|
<p class="stat-value">{{ .Stats.Acknowledged }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">seen</span></p>
|
||||||
|
</article>
|
||||||
|
<article class="stat-card sunken-panel is-ok">
|
||||||
|
<p class="stat-label">Closed</p>
|
||||||
|
<p class="stat-value">{{ .Stats.Closed }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">done</span></p>
|
||||||
|
</article>
|
||||||
|
<article class="stat-card sunken-panel is-danger">
|
||||||
|
<p class="stat-label">High</p>
|
||||||
|
<p class="stat-value">{{ .Stats.High }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">{{ .Stats.Medium }} medium</span><span class="stat-note-pill">{{ .Stats.Low }} low</span></p>
|
||||||
|
</article>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<form class="alerts-filterbar raised-panel" action="/account/alerts" method="get">
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Search</span>
|
||||||
|
<input class="account-control" name="q" value="{{ .Filters.Query }}" placeholder="title, code, trace">
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Severity</span>
|
||||||
|
<select class="account-control" name="severity">
|
||||||
|
<option value="all" {{ if eq .Filters.Severity "all" }}selected{{ end }}>All</option>
|
||||||
|
<option value="low" {{ if eq .Filters.Severity "low" }}selected{{ end }}>Low</option>
|
||||||
|
<option value="medium" {{ if eq .Filters.Severity "medium" }}selected{{ end }}>Medium</option>
|
||||||
|
<option value="high" {{ if eq .Filters.Severity "high" }}selected{{ end }}>High</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Status</span>
|
||||||
|
<select class="account-control" name="status">
|
||||||
|
<option value="all" {{ if eq .Filters.Status "all" }}selected{{ end }}>All</option>
|
||||||
|
<option value="open" {{ if eq .Filters.Status "open" }}selected{{ end }}>Open</option>
|
||||||
|
<option value="acknowledged" {{ if eq .Filters.Status "acknowledged" }}selected{{ end }}>Acknowledged</option>
|
||||||
|
<option value="closed" {{ if eq .Filters.Status "closed" }}selected{{ end }}>Closed</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Group</span>
|
||||||
|
<select class="account-control" name="group">
|
||||||
|
<option value="all" {{ if eq .Filters.Group "all" }}selected{{ end }}>All</option>
|
||||||
|
{{ range .Groups }}
|
||||||
|
<option value="{{ . }}" {{ if eq $.Filters.Group . }}selected{{ end }}>{{ . }}</option>
|
||||||
|
{{ end }}
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Sort</span>
|
||||||
|
<select class="account-control" name="sort">
|
||||||
|
<option value="newest" {{ if eq .Filters.Sort "newest" }}selected{{ end }}>Newest</option>
|
||||||
|
<option value="oldest" {{ if eq .Filters.Sort "oldest" }}selected{{ end }}>Oldest</option>
|
||||||
|
<option value="severity" {{ if eq .Filters.Sort "severity" }}selected{{ end }}>Severity</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<button class="win98-button" type="submit">Apply</button>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<section class="alerts-workspace">
|
||||||
|
<form class="win98-window section-window" action="/account/alerts/bulk/acknowledge" method="post">
|
||||||
|
<div class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<span class="win98-titlebar-icon">!</span>
|
||||||
|
<h2>Alert List</h2>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="section-body sunken-panel">
|
||||||
|
{{ template "account_csrf_field" . }}
|
||||||
|
<div class="scroll-panel alerts-table-scroll">
|
||||||
|
<table class="account-table alerts-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Select</th>
|
||||||
|
<th>Severity</th>
|
||||||
|
<th>Status</th>
|
||||||
|
<th>Code</th>
|
||||||
|
<th>Title</th>
|
||||||
|
<th>Trace</th>
|
||||||
|
<th>Created</th>
|
||||||
|
<th>Actions</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{{ range .Alerts }}
|
||||||
|
<tr data-alert-row data-alert-id="{{ .ID }}" data-alert-title="{{ .Title }}" data-alert-description="{{ .Description }}" data-alert-metadata="{{ .MetadataPretty }}" class="{{ if eq $.SelectedAlert.ID .ID }}is-selected{{ end }}">
|
||||||
|
<td><input type="checkbox" name="alert_ids" value="{{ .ID }}"></td>
|
||||||
|
<td><span class="badge is-{{ .Severity }}">{{ .Severity }}</span></td>
|
||||||
|
<td><span class="badge">{{ .Status }}</span></td>
|
||||||
|
<td>{{ .Code }}</td>
|
||||||
|
<td>{{ .Title }}</td>
|
||||||
|
<td>{{ .Trace }}</td>
|
||||||
|
<td>{{ .CreatedAt }}</td>
|
||||||
|
<td>
|
||||||
|
<div class="box-actions">
|
||||||
|
{{ if $.CanManageAlerts }}
|
||||||
|
<button class="tiny-button" type="submit" formaction="/account/alerts/{{ .ID }}/acknowledge">Ack</button>
|
||||||
|
<button class="tiny-button" type="submit" formaction="/account/alerts/{{ .ID }}/close">Close</button>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{{ else }}
|
||||||
|
<tr><td colspan="8">No alerts found.</td></tr>
|
||||||
|
{{ end }}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
{{ if .CanManageAlerts }}
|
||||||
|
<div class="bulk-actions raised-panel">
|
||||||
|
<button class="win98-button" type="submit">Acknowledge selected</button>
|
||||||
|
<button class="win98-button" type="submit" formaction="/account/alerts/bulk/close">Close selected</button>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<aside class="alerts-detail sunken-panel" aria-label="Alert details">
|
||||||
|
{{ if .SelectedAlert }}
|
||||||
|
<div>
|
||||||
|
<h2 data-alert-detail-title>{{ .SelectedAlert.Title }}</h2>
|
||||||
|
<p data-alert-detail-description>{{ .SelectedAlert.Description }}</p>
|
||||||
|
</div>
|
||||||
|
<pre class="metadata-pre" data-alert-detail-metadata>{{ .SelectedAlert.MetadataPretty }}</pre>
|
||||||
|
<div class="setting-source">
|
||||||
|
<span class="badge is-{{ .SelectedAlert.Severity }}">{{ .SelectedAlert.Severity }}</span>
|
||||||
|
<span class="badge">{{ .SelectedAlert.Status }}</span>
|
||||||
|
<span class="setting-env">{{ .SelectedAlert.Trace }}</span>
|
||||||
|
</div>
|
||||||
|
{{ else }}
|
||||||
|
<div>
|
||||||
|
<h2 data-alert-detail-title>No alert selected</h2>
|
||||||
|
<p data-alert-detail-description>Select an alert row to inspect metadata.</p>
|
||||||
|
</div>
|
||||||
|
<pre class="metadata-pre" data-alert-detail-metadata>{}</pre>
|
||||||
|
{{ end }}
|
||||||
|
</aside>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<footer class="win98-statusbar" aria-label="Alerts status">
|
||||||
|
<span>alerts</span>
|
||||||
|
<span>{{ .Stats.Open }} open</span>
|
||||||
|
<span>ready</span>
|
||||||
|
</footer>
|
||||||
|
</main>
|
||||||
|
{{ template "account_shell_end" . }}
|
||||||
174
templates/account_boxes.html
Normal file
174
templates/account_boxes.html
Normal file
@@ -0,0 +1,174 @@
|
|||||||
|
{{ template "account_shell_start" . }}
|
||||||
|
<main class="account-window" aria-labelledby="account-boxes-title">
|
||||||
|
{{ template "account_window_titlebar" . }}
|
||||||
|
|
||||||
|
<nav class="menu-bar" aria-label="Boxes toolbar">
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">File</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
<a class="menu-action" href="/account/boxes"><span>R</span><span>Refresh boxes</span><span></span></a>
|
||||||
|
<a class="menu-action" href="/account/boxes/export.csv"><span>E</span><span>Export visible CSV</span><span></span></a>
|
||||||
|
<div class="menu-separator"></div>
|
||||||
|
<form action="/account/logout" method="post">
|
||||||
|
{{ template "account_csrf_field" . }}
|
||||||
|
<button class="menu-action" type="submit"><span>Q</span><span>Log out</span><span></span></button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">View</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
<a class="menu-action" href="/account/boxes?status=active"><span>A</span><span>Active</span><span></span></a>
|
||||||
|
<a class="menu-action" href="/account/boxes?status=expired"><span>X</span><span>Expired</span><span></span></a>
|
||||||
|
<a class="menu-action" href="/account/boxes?sort=largest"><span>L</span><span>Largest first</span><span></span></a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<div class="boxes-layout account-body-content">
|
||||||
|
{{ if .Error }}<p class="account-error">{{ .Error }}</p>{{ end }}
|
||||||
|
|
||||||
|
<section class="stats-grid" aria-label="Box statistics">
|
||||||
|
<article class="stat-card sunken-panel is-info">
|
||||||
|
<p class="stat-label">Visible</p>
|
||||||
|
<p class="stat-value">{{ .Stats.Visible }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">{{ .Stats.Total }} matching</span></p>
|
||||||
|
</article>
|
||||||
|
<article class="stat-card sunken-panel is-warning">
|
||||||
|
<p class="stat-label">Expired</p>
|
||||||
|
<p class="stat-value">{{ .Stats.Expired }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">visible page</span></p>
|
||||||
|
</article>
|
||||||
|
<article class="stat-card sunken-panel is-info">
|
||||||
|
<p class="stat-label">Storage</p>
|
||||||
|
<p class="stat-value">{{ .Stats.Storage }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">visible page</span></p>
|
||||||
|
</article>
|
||||||
|
<article class="stat-card sunken-panel is-ok">
|
||||||
|
<p class="stat-label">Policy</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">{{ .PolicySummary }}</span></p>
|
||||||
|
</article>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<form class="boxes-filterbar raised-panel" action="/account/boxes" method="get">
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Search</span>
|
||||||
|
<input class="account-control" name="q" value="{{ .Filters.Query }}" placeholder="id, owner, file">
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Owner</span>
|
||||||
|
<input class="account-control" name="owner" value="{{ .Filters.Owner }}" placeholder="all">
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Status</span>
|
||||||
|
<select class="account-control" name="status">
|
||||||
|
<option value="all" {{ if eq .Filters.Status "all" }}selected{{ end }}>All</option>
|
||||||
|
<option value="active" {{ if eq .Filters.Status "active" }}selected{{ end }}>Active</option>
|
||||||
|
<option value="pending" {{ if eq .Filters.Status "pending" }}selected{{ end }}>Pending</option>
|
||||||
|
<option value="expired" {{ if eq .Filters.Status "expired" }}selected{{ end }}>Expired</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Flag</span>
|
||||||
|
<select class="account-control" name="flag">
|
||||||
|
<option value="all" {{ if eq .Filters.Flag "all" }}selected{{ end }}>All</option>
|
||||||
|
<option value="password" {{ if eq .Filters.Flag "password" }}selected{{ end }}>Password</option>
|
||||||
|
<option value="one-time" {{ if eq .Filters.Flag "one-time" }}selected{{ end }}>One-time</option>
|
||||||
|
<option value="zip-disabled" {{ if eq .Filters.Flag "zip-disabled" }}selected{{ end }}>ZIP disabled</option>
|
||||||
|
<option value="expired" {{ if eq .Filters.Flag "expired" }}selected{{ end }}>Expired</option>
|
||||||
|
<option value="refreshable" {{ if eq .Filters.Flag "refreshable" }}selected{{ end }}>Refreshable</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Sort</span>
|
||||||
|
<select class="account-control" name="sort">
|
||||||
|
<option value="newest" {{ if eq .Filters.Sort "newest" }}selected{{ end }}>Newest</option>
|
||||||
|
<option value="oldest" {{ if eq .Filters.Sort "oldest" }}selected{{ end }}>Oldest</option>
|
||||||
|
<option value="largest" {{ if eq .Filters.Sort "largest" }}selected{{ end }}>Largest</option>
|
||||||
|
<option value="expires" {{ if eq .Filters.Sort "expires" }}selected{{ end }}>Expires soon</option>
|
||||||
|
<option value="expired" {{ if eq .Filters.Sort "expired" }}selected{{ end }}>Expired first</option>
|
||||||
|
</select>
|
||||||
|
</label>
|
||||||
|
<input type="hidden" name="page_size" value="{{ .PageSize }}">
|
||||||
|
<button class="win98-button" type="submit">Apply</button>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<form class="win98-window section-window" action="/account/boxes/bulk/expire" method="post">
|
||||||
|
{{ template "account_csrf_field" . }}
|
||||||
|
<div class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<span class="win98-titlebar-icon">B</span>
|
||||||
|
<h2>Box Index</h2>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="section-body sunken-panel">
|
||||||
|
<div class="scroll-panel boxes-table-scroll">
|
||||||
|
<table class="account-table boxes-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Select</th>
|
||||||
|
<th>Box</th>
|
||||||
|
<th>Owner</th>
|
||||||
|
<th>Status</th>
|
||||||
|
<th>Files</th>
|
||||||
|
<th>Size</th>
|
||||||
|
<th>Created</th>
|
||||||
|
<th>Expires</th>
|
||||||
|
<th>Flags</th>
|
||||||
|
<th>Refresh policy</th>
|
||||||
|
<th>Actions</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{{ range .Rows }}
|
||||||
|
<tr>
|
||||||
|
<td><input type="checkbox" name="box_ids" value="{{ .ID }}"></td>
|
||||||
|
<td>{{ .ID }}</td>
|
||||||
|
<td>{{ .Owner }}</td>
|
||||||
|
<td><span class="badge">{{ .Status }}</span></td>
|
||||||
|
<td>{{ .FileCount }}</td>
|
||||||
|
<td>{{ .Size }}</td>
|
||||||
|
<td>{{ .CreatedAt }}</td>
|
||||||
|
<td>{{ .ExpiresAt }}</td>
|
||||||
|
<td>{{ .Flags }}</td>
|
||||||
|
<td>{{ .Policy }}</td>
|
||||||
|
<td>
|
||||||
|
<div class="box-actions">
|
||||||
|
<a class="tiny-button" href="{{ .OpenURL }}">Open</a>
|
||||||
|
{{ if .CanManage }}<a class="tiny-button" href="{{ .ManageURL }}">Manage</a>{{ end }}
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{{ else }}
|
||||||
|
<tr><td colspan="11">No indexed boxes found.</td></tr>
|
||||||
|
{{ end }}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
<div class="bulk-actions raised-panel">
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Bump seconds</span>
|
||||||
|
<input class="account-control" name="bump_seconds" value="3600" inputmode="numeric">
|
||||||
|
</label>
|
||||||
|
<button class="win98-button" type="submit" data-confirm="Expire selected boxes?">Expire selected</button>
|
||||||
|
<button class="win98-button" type="submit" formaction="/account/boxes/bulk/bump-expiry">Bump selected</button>
|
||||||
|
<button class="win98-button" type="submit" formaction="/account/boxes/bulk/delete" data-confirm="Delete selected boxes permanently?">Delete selected</button>
|
||||||
|
<button class="win98-button" type="submit" formaction="/account/boxes/delete-largest" data-confirm="Delete 10 biggest matching boxes permanently?">Delete largest 10</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<nav class="pagination-strip raised-panel" aria-label="Pagination">
|
||||||
|
<span class="badge">Page {{ .Page }} / {{ .TotalPages }}</span>
|
||||||
|
{{ if .HasPrev }}<a class="win98-button" href="{{ .PrevURL }}">Prev</a>{{ end }}
|
||||||
|
{{ if .HasNext }}<a class="win98-button" href="{{ .NextURL }}">Next</a>{{ end }}
|
||||||
|
</nav>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<footer class="win98-statusbar" aria-label="Boxes status">
|
||||||
|
<span>boxes index</span>
|
||||||
|
<span>{{ .Total }} matching</span>
|
||||||
|
<span>ready</span>
|
||||||
|
</footer>
|
||||||
|
</main>
|
||||||
|
{{ template "account_shell_end" . }}
|
||||||
198
templates/account_dashboard.html
Normal file
198
templates/account_dashboard.html
Normal file
@@ -0,0 +1,198 @@
|
|||||||
|
{{ template "account_shell_start" . }}
|
||||||
|
<main class="account-window" aria-labelledby="account-dashboard-title">
|
||||||
|
{{ template "account_window_titlebar" . }}
|
||||||
|
|
||||||
|
<nav class="menu-bar" aria-label="Dashboard toolbar">
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">File</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
<a class="menu-action" href="/account"><span>R</span><span>Refresh dashboard</span><span class="shortcut">F5</span></a>
|
||||||
|
<div class="menu-separator"></div>
|
||||||
|
<form action="/account/logout" method="post">
|
||||||
|
{{ template "account_csrf_field" . }}
|
||||||
|
<button class="menu-action" type="submit"><span>Q</span><span>Log out</span><span></span></button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">View</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
<a class="menu-action" href="#alerts"><span>!</span><span>Go to alerts</span><span></span></a>
|
||||||
|
<a class="menu-action" href="#recent-boxes"><span>B</span><span>Go to recent boxes</span><span></span></a>
|
||||||
|
<a class="menu-action" href="#recent-activity"><span>T</span><span>Go to recent activity</span><span></span></a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">Tools</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
<a class="menu-action" href="/account/boxes"><span>B</span><span>Boxes</span><span></span></a>
|
||||||
|
<a class="menu-action" href="/account/alerts"><span>!</span><span>Alerts</span><span></span></a>
|
||||||
|
{{ if .CanManageUsers }}
|
||||||
|
<a class="menu-action" href="/account/users"><span>U</span><span>Users</span><span></span></a>
|
||||||
|
{{ end }}
|
||||||
|
{{ if .CanViewSettings }}
|
||||||
|
<a class="menu-action" href="/account/settings"><span>S</span><span>Settings</span><span></span></a>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<div class="account-body-content">
|
||||||
|
<section class="dashboard-hero raised-panel" aria-labelledby="account-dashboard-title">
|
||||||
|
<div class="hero-copy">
|
||||||
|
<h2 id="account-dashboard-title">Dashboard</h2>
|
||||||
|
<p>Account overview for boxes, alerts, storage, users, and recent activity.</p>
|
||||||
|
</div>
|
||||||
|
<div class="hero-status" aria-label="System summary">
|
||||||
|
{{ range .Statuses }}
|
||||||
|
<div class="hero-status-row"><span>{{ .Label }}</span><strong class="status-{{ .Severity }}">{{ .Value }}</strong></div>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="stats-grid" aria-label="Dashboard statistics">
|
||||||
|
<article class="stat-card sunken-panel is-info">
|
||||||
|
<p class="stat-label">Active boxes</p>
|
||||||
|
<p class="stat-value">{{ .Stats.ActiveBoxes }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">live filesystem scan</span></p>
|
||||||
|
</article>
|
||||||
|
<article class="stat-card sunken-panel is-info">
|
||||||
|
<p class="stat-label">Storage used</p>
|
||||||
|
<p class="stat-value">{{ .Stats.StorageUsedLabel }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">local backend</span></p>
|
||||||
|
</article>
|
||||||
|
<article class="stat-card sunken-panel is-warning">
|
||||||
|
<p class="stat-label">Alerts</p>
|
||||||
|
<p class="stat-value">{{ .Stats.AlertCount }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">alert model pending</span></p>
|
||||||
|
</article>
|
||||||
|
{{ if .ShowUsersStat }}
|
||||||
|
<article class="stat-card sunken-panel is-ok">
|
||||||
|
<p class="stat-label">Users</p>
|
||||||
|
<p class="stat-value">{{ .Stats.TotalUsers }}</p>
|
||||||
|
<p class="stat-note"><span class="stat-note-pill">{{ .Stats.ActiveUsers }} active</span><span class="stat-note-pill">{{ .Stats.DisabledUsers }} disabled</span></p>
|
||||||
|
</article>
|
||||||
|
{{ end }}
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="main-grid" aria-label="Dashboard panels">
|
||||||
|
<article id="alerts" class="win98-window section-window">
|
||||||
|
<div class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<span class="win98-titlebar-icon">!</span>
|
||||||
|
<h2>Alerts Preview</h2>
|
||||||
|
</div>
|
||||||
|
<div class="titlebar-actions">
|
||||||
|
<a class="titlebar-link-button" href="/account/alerts">Show all</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="section-body sunken-panel">
|
||||||
|
<div class="scroll-panel alerts-scroll">
|
||||||
|
<div class="alert-list">
|
||||||
|
{{ range .Alerts }}
|
||||||
|
<div class="alert-row">
|
||||||
|
<span class="badge is-{{ .Severity }}">{{ .Severity }}</span>
|
||||||
|
<div>
|
||||||
|
<p class="alert-title">{{ .Title }}</p>
|
||||||
|
<p class="alert-desc">{{ .Detail }}</p>
|
||||||
|
</div>
|
||||||
|
<div class="alert-actions">
|
||||||
|
<a class="tiny-button" href="/account/alerts">Open</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{ else }}
|
||||||
|
<div class="alert-row">
|
||||||
|
<span class="badge is-ok">ok</span>
|
||||||
|
<div><p class="alert-title">No alerts</p><p class="alert-desc">Nothing needs attention.</p></div>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
|
||||||
|
<article id="recent-boxes" class="win98-window section-window">
|
||||||
|
<div class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<span class="win98-titlebar-icon">B</span>
|
||||||
|
<h2>Recent Boxes</h2>
|
||||||
|
</div>
|
||||||
|
<div class="titlebar-actions">
|
||||||
|
<a class="titlebar-link-button" href="/account/boxes">Show all</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="section-body sunken-panel">
|
||||||
|
<div class="scroll-panel boxes-scroll">
|
||||||
|
<table class="account-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Box</th>
|
||||||
|
<th>Files</th>
|
||||||
|
<th>Size</th>
|
||||||
|
<th>Created</th>
|
||||||
|
<th>Expires</th>
|
||||||
|
<th>Flags</th>
|
||||||
|
<th>Actions</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{{ range .RecentBoxes }}
|
||||||
|
<tr>
|
||||||
|
<td>{{ .ID }}</td>
|
||||||
|
<td>{{ .FileCount }}</td>
|
||||||
|
<td>{{ .TotalSizeLabel }}</td>
|
||||||
|
<td>{{ .CreatedAt }}</td>
|
||||||
|
<td>{{ .ExpiresAt }}</td>
|
||||||
|
<td>{{ .Flags }}</td>
|
||||||
|
<td>
|
||||||
|
<div class="box-actions">
|
||||||
|
<a class="tiny-button" href="/box/{{ .ID }}">Open</a>
|
||||||
|
{{ if .CanManage }}
|
||||||
|
<a class="tiny-button" href="/account/boxes/{{ .ID }}">Manage</a>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{{ else }}
|
||||||
|
<tr><td colspan="7">No boxes found.</td></tr>
|
||||||
|
{{ end }}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
|
||||||
|
<article id="recent-activity" class="win98-window section-window span-2">
|
||||||
|
<div class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<span class="win98-titlebar-icon">T</span>
|
||||||
|
<h2>Recent Activity</h2>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="section-body sunken-panel">
|
||||||
|
<div class="scroll-panel activity-scroll">
|
||||||
|
<div class="activity-list">
|
||||||
|
{{ range .RecentActivity }}
|
||||||
|
<div class="activity-row">
|
||||||
|
<span class="activity-time">{{ .Time }}</span>
|
||||||
|
<div>
|
||||||
|
<p class="activity-title">{{ .Title }}</p>
|
||||||
|
<p class="activity-meta">{{ .Meta }}</p>
|
||||||
|
</div>
|
||||||
|
<span class="tag info">account</span>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</article>
|
||||||
|
</section>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<footer class="win98-statusbar" aria-label="Dashboard status">
|
||||||
|
<span>signed in: {{ .AccountNav.Username }}</span>
|
||||||
|
<span>{{ if .AccountNav.IsAdmin }}admin{{ else }}account{{ end }}</span>
|
||||||
|
<span>ready</span>
|
||||||
|
</footer>
|
||||||
|
</main>
|
||||||
|
{{ template "account_shell_end" . }}
|
||||||
45
templates/account_login.html
Normal file
45
templates/account_login.html
Normal file
@@ -0,0 +1,45 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>{{ .PageTitle }}</title>
|
||||||
|
{{ template "account_head_assets" . }}
|
||||||
|
</head>
|
||||||
|
<body class="account-body">
|
||||||
|
<div class="app-shell">
|
||||||
|
<div class="app-frame">
|
||||||
|
<main class="account-window" aria-labelledby="account-login-title">
|
||||||
|
<div class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<span class="win98-titlebar-icon">W</span>
|
||||||
|
<h1 id="account-login-title">WarpBox Account Login</h1>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="account-body-content">
|
||||||
|
{{ if .Error }}
|
||||||
|
<p class="account-error">{{ .Error }}</p>
|
||||||
|
{{ end }}
|
||||||
|
{{ if .AccountLoginEnabled }}
|
||||||
|
<form class="account-form sunken-panel" action="/account/login" method="post">
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Username</span>
|
||||||
|
<input name="username" autocomplete="username" required>
|
||||||
|
</label>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Password</span>
|
||||||
|
<input name="password" type="password" autocomplete="current-password" required>
|
||||||
|
</label>
|
||||||
|
<button class="win98-button" type="submit">Login</button>
|
||||||
|
</form>
|
||||||
|
{{ else }}
|
||||||
|
<p class="sunken-panel section-body">Account login is disabled. Set bootstrap admin credentials and restart to enable account access.</p>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</main>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{ template "account_toast_modal_containers" . }}
|
||||||
|
<script src="/static/js/account-ui.js"></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
110
templates/account_partials.html
Normal file
110
templates/account_partials.html
Normal file
@@ -0,0 +1,110 @@
|
|||||||
|
{{ define "account_head_assets" }}
|
||||||
|
<link rel="icon" type="image/png" href="/static/WarpBoxLogo.png">
|
||||||
|
<link rel="stylesheet" href="/static/css/account.css">
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
{{ define "account_shell_start" }}
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>{{ if .PageTitle }}{{ .PageTitle }}{{ else }}WarpBox Account{{ end }}</title>
|
||||||
|
{{ template "account_head_assets" . }}
|
||||||
|
</head>
|
||||||
|
<body class="account-body">
|
||||||
|
<div class="app-shell">
|
||||||
|
<div class="app-frame">
|
||||||
|
{{ template "account_taskbar" . }}
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
{{ define "account_shell_end" }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{ template "account_toast_modal_containers" . }}
|
||||||
|
<script src="/static/js/account-ui.js"></script>
|
||||||
|
{{ range .PageScripts }}
|
||||||
|
<script src="{{ . }}"></script>
|
||||||
|
{{ end }}
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
{{ define "account_taskbar" }}
|
||||||
|
{{ $nav := .AccountNav }}
|
||||||
|
<header class="top-taskbar" aria-label="Account navigation">
|
||||||
|
<a class="start-button" href="/account">
|
||||||
|
<span class="start-logo">W</span>
|
||||||
|
<span>WarpBox</span>
|
||||||
|
</a>
|
||||||
|
|
||||||
|
<nav class="taskbar-nav" aria-label="Primary">
|
||||||
|
<a class="taskbar-button{{ if eq $nav.ActiveSection "dashboard" }} is-active{{ end }}" href="/account">Dashboard</a>
|
||||||
|
{{ if $nav.CanViewBoxes }}
|
||||||
|
<a class="taskbar-button{{ if eq $nav.ActiveSection "boxes" }} is-active{{ end }}" href="/account/boxes">Boxes</a>
|
||||||
|
{{ end }}
|
||||||
|
{{ if $nav.CanViewAlerts }}
|
||||||
|
<a class="taskbar-button{{ if eq $nav.ActiveSection "alerts" }} is-active{{ end }}" href="/account/alerts">Alerts</a>
|
||||||
|
{{ end }}
|
||||||
|
{{ if $nav.CanViewUsers }}
|
||||||
|
<a class="taskbar-button{{ if eq $nav.ActiveSection "users" }} is-active{{ end }}" href="/account/users">Users</a>
|
||||||
|
{{ end }}
|
||||||
|
{{ if $nav.CanViewAPIKeys }}
|
||||||
|
<a class="taskbar-button{{ if eq $nav.ActiveSection "api-keys" }} is-active{{ end }}" href="/account/api-keys">API Keys</a>
|
||||||
|
{{ end }}
|
||||||
|
{{ if $nav.CanViewSettings }}
|
||||||
|
<a class="taskbar-button{{ if eq $nav.ActiveSection "settings" }} is-active{{ end }}" href="/account/settings">Settings</a>
|
||||||
|
{{ end }}
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<div class="taskbar-session" aria-label="Current session summary">
|
||||||
|
{{ if gt $nav.AlertCount 0 }}
|
||||||
|
<a class="alert-chip is-{{ $nav.AlertSeverity }}" href="/account/alerts">! {{ $nav.AlertCount }} alerts</a>
|
||||||
|
{{ else }}
|
||||||
|
<span class="alert-chip is-ok">0 alerts</span>
|
||||||
|
{{ end }}
|
||||||
|
<span class="session-chip">signed in: {{ $nav.Username }}</span>
|
||||||
|
{{ if $nav.IsAdmin }}
|
||||||
|
<span class="session-chip">admin</span>
|
||||||
|
{{ else }}
|
||||||
|
<span class="session-chip">account</span>
|
||||||
|
{{ end }}
|
||||||
|
<span class="dirty-chip" data-dirty-chip></span>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
{{ define "account_window_titlebar" }}
|
||||||
|
<div class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<span class="win98-titlebar-icon">{{ if .WindowIcon }}{{ .WindowIcon }}{{ else }}W{{ end }}</span>
|
||||||
|
<h1>{{ if .WindowTitle }}{{ .WindowTitle }}{{ else }}WarpBox Account Control Panel{{ end }}</h1>
|
||||||
|
</div>
|
||||||
|
<div class="win98-window-controls" aria-hidden="true">
|
||||||
|
<button class="win98-control" type="button">_</button>
|
||||||
|
<button class="win98-control" type="button">[]</button>
|
||||||
|
<button class="win98-control" type="button">x</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
{{ define "account_csrf_field" }}
|
||||||
|
<input type="hidden" name="csrf_token" value="{{ .CSRFToken }}">
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
{{ define "account_toast_modal_containers" }}
|
||||||
|
<div class="toast" id="account-toast" role="status" aria-live="polite"></div>
|
||||||
|
<div class="modal-backdrop" id="account-modal-backdrop" aria-hidden="true"></div>
|
||||||
|
<section class="account-modal win98-window" id="account-modal" role="dialog" aria-modal="true" aria-labelledby="account-modal-title">
|
||||||
|
<div class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<span class="win98-titlebar-icon">W</span>
|
||||||
|
<h2 id="account-modal-title">WarpBox</h2>
|
||||||
|
</div>
|
||||||
|
<div class="win98-window-controls">
|
||||||
|
<button class="win98-control" type="button" data-modal-close aria-label="Close">x</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="modal-body sunken-panel" id="account-modal-body"></div>
|
||||||
|
</section>
|
||||||
|
{{ end }}
|
||||||
134
templates/account_settings.html
Normal file
134
templates/account_settings.html
Normal file
@@ -0,0 +1,134 @@
|
|||||||
|
{{ template "account_shell_start" . }}
|
||||||
|
<main class="account-window" aria-labelledby="account-settings-title">
|
||||||
|
{{ template "account_window_titlebar" . }}
|
||||||
|
|
||||||
|
<nav class="menu-bar" aria-label="Settings toolbar">
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">File</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
<a class="menu-action" href="/account/settings"><span>R</span><span>Refresh settings</span><span></span></a>
|
||||||
|
<a class="menu-action" href="/account/settings/export.json"><span>E</span><span>Export JSON</span><span></span></a>
|
||||||
|
<div class="menu-separator"></div>
|
||||||
|
<form action="/account/logout" method="post">
|
||||||
|
{{ template "account_csrf_field" . }}
|
||||||
|
<button class="menu-action" type="submit"><span>Q</span><span>Log out</span><span></span></button>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="menu-item">
|
||||||
|
<button class="menu-button" type="button" aria-expanded="false">View</button>
|
||||||
|
<div class="menu-popup" role="menu">
|
||||||
|
{{ range .Groups }}
|
||||||
|
<a class="menu-action" href="#settings-{{ .Key }}"><span>S</span><span>{{ .Label }}</span><span></span></a>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</nav>
|
||||||
|
|
||||||
|
<form class="settings-layout account-body-content" action="/account/settings" method="post">
|
||||||
|
{{ template "account_csrf_field" . }}
|
||||||
|
|
||||||
|
<section class="settings-summary raised-panel" aria-label="Settings status">
|
||||||
|
{{ if .Error }}<span class="badge is-danger">{{ .Error }}</span>{{ end }}
|
||||||
|
{{ if .Notice }}<span class="badge is-ok">{{ .Notice }}</span>{{ end }}
|
||||||
|
{{ if .OverridesAllowed }}
|
||||||
|
<span class="badge is-ok">overrides enabled</span>
|
||||||
|
{{ else }}
|
||||||
|
<span class="badge is-warning">read-only: overrides disabled</span>
|
||||||
|
{{ end }}
|
||||||
|
<a class="tiny-button" href="/account/settings/export.json">Export JSON</a>
|
||||||
|
<button class="tiny-button" type="button" data-settings-import-toggle>Import JSON</button>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<section class="settings-import raised-panel" data-settings-import-panel hidden>
|
||||||
|
<label class="account-form-row">
|
||||||
|
<span>Settings backup JSON</span>
|
||||||
|
<textarea class="account-control" rows="5" data-settings-import-json></textarea>
|
||||||
|
</label>
|
||||||
|
<button class="win98-button" type="button" data-settings-import-submit {{ if not .CanEdit }}disabled{{ end }}>Import</button>
|
||||||
|
</section>
|
||||||
|
|
||||||
|
<div class="settings-scroll scroll-panel" aria-label="Grouped settings">
|
||||||
|
{{ range .Groups }}
|
||||||
|
<section class="settings-group" id="settings-{{ .Key }}">
|
||||||
|
<header class="settings-group-header">
|
||||||
|
<h2>{{ .Label }}</h2>
|
||||||
|
<p>{{ .Description }}</p>
|
||||||
|
</header>
|
||||||
|
<table class="account-table settings-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>Setting</th>
|
||||||
|
<th>Description</th>
|
||||||
|
<th>Value</th>
|
||||||
|
<th>Source</th>
|
||||||
|
<th>Reset</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{{ range .Rows }}
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
<strong>{{ .Label }}</strong>
|
||||||
|
<span class="setting-key">{{ .Key }}</span>
|
||||||
|
</td>
|
||||||
|
<td><p class="setting-description">{{ .Description }}</p></td>
|
||||||
|
<td>
|
||||||
|
{{ if .Editable }}
|
||||||
|
{{ if eq .Type "bool" }}
|
||||||
|
<label class="account-checks"><span><input type="checkbox" name="{{ .Key }}" value="true" {{ if eq .Value "true" }}checked{{ end }}> enabled</span></label>
|
||||||
|
{{ else }}
|
||||||
|
<input class="account-control" name="{{ .Key }}" value="{{ .Value }}" inputmode="numeric">
|
||||||
|
<span class="setting-key">{{ .DisplayValue }}</span>
|
||||||
|
{{ end }}
|
||||||
|
{{ else }}
|
||||||
|
<span>{{ .DisplayValue }}</span>
|
||||||
|
{{ if .LockedReason }}<span class="setting-key">{{ .LockedReason }}</span>{{ end }}
|
||||||
|
{{ end }}
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<span class="setting-source">
|
||||||
|
<span class="badge is-info">{{ .Source }}</span>
|
||||||
|
<span class="setting-env">{{ .EnvName }}</span>
|
||||||
|
</span>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
{{ if .Editable }}
|
||||||
|
<button class="tiny-button" type="submit" form="reset-{{ .Key }}">Reset</button>
|
||||||
|
{{ else }}
|
||||||
|
<span class="badge">locked</span>
|
||||||
|
{{ end }}
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{{ else }}
|
||||||
|
<tr><td colspan="5">No settings in this group.</td></tr>
|
||||||
|
{{ end }}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</section>
|
||||||
|
{{ end }}
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<section class="settings-actions raised-panel" aria-label="Settings actions">
|
||||||
|
<button class="win98-button" type="submit" {{ if not .CanEdit }}disabled{{ end }}>Save Settings</button>
|
||||||
|
</section>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
{{ range .Groups }}
|
||||||
|
{{ range .Rows }}
|
||||||
|
{{ if .Editable }}
|
||||||
|
<form id="reset-{{ .Key }}" action="/account/settings/reset" method="post" hidden>
|
||||||
|
{{ template "account_csrf_field" $ }}
|
||||||
|
<input type="hidden" name="key" value="{{ .Key }}">
|
||||||
|
</form>
|
||||||
|
{{ end }}
|
||||||
|
{{ end }}
|
||||||
|
{{ end }}
|
||||||
|
|
||||||
|
<footer class="win98-statusbar" aria-label="Settings status">
|
||||||
|
<span>settings</span>
|
||||||
|
<span>{{ if .CanEdit }}editable{{ else }}read-only{{ end }}</span>
|
||||||
|
<span>ready</span>
|
||||||
|
</footer>
|
||||||
|
</main>
|
||||||
|
{{ template "account_shell_end" . }}
|
||||||
40
templates/admin.html
Normal file
40
templates/admin.html
Normal file
@@ -0,0 +1,40 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<title>WarpBox Admin</title>
|
||||||
|
<link rel="icon" type="image/png" href="/static/WarpBoxLogo.png">
|
||||||
|
<link rel="stylesheet" href="/static/css/app.css">
|
||||||
|
<link rel="stylesheet" href="/static/css/window.css">
|
||||||
|
<link rel="stylesheet" href="/static/css/admin.css">
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<main>
|
||||||
|
<section class="win98-window admin-window" aria-labelledby="admin-title">
|
||||||
|
<header class="win98-titlebar">
|
||||||
|
<div class="win98-titlebar-label">
|
||||||
|
<img class="win98-titlebar-icon" src="/static/WarpBoxLogo.png" alt="" aria-hidden="true">
|
||||||
|
<h1 id="admin-title">WarpBox Admin</h1>
|
||||||
|
</div>
|
||||||
|
</header>
|
||||||
|
<div class="win98-panel admin-panel">
|
||||||
|
<nav class="admin-nav">
|
||||||
|
<span>Signed in as {{ .CurrentUser }}</span>
|
||||||
|
<span class="admin-spacer"></span>
|
||||||
|
<form action="/admin/logout" method="post">
|
||||||
|
<input type="hidden" name="csrf_token" value="{{ .CSRFToken }}">
|
||||||
|
<button class="win98-button" type="submit">Logout</button>
|
||||||
|
</form>
|
||||||
|
</nav>
|
||||||
|
<div class="admin-grid">
|
||||||
|
<a class="win98-panel admin-link" href="/admin/boxes"><strong>Boxes</strong></a>
|
||||||
|
<a class="win98-panel admin-link" href="/admin/users"><strong>Users</strong></a>
|
||||||
|
<a class="win98-panel admin-link" href="/admin/tags"><strong>Tags</strong></a>
|
||||||
|
<a class="win98-panel admin-link" href="/admin/settings"><strong>Settings</strong></a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</section>
|
||||||
|
</main>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
@@ -1,92 +0,0 @@
|
|||||||
{{ define "admin/activity.html" }}
|
|
||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="UTF-8">
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
||||||
<title>WarpBox Admin Activity</title>
|
|
||||||
<link rel="icon" type="image/png" href="/static/WarpBoxLogo.png">
|
|
||||||
<link rel="stylesheet" href="/static/css/app.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/window.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/components/buttons.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/components/toast.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/admin.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/activity.css">
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<div class="admin-shell">
|
|
||||||
<div class="admin-frame">
|
|
||||||
{{ template "admin/header.html" . }}
|
|
||||||
<div class="win98-window admin-workspace-window" role="main">
|
|
||||||
<div class="win98-titlebar">
|
|
||||||
<div class="win98-titlebar-label">
|
|
||||||
<img class="win98-titlebar-icon" src="/static/WarpBoxLogo.png" alt="" aria-hidden="true">
|
|
||||||
<h1>WarpBox Activity</h1>
|
|
||||||
</div>
|
|
||||||
<div class="win98-window-controls" aria-hidden="true">
|
|
||||||
<button class="win98-control" type="button">_</button>
|
|
||||||
<button class="win98-control" type="button">□</button>
|
|
||||||
<button class="win98-control" type="button">x</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<nav class="menu-bar" aria-label="Activity toolbar">
|
|
||||||
<div class="menu-item">
|
|
||||||
<button class="menu-button" type="button" aria-expanded="false">File</button>
|
|
||||||
<div class="menu-popup">
|
|
||||||
<button class="menu-action" type="button" data-command="refresh"><span>R</span><span>Refresh list</span><span>F5</span></button>
|
|
||||||
<button class="menu-action" type="button" data-command="export"><span>E</span><span>Export visible JSON</span><span></span></button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
|
|
||||||
<div class="admin-workspace-body activity-page-body">
|
|
||||||
<section class="activity-panel">
|
|
||||||
<div class="activity-toolbar-grid">
|
|
||||||
<input class="activity-input" id="activity-search" type="search" placeholder="Search kind, message, ip, path">
|
|
||||||
<select class="activity-select" id="activity-severity">
|
|
||||||
<option value="all" selected>All severities</option>
|
|
||||||
<option value="high">High</option>
|
|
||||||
<option value="medium">Medium</option>
|
|
||||||
<option value="low">Low</option>
|
|
||||||
</select>
|
|
||||||
<select class="activity-select" id="activity-kind">
|
|
||||||
<option value="all" selected>All kinds</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="activity-table-wrap">
|
|
||||||
<table class="activity-table">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th>Time</th>
|
|
||||||
<th>Kind</th>
|
|
||||||
<th>Severity</th>
|
|
||||||
<th>IP</th>
|
|
||||||
<th>Method</th>
|
|
||||||
<th>Path</th>
|
|
||||||
<th>Message</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody id="activity-body"></tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<footer class="status-bar admin-dashboard-statusbar">
|
|
||||||
<span id="activity-status-left">{{ len .Events }} events loaded</span>
|
|
||||||
<span id="activity-status-middle">retention from settings</span>
|
|
||||||
<span id="activity-status-right">admin only</span>
|
|
||||||
</footer>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="toast" id="toast" role="status" aria-live="polite"></div>
|
|
||||||
<script id="activity-data" type="application/json">{{ toJSON .Events }}</script>
|
|
||||||
<script src="/static/js/warpbox-ui.js"></script>
|
|
||||||
<script src="/static/js/admin/activity.js"></script>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
{{ end }}
|
|
||||||
@@ -1,223 +0,0 @@
|
|||||||
{{ define "admin/alerts.html" }}
|
|
||||||
<!DOCTYPE html>
|
|
||||||
<html lang="en">
|
|
||||||
<head>
|
|
||||||
<meta charset="UTF-8">
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1">
|
|
||||||
<title>WarpBox Admin Alerts</title>
|
|
||||||
<link rel="icon" type="image/png" href="/static/WarpBoxLogo.png">
|
|
||||||
<link rel="stylesheet" href="/static/css/app.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/window.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/components/buttons.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/components/toast.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/admin.css">
|
|
||||||
<link rel="stylesheet" href="/static/css/alerts.css">
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<div class="admin-shell">
|
|
||||||
<div class="admin-frame">
|
|
||||||
{{ template "admin/header.html" . }}
|
|
||||||
|
|
||||||
<div class="win98-window admin-workspace-window" role="main">
|
|
||||||
<div class="win98-titlebar">
|
|
||||||
<div class="win98-titlebar-label">
|
|
||||||
<img class="win98-titlebar-icon" src="/static/WarpBoxLogo.png" alt="" aria-hidden="true">
|
|
||||||
<h1>WarpBox Alerts</h1>
|
|
||||||
</div>
|
|
||||||
<div class="win98-window-controls" aria-hidden="true">
|
|
||||||
<button class="win98-control" type="button">_</button>
|
|
||||||
<button class="win98-control" type="button">□</button>
|
|
||||||
<button class="win98-control" type="button">x</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<nav class="menu-bar" aria-label="Alerts toolbar">
|
|
||||||
<div class="menu-item">
|
|
||||||
<button class="menu-button" type="button" aria-expanded="false">File</button>
|
|
||||||
<div class="menu-popup">
|
|
||||||
<button class="menu-action" type="button" data-command="refresh"><span>R</span><span>Refresh alerts</span><span class="shortcut">F5</span></button>
|
|
||||||
<button class="menu-action" type="button" data-command="export"><span>E</span><span>Export visible alerts</span><span></span></button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="menu-item">
|
|
||||||
<button class="menu-button" type="button" aria-expanded="false">Alerts</button>
|
|
||||||
<div class="menu-popup">
|
|
||||||
<button class="menu-action" type="button" data-command="ack"><span>A</span><span>Acknowledge selected</span><span></span></button>
|
|
||||||
<button class="menu-action" type="button" data-command="close"><span>C</span><span>Close selected</span><span></span></button>
|
|
||||||
<div class="menu-separator"></div>
|
|
||||||
<button class="menu-action" type="button" data-command="open-only"><span>O</span><span>Show open only</span><span></span></button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="menu-item">
|
|
||||||
<button class="menu-button" type="button" aria-expanded="false">Help</button>
|
|
||||||
<div class="menu-popup">
|
|
||||||
<button class="menu-action" type="button" data-command="help-codes"><span>?</span><span>Tracing and codes</span><span></span></button>
|
|
||||||
<button class="menu-action" type="button" data-command="help-meta"><span>I</span><span>Metadata preview</span><span></span></button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</nav>
|
|
||||||
|
|
||||||
<div class="admin-workspace-body alerts-page-body">
|
|
||||||
<section class="alerts-summary-grid" aria-label="Alerts summary">
|
|
||||||
<article class="alerts-stat-card is-danger">
|
|
||||||
<p class="alerts-stat-label">Open alerts</p>
|
|
||||||
<p class="alerts-stat-value" data-open-count>{{ .OpenCount }}</p>
|
|
||||||
<p class="alerts-stat-note">Requires attention</p>
|
|
||||||
</article>
|
|
||||||
<article class="alerts-stat-card is-warning">
|
|
||||||
<p class="alerts-stat-label">High severity</p>
|
|
||||||
<p class="alerts-stat-value" data-high-count>{{ .HighCount }}</p>
|
|
||||||
<p class="alerts-stat-note">Escalate first</p>
|
|
||||||
</article>
|
|
||||||
<article class="alerts-stat-card is-info">
|
|
||||||
<p class="alerts-stat-label">Acknowledged</p>
|
|
||||||
<p class="alerts-stat-value" data-ack-count>{{ .AckCount }}</p>
|
|
||||||
<p class="alerts-stat-note">Seen but not closed</p>
|
|
||||||
</article>
|
|
||||||
<article class="alerts-stat-card is-info">
|
|
||||||
<p class="alerts-stat-label">Closed today</p>
|
|
||||||
<p class="alerts-stat-value" data-closed-count>{{ .ClosedCount }}</p>
|
|
||||||
<p class="alerts-stat-note">History stays lightweight</p>
|
|
||||||
</article>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<section class="alerts-content-grid">
|
|
||||||
<div class="alerts-column">
|
|
||||||
<section class="alerts-panel alerts-list-panel">
|
|
||||||
<div class="alerts-panel-header">
|
|
||||||
<div class="alerts-panel-title">Alert list <span class="alerts-panel-sub">search, filter, review</span></div>
|
|
||||||
<div class="alerts-panel-tools">
|
|
||||||
<button class="win98-button alerts-tool-button" type="button" data-command="ack">Acknowledge</button>
|
|
||||||
<button class="win98-button alerts-tool-button" type="button" data-command="close">Close</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="alerts-panel-body">
|
|
||||||
<div class="alerts-toolbar-grid">
|
|
||||||
<input class="alerts-input" id="search-input" type="search" placeholder="Search title, code, trace or text">
|
|
||||||
<select class="alerts-select" id="severity-filter">
|
|
||||||
<option value="all" selected>All severities</option>
|
|
||||||
<option value="low">Low</option>
|
|
||||||
<option value="medium">Medium</option>
|
|
||||||
<option value="high">High</option>
|
|
||||||
</select>
|
|
||||||
<select class="alerts-select" id="status-filter">
|
|
||||||
<option value="all" selected>All statuses</option>
|
|
||||||
<option value="open">Open</option>
|
|
||||||
<option value="acked">Acknowledged</option>
|
|
||||||
<option value="closed">Closed</option>
|
|
||||||
</select>
|
|
||||||
<select class="alerts-select" id="source-filter">
|
|
||||||
<option value="all" selected>All groups</option>
|
|
||||||
<option value="thumbnails">Thumbnails</option>
|
|
||||||
<option value="storage">Storage</option>
|
|
||||||
<option value="uploads">Uploads</option>
|
|
||||||
<option value="auth">Auth</option>
|
|
||||||
</select>
|
|
||||||
<select class="alerts-select" id="sort-filter">
|
|
||||||
<option value="newest" selected>Newest first</option>
|
|
||||||
<option value="severity">Severity first</option>
|
|
||||||
<option value="oldest">Oldest first</option>
|
|
||||||
</select>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="alerts-table-wrap">
|
|
||||||
<table class="alerts-table">
|
|
||||||
<thead>
|
|
||||||
<tr>
|
|
||||||
<th class="alerts-col-check"><input type="checkbox" id="select-all"></th>
|
|
||||||
<th>Title</th>
|
|
||||||
<th class="alerts-col-severity">Severity</th>
|
|
||||||
<th class="alerts-col-status">Status</th>
|
|
||||||
<th class="alerts-col-code">Code</th>
|
|
||||||
<th>Trace</th>
|
|
||||||
<th class="alerts-col-time">Created</th>
|
|
||||||
<th class="alerts-col-actions">Actions</th>
|
|
||||||
</tr>
|
|
||||||
</thead>
|
|
||||||
<tbody id="alerts-body"></tbody>
|
|
||||||
</table>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="alerts-column alerts-column-side">
|
|
||||||
<section class="alerts-panel">
|
|
||||||
<div class="alerts-panel-header">
|
|
||||||
<div class="alerts-panel-title">Alert details <span class="alerts-panel-sub">selected alert preview</span></div>
|
|
||||||
</div>
|
|
||||||
<div class="alerts-panel-body">
|
|
||||||
<ul class="alerts-info-list">
|
|
||||||
<li class="alerts-info-item"><strong>Title</strong><span id="detail-title">Storage connector unavailable</span></li>
|
|
||||||
<li class="alerts-info-item"><strong>Severity</strong><span id="detail-severity">high</span></li>
|
|
||||||
<li class="alerts-info-item"><strong>Status</strong><span id="detail-status">open</span></li>
|
|
||||||
<li class="alerts-info-item"><strong>Code</strong><span id="detail-code">301</span></li>
|
|
||||||
<li class="alerts-info-item"><strong>Trace</strong><span id="detail-trace">storage.connector.health_failed</span></li>
|
|
||||||
<li class="alerts-info-item"><strong>Created</strong><span id="detail-time">today 14:08</span></li>
|
|
||||||
<li class="alerts-info-item"><strong>Description</strong><span id="detail-description">Primary local storage connector failed health check and new writes are paused.</span></li>
|
|
||||||
</ul>
|
|
||||||
<div class="alerts-mini-note">
|
|
||||||
TO-DO: later, limited alert access should only show alerts scoped to the user’s permissions, tags, or groups.
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<section class="alerts-panel">
|
|
||||||
<div class="alerts-panel-header">
|
|
||||||
<div class="alerts-panel-title">Metadata <span class="alerts-panel-sub">simple JSON preview</span></div>
|
|
||||||
<div class="alerts-panel-tools">
|
|
||||||
<button class="win98-button alerts-tool-button" type="button" data-command="copy-meta">Copy</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
<div class="alerts-panel-body">
|
|
||||||
<pre class="alerts-json-box" id="detail-metadata">{
|
|
||||||
"connector": "local-main",
|
|
||||||
"mode": "read_only",
|
|
||||||
"retry_in": "30s"
|
|
||||||
}</pre>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
|
|
||||||
<section class="alerts-panel alerts-actions-panel">
|
|
||||||
<div class="alerts-panel-header">
|
|
||||||
<div class="alerts-panel-title">Actions <span class="alerts-panel-sub">simple first version</span></div>
|
|
||||||
</div>
|
|
||||||
<div class="alerts-panel-body">
|
|
||||||
<div class="alerts-action-stack">
|
|
||||||
<button class="win98-button alerts-action-button" type="button" data-command="ack">Acknowledge selected</button>
|
|
||||||
<button class="win98-button alerts-action-button" type="button" data-command="close">Close selected</button>
|
|
||||||
<button class="win98-button alerts-action-button" type="button" data-command="delete">Delete selected</button>
|
|
||||||
<button class="win98-button alerts-action-button" type="button" data-command="refresh">Refresh alerts</button>
|
|
||||||
</div>
|
|
||||||
<div class="alerts-mini-note">
|
|
||||||
Alerts persist until deleted. Acknowledge and close update state; delete removes permanently.
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
</section>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="alerts-footerbar">
|
|
||||||
<div class="alerts-footer-left">
|
|
||||||
<span class="alerts-status-pill" id="selected-count">Selected: 0</span>
|
|
||||||
<span class="alerts-status-pill" id="alerts-total-pill">{{ len .Alerts }} alerts</span>
|
|
||||||
</div>
|
|
||||||
<div class="alerts-footer-right">
|
|
||||||
<button class="win98-button alerts-footer-button" type="button" data-command="ack">Acknowledge</button>
|
|
||||||
<button class="win98-button alerts-footer-button" type="button" data-command="close">Close</button>
|
|
||||||
<button class="win98-button alerts-footer-button" type="button" data-command="delete">Delete</button>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
|
|
||||||
<div class="toast" id="toast" role="status" aria-live="polite"></div>
|
|
||||||
|
|
||||||
<script id="alerts-data" type="application/json">{{ toJSON .Alerts }}</script>
|
|
||||||
<script src="/static/js/warpbox-ui.js"></script>
|
|
||||||
<script src="/static/js/admin/alerts.js"></script>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
{{ end }}
|
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user