Agent-almanac deploy-shinyproxy
install
source · Clone the upstream repo
git clone https://github.com/pjt222/agent-almanac
Claude Code · Install into ~/.claude/skills/
T=$(mktemp -d) && git clone --depth=1 https://github.com/pjt222/agent-almanac "$T" && mkdir -p ~/.claude/skills && cp -r "$T/i18n/zh-CN/skills/deploy-shinyproxy" ~/.claude/skills/pjt222-agent-almanac-deploy-shinyproxy-ac3860 && rm -rf "$T"
manifest:
i18n/zh-CN/skills/deploy-shinyproxy/SKILL.mdsource content
部署 ShinyProxy
部署 ShinyProxy 以托管多个容器化 Shiny 应用程序,并提供认证和使用追踪功能。
适用场景
- 在单一入口点后托管多个 Shiny 应用
- 需要按应用认证和访问控制
- 将 Shiny 应用作为隔离 Docker 容器部署
- 扩展超出单应用部署范围(shinyapps.io 或独立 Docker)
- 需要使用分析和审计日志
输入
- 必需:一个或多个待部署的 Shiny 应用
- 必需:安装了 Docker 的服务器
- 可选:认证提供商(LDAP、OpenID、社交登录)
- 可选:域名和 SSL 证书
- 可选:容器编排器(Docker 或 Kubernetes)
步骤
第 1 步:创建 Shiny 应用 Docker 镜像
每个 Shiny 应用需要自己的 Docker 镜像。Shiny 应用的示例
Dockerfile:
FROM rocker/shiny:4.5.0 RUN apt-get update && apt-get install -y \ libcurl4-openssl-dev \ libssl-dev \ && rm -rf /var/lib/apt/lists/* RUN R -e "install.packages(c('shiny', 'bslib', 'DT', 'dplyr'), \ repos='https://cloud.r-project.org/')" COPY app/ /srv/shiny-server/app/ RUN chown -R shiny:shiny /srv/shiny-server/app USER shiny EXPOSE 3838 CMD ["R", "-e", "shiny::runApp('/srv/shiny-server/app', host='0.0.0.0', port=3838)"]
构建并测试每个应用:
docker build -t myorg/dashboard:latest ./apps/dashboard/ docker run --rm -p 3838:3838 myorg/dashboard:latest
预期结果: 每个 Shiny 应用在其自己的容器中独立运行。
第 2 步:配置 ShinyProxy
application.yml:
proxy: title: "Shiny Applications" port: 8080 container-backend: docker docker: internal-networking: true authentication: simple admin-groups: admins users: - name: admin password: admin_password groups: admins - name: analyst password: analyst_password groups: users specs: - id: dashboard display-name: "Analytics Dashboard" description: "Interactive data analysis dashboard" container-image: myorg/dashboard:latest container-cmd: ["R", "-e", "shiny::runApp('/srv/shiny-server/app', host='0.0.0.0', port=3838)"] container-network: shinyproxy-net port: 3838 access-groups: [admins, users] - id: report-builder display-name: "Report Builder" description: "Generate custom reports" container-image: myorg/report-builder:latest container-cmd: ["R", "-e", "shiny::runApp('/srv/shiny-server/app', host='0.0.0.0', port=3838)"] container-network: shinyproxy-net port: 3838 access-groups: [admins] logging: file: name: /opt/shinyproxy/log/shinyproxy.log server: forward-headers-strategy: native
第 3 步:使用 Docker Compose 部署 ShinyProxy
docker-compose.yml:
services: shinyproxy: image: openanalytics/shinyproxy:3.1.1 container_name: shinyproxy ports: - "8080:8080" volumes: - ./application.yml:/opt/shinyproxy/application.yml:ro - /var/run/docker.sock:/var/run/docker.sock - shinyproxy-logs:/opt/shinyproxy/log networks: - shinyproxy-net restart: unless-stopped networks: shinyproxy-net: name: shinyproxy-net driver: bridge volumes: shinyproxy-logs:
# Create the network first (ShinyProxy spawns containers on this network) docker network create shinyproxy-net # Start ShinyProxy docker compose up -d # Check logs docker compose logs -f shinyproxy
预期结果: ShinyProxy 在端口 8080 上启动,显示登录页面,并列出已配置的应用。
失败处理: 检查
docker compose logs shinyproxy。验证应用镜像本地可用(docker images)。
第 4 步:配置认证
简单认证(内置)
如第 2 步所示,使用
authentication: simple 和内联用户。
LDAP
proxy: authentication: ldap ldap: url: ldap://ldap.example.com:389/dc=example,dc=com manager-dn: cn=admin,dc=example,dc=com manager-password: ldap_admin_password user-search-base: ou=users user-search-filter: (uid={0}) group-search-base: ou=groups group-search-filter: (member={0})
OpenID Connect(Keycloak、Auth0 等)
proxy: authentication: openid openid: auth-url: https://auth.example.com/realms/myrealm/protocol/openid-connect/auth token-url: https://auth.example.com/realms/myrealm/protocol/openid-connect/token jwks-url: https://auth.example.com/realms/myrealm/protocol/openid-connect/certs client-id: shinyproxy client-secret: your_client_secret roles-claim: realm_access.roles
第 5 步:使用 Nginx 添加反向代理
生产环境中,在 ShinyProxy 前放置 Nginx:
map $http_upgrade $connection_upgrade { default upgrade; '' close; } server { listen 443 ssl; server_name shiny.example.com; ssl_certificate /etc/letsencrypt/live/shiny.example.com/fullchain.pem; ssl_certificate_key /etc/letsencrypt/live/shiny.example.com/privkey.pem; location / { proxy_pass http://shinyproxy:8080; proxy_http_version 1.1; proxy_set_header Upgrade $http_upgrade; proxy_set_header Connection $connection_upgrade; proxy_set_header Host $host; proxy_set_header X-Real-IP $remote_addr; proxy_set_header X-Forwarded-For $proxy_add_x_forwarded_for; proxy_set_header X-Forwarded-Proto $scheme; proxy_read_timeout 600s; proxy_buffering off; } }
WebSocket 支持至关重要——ShinyProxy 和 Shiny 大量使用 WebSocket。
第 6 步:使用追踪
ShinyProxy 将使用事件记录到日志文件。对于结构化追踪,配置 InfluxDB:
proxy: usage-stats-url: http://influxdb:8086/write?db=shinyproxy usage-stats-username: shinyproxy usage-stats-password: stats_password
将 InfluxDB 添加到 compose 堆栈:
services: influxdb: image: influxdb:1.8 environment: INFLUXDB_DB: shinyproxy INFLUXDB_ADMIN_USER: admin INFLUXDB_ADMIN_PASSWORD: admin_password volumes: - influxdata:/var/lib/influxdb networks: - shinyproxy-net volumes: influxdata:
第 7 步:应用资源限制
specs: - id: dashboard container-image: myorg/dashboard:latest container-memory-limit: 1g container-cpu-limit: 1.0 max-instances: 5 container-env: R_MAX_MEM_SIZE: 768m
第 8 步:验证部署
# Check ShinyProxy health curl -s http://localhost:8080/actuator/health # Test login curl -s -c cookies.txt -d "username=admin&password=admin_password" \ http://localhost:8080/login # List apps via API curl -s -b cookies.txt http://localhost:8080/api/proxyspec
预期结果: 健康端点返回
UP。登录成功。应用在隔离容器中启动。
验证清单
- ShinyProxy 启动并显示登录页面
- 所有配置用户的认证正常
- 每个 Shiny 应用在其自己的容器中启动
- WebSocket 连接正常(Shiny 响应性功能正常)
- 访问组正确限制应用可见性
- 用户断开连接时容器清理正常
- 日志捕获使用事件
常见问题
- Docker socket 权限:ShinyProxy 需要 Docker socket 访问来启动容器。以
组中的用户运行或挂载 socket。docker - 网络不匹配:应用容器必须与 ShinyProxy 在同一 Docker 网络上(spec 中的
必须匹配)。container-network - WebSocket 代理:ShinyProxy 前的 Nginx 或其他代理必须转发 WebSocket 升级头。
- 镜像未找到:应用镜像必须在 ShinyProxy 尝试使用之前在 Docker 主机上拉取或构建。
- 容器清理:如果 ShinyProxy 崩溃,可能会留下孤立的应用容器。使用
检查并清理。docker ps - 内存限制:Shiny 应用可能消耗大量内存。设置
以防止单个应用耗尽其他应用的资源。container-memory-limit
相关技能
— 单应用部署到 shinyapps.io、Posit Connect 或 Dockerdeploy-shiny-app
— 包括 WebSocket 代理的反向代理模式configure-reverse-proxy
— 应用镜像的通用 Dockerfile 创建create-dockerfile
— 使用 rocker 镜像的 R 专用 Dockerfilescreate-r-dockerfile