Files
Arch1Panel/skills/tests/test_skills_scripts.sh
T

362 lines
11 KiB
Bash
Raw Normal View History

2026-05-26 01:07:55 +08:00
#!/usr/bin/env bash
set -euo pipefail
ROOT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")/../.." && pwd)"
TMP_DIR="$(mktemp -d)"
trap 'rm -rf "$TMP_DIR"' EXIT
fail() {
echo "FAIL: $*" >&2
exit 1
}
assert_file() {
[[ -f "$1" ]] || fail "expected file: $1"
}
assert_not_file() {
[[ ! -f "$1" ]] || fail "expected missing file: $1"
}
assert_contains() {
local file="$1"
local pattern="$2"
grep -qE -- "$pattern" "$file" || fail "expected pattern '$pattern' in $file"
}
2026-05-26 23:08:37 +08:00
make_fake_path_without() {
local dir="$1"
shift
mkdir -p "$dir"
local tool
for tool in awk basename cat chmod cmp cp dirname env find grep head mkdir mktemp mv printf rm sed sort tail tr; do
if command -v "$tool" >/dev/null 2>&1; then
ln -s "$(command -v "$tool")" "$dir/$tool"
fi
done
}
2026-05-26 01:07:55 +08:00
run_test() {
local name="$1"
shift
echo "== $name =="
"$@"
}
2026-05-26 23:08:37 +08:00
test_generate_app_reports_missing_dependencies() {
local fake_path="$TMP_DIR/fake-path"
make_fake_path_without "$fake_path"
if PATH="$fake_path" /usr/bin/bash "$ROOT_DIR/skills/scripts/generate-app.sh" --check-deps >/tmp/generate_missing_deps.out 2>&1; then
fail "dependency check should fail when yq/jq/curl are missing"
fi
assert_contains /tmp/generate_missing_deps.out "缺少依赖"
assert_contains /tmp/generate_missing_deps.out "yq"
assert_contains /tmp/generate_missing_deps.out "jq"
assert_contains /tmp/generate_missing_deps.out "curl"
}
2026-05-26 01:07:55 +08:00
test_download_icon_skip_mode_does_not_create_logo() {
local out="$TMP_DIR/skip/logo.png"
bash "$ROOT_DIR/skills/scripts/download-icon.sh" --mode skip demo-app "$out" >/tmp/download_icon_skip.out
assert_not_file "$out"
assert_contains /tmp/download_icon_skip.out "跳过图标下载"
}
test_download_icon_cache_only_uses_cached_logo() {
local cache_dir="$TMP_DIR/icon-cache"
local out="$TMP_DIR/cache/logo.png"
mkdir -p "$cache_dir"
printf 'cached-logo' > "$cache_dir/demo-app.png"
bash "$ROOT_DIR/skills/scripts/download-icon.sh" --mode cache-only --cache-dir "$cache_dir" demo-app "$out" >/tmp/download_icon_cache.out
assert_file "$out"
cmp "$cache_dir/demo-app.png" "$out" || fail "cached logo was not copied"
}
test_generate_app_accepts_explicit_options_and_skips_icon() {
local compose="$TMP_DIR/compose.yml"
local output="$TMP_DIR/generated"
cat > "$compose" <<'YAML'
services:
demo:
image: nginx:1.25.3
ports:
- "18080:80"
YAML
bash "$ROOT_DIR/skills/scripts/generate-app.sh" \
--app-key demo-app \
--name DemoApp \
--version 1.25.3 \
--output "$output" \
--icon-mode skip \
"$compose" >/tmp/generate_app_options.out
assert_file "$output/demo-app/latest/docker-compose.yml"
assert_file "$output/demo-app/1.25.3/docker-compose.yml"
assert_contains "$output/demo-app/data.yml" "^name: DemoApp$"
assert_contains "$output/demo-app/latest/docker-compose.yml" "image: nginx:latest"
assert_contains "$output/demo-app/1.25.3/docker-compose.yml" "image: nginx:1.25.3"
assert_not_file "$output/demo-app/logo.png"
}
test_generate_app_preserves_compose_environment_and_volumes() {
local compose="$TMP_DIR/compose-preserve.yml"
local output="$TMP_DIR/generated-preserve"
cat > "$compose" <<'YAML'
services:
demo:
image: nginx:1.25.3
ports:
- "18080:80"
volumes:
- ./data/nginx:/etc/nginx/conf.d
environment:
- TZ=Asia/Shanghai
- DEMO_FLAG=true
YAML
bash "$ROOT_DIR/skills/scripts/generate-app.sh" \
--app-key demo-preserve \
--name DemoPreserve \
--version 1.25.3 \
--output "$output" \
--icon-mode skip \
"$compose" >/tmp/generate_app_preserve.out
assert_contains "$output/demo-preserve/1.25.3/docker-compose.yml" "./data/nginx:/etc/nginx/conf.d"
assert_contains "$output/demo-preserve/1.25.3/docker-compose.yml" "TZ=Asia/Shanghai"
assert_contains "$output/demo-preserve/1.25.3/docker-compose.yml" "DEMO_FLAG=true"
}
2026-05-26 23:08:37 +08:00
test_generate_app_uses_explicit_compose_service() {
local compose="$TMP_DIR/compose-service.yml"
local output="$TMP_DIR/generated-service"
cat > "$compose" <<'YAML'
services:
db:
image: postgres:16
ports:
- "15432:5432"
web:
image: ghcr.io/example/webapp:2.3.4
ports:
- "18080:8080"
environment:
- WEB_MODE=prod
YAML
bash "$ROOT_DIR/skills/scripts/generate-app.sh" \
--service web \
--app-key demo-service \
--name DemoService \
--version 2.3.4 \
--output "$output" \
--icon-mode skip \
"$compose" >/tmp/generate_app_service.out
assert_contains "$output/demo-service/2.3.4/docker-compose.yml" "web:"
assert_contains "$output/demo-service/2.3.4/docker-compose.yml" "image: ghcr.io/example/webapp:2.3.4"
assert_contains "$output/demo-service/2.3.4/docker-compose.yml" "8080"
assert_contains "$output/demo-service/2.3.4/data.yml" "default: 18080"
assert_contains "$output/demo-service/2.3.4/docker-compose.yml" "WEB_MODE=prod"
}
test_generate_app_discovers_github_default_branch_compose() {
local fixture="$TMP_DIR/github-fixture"
local output="$TMP_DIR/generated-github"
mkdir -p "$fixture/api/repos/example" "$fixture/raw/example/demo/master"
cat > "$fixture/api/repos/example/demo" <<'JSON'
{
"name": "demo",
"description": "Demo GitHub app",
"homepage": "https://demo.example",
"default_branch": "master"
}
JSON
cat > "$fixture/raw/example/demo/master/compose.yaml" <<'YAML'
services:
demo:
image: nginx:1.25.3
ports:
- "18080:80"
YAML
GITHUB_API_BASE_URL="file://${fixture}/api/repos" \
GITHUB_RAW_BASE_URL="file://${fixture}/raw" \
bash "$ROOT_DIR/skills/scripts/generate-app.sh" \
--app-key github-demo \
--version 1.25.3 \
--output "$output" \
--icon-mode skip \
"https://github.com/example/demo" >/tmp/generate_app_github.out
assert_contains "$output/github-demo/data.yml" "Demo GitHub app"
assert_contains "$output/github-demo/1.25.3/docker-compose.yml" "image: nginx:1.25.3"
}
test_generate_app_discovers_github_readme_docker_run() {
local fixture="$TMP_DIR/github-readme-fixture"
local output="$TMP_DIR/generated-github-readme"
mkdir -p "$fixture/api/repos/example" "$fixture/raw/example/readme-demo/main"
cat > "$fixture/api/repos/example/readme-demo" <<'JSON'
{
"name": "readme-demo",
"description": "README docker run app",
"homepage": "",
"default_branch": "main"
}
JSON
cat > "$fixture/raw/example/readme-demo/main/README.md" <<'MD'
# README Demo
```bash
docker run -d --name readme-demo -p 18080:80 -e "APP_TITLE=Readme Demo" nginx:1.25.3
```
MD
GITHUB_API_BASE_URL="file://${fixture}/api/repos" \
GITHUB_RAW_BASE_URL="file://${fixture}/raw" \
bash "$ROOT_DIR/skills/scripts/generate-app.sh" \
--app-key github-readme-demo \
--version 1.25.3 \
--output "$output" \
--icon-mode skip \
"https://github.com/example/readme-demo" >/tmp/generate_app_github_readme.out
assert_contains "$output/github-readme-demo/1.25.3/docker-compose.yml" "image: nginx:1.25.3"
assert_contains "$output/github-readme-demo/1.25.3/docker-compose.yml" "APP_TITLE=Readme Demo"
assert_contains "$output/github-readme-demo/1.25.3/data.yml" "default: 18080"
}
test_generate_app_parses_complex_docker_run_flags() {
local output="$TMP_DIR/generated-run"
bash "$ROOT_DIR/skills/scripts/generate-app.sh" \
--app-key docker-run-demo \
--name DockerRunDemo \
--version 1.2.3 \
--output "$output" \
--icon-mode skip \
'docker run -d --name docker-run-demo --env "APP_TITLE=My Demo" --env-file ./data/app.env --publish=127.0.0.1:18080:8080/tcp --volume=./data/demo:/data ghcr.io/example/demo:1.2.3' >/tmp/generate_app_run.out
assert_contains "$output/docker-run-demo/1.2.3/docker-compose.yml" "image: ghcr.io/example/demo:1.2.3"
assert_contains "$output/docker-run-demo/1.2.3/docker-compose.yml" "APP_TITLE=My Demo"
assert_contains "$output/docker-run-demo/1.2.3/docker-compose.yml" "./data/demo:/data"
assert_contains "$output/docker-run-demo/1.2.3/docker-compose.yml" "8080"
}
2026-05-26 01:07:55 +08:00
test_validate_app_rejects_undefined_port_variable() {
local app="$TMP_DIR/bad-port/apps/demo-app"
mkdir -p "$app/1.0.0"
printf 'png' > "$app/logo.png"
cat > "$app/data.yml" <<'YAML'
name: DemoApp
tags:
- 实用工具
title: Demo
description: Demo
additionalProperties:
key: demo-app
name: DemoApp
architectures:
- amd64
YAML
cat > "$app/1.0.0/data.yml" <<'YAML'
additionalProperties:
formFields:
- default: 8080
edit: true
envKey: PANEL_APP_PORT_HTTP
labelEn: Web Port
labelZh: Web端口
required: true
rule: paramPort
type: number
YAML
cat > "$app/1.0.0/docker-compose.yml" <<'YAML'
services:
demo:
container_name: ${CONTAINER_NAME}
restart: always
networks:
- 1panel-network
ports:
- "${PANEL_APP_PORT_ADMIN}:80"
image: nginx:1.0.0
labels:
createdBy: "Apps"
networks:
1panel-network:
external: true
YAML
if bash "$ROOT_DIR/skills/scripts/validate-app.sh" "$app" >/tmp/validate_bad_port.out 2>&1; then
fail "validator should reject undefined port variable"
fi
assert_contains /tmp/validate_bad_port.out "未在版本 data.yml 中定义"
}
test_validate_app_rejects_latest_wrong_tag() {
local app="$TMP_DIR/bad-latest/apps/demo-app"
mkdir -p "$app/latest"
printf 'png' > "$app/logo.png"
cat > "$app/data.yml" <<'YAML'
name: DemoApp
tags:
- 实用工具
title: Demo
description: Demo
additionalProperties:
key: demo-app
name: DemoApp
architectures:
- amd64
YAML
cat > "$app/latest/data.yml" <<'YAML'
additionalProperties:
formFields: []
YAML
cat > "$app/latest/docker-compose.yml" <<'YAML'
services:
demo:
container_name: ${CONTAINER_NAME}
restart: always
networks:
- 1panel-network
image: nginx:1.25.3
labels:
createdBy: "Apps"
networks:
1panel-network:
external: true
YAML
if bash "$ROOT_DIR/skills/scripts/validate-app.sh" "$app" >/tmp/validate_bad_latest.out 2>&1; then
fail "validator should reject latest directory with concrete tag"
fi
assert_contains /tmp/validate_bad_latest.out "latest 目录"
}
run_test "download icon skip mode" test_download_icon_skip_mode_does_not_create_logo
run_test "download icon cache-only mode" test_download_icon_cache_only_uses_cached_logo
2026-05-26 23:08:37 +08:00
run_test "generate app dependency check" test_generate_app_reports_missing_dependencies
2026-05-26 01:07:55 +08:00
run_test "generate app explicit options" test_generate_app_accepts_explicit_options_and_skips_icon
run_test "generate app preserves compose fields" test_generate_app_preserves_compose_environment_and_volumes
2026-05-26 23:08:37 +08:00
run_test "generate app explicit compose service" test_generate_app_uses_explicit_compose_service
run_test "generate app github compose discovery" test_generate_app_discovers_github_default_branch_compose
run_test "generate app github README docker run discovery" test_generate_app_discovers_github_readme_docker_run
run_test "generate app complex docker run flags" test_generate_app_parses_complex_docker_run_flags
2026-05-26 01:07:55 +08:00
run_test "validate undefined port variable" test_validate_app_rejects_undefined_port_variable
run_test "validate latest image tag" test_validate_app_rejects_latest_wrong_tag
echo "All skills script tests passed."