fix: route Claude API through backend instead of nginx envsubst

- Add Claude API proxy in mock-backend.js (reads ANTHROPIC_API_KEY from env)
- Supports SSE streaming via pipe
- Move ANTHROPIC_API_KEY to backend service in docker-compose.demo.yml
- Remove envsubst from entrypoint (no longer needed)
- nginx-demo.conf proxies /aiui/api/claude/ to backend

This fixes the 401 error when Portainer doesn't pass env vars to
nginx correctly — the Node.js backend reads process.env directly.

Co-Authored-By: Claude Opus 4.6 <noreply@anthropic.com>
This commit is contained in:
Dorian 2026-03-07 23:07:38 +00:00
parent 08eb3b61e0
commit f8e5e947ec
4 changed files with 57 additions and 16 deletions

View File

@ -13,6 +13,7 @@ services:
container_name: archy-demo-backend
environment:
VITE_DEV_MODE: "existing" # Skip setup/onboarding, go straight to login
ANTHROPIC_API_KEY: ${ANTHROPIC_API_KEY:-}
expose:
- "5959"
restart: unless-stopped
@ -22,8 +23,6 @@ services:
context: .
dockerfile: neode-ui/Dockerfile.web
container_name: archy-demo-web
environment:
ANTHROPIC_API_KEY: ${ANTHROPIC_API_KEY:-}
ports:
- "4848:80"
depends_on:

View File

@ -1,4 +1,4 @@
#!/bin/sh
# Substitute only ANTHROPIC_API_KEY in nginx config, leave nginx variables untouched
envsubst '${ANTHROPIC_API_KEY}' < /etc/nginx/nginx.conf.template > /etc/nginx/nginx.conf
# Copy nginx config (no envsubst needed — API key is handled by backend)
cp /etc/nginx/nginx.conf.template /etc/nginx/nginx.conf
exec nginx -g 'daemon off;'

View File

@ -78,22 +78,15 @@ http {
}
}
# Proxy Claude API requests from AIUI to Anthropic
# Proxy Claude API requests to backend (which handles API key + streaming)
location /aiui/api/claude/ {
rewrite ^/aiui/api/claude/(.*)$ /$1 break;
proxy_pass https://api.anthropic.com;
proxy_ssl_server_name on;
proxy_set_header Host api.anthropic.com;
proxy_set_header x-api-key "${ANTHROPIC_API_KEY}";
proxy_set_header anthropic-version "2023-06-01";
proxy_set_header Content-Type "application/json";
# SSE streaming support
proxy_pass http://neode-backend:5959;
proxy_http_version 1.1;
proxy_set_header Host $host;
proxy_set_header X-Real-IP $remote_addr;
proxy_buffering off;
proxy_cache off;
proxy_read_timeout 300s;
proxy_http_version 1.1;
proxy_set_header Connection "";
}

View File

@ -1016,6 +1016,55 @@ app.get('/app/filebrowser/api/raw/*', (req, res) => {
}
})
// =============================================================================
// Claude API Proxy (reads ANTHROPIC_API_KEY from environment)
// =============================================================================
import https from 'https'
app.post('/aiui/api/claude/*', (req, res) => {
const apiKey = process.env.ANTHROPIC_API_KEY
if (!apiKey) {
return res.status(500).json({
type: 'error',
error: { type: 'configuration_error', message: 'ANTHROPIC_API_KEY not configured on server' }
})
}
const apiPath = '/' + req.params[0]
const bodyStr = JSON.stringify(req.body)
const options = {
hostname: 'api.anthropic.com',
port: 443,
path: apiPath,
method: 'POST',
headers: {
'Content-Type': 'application/json',
'x-api-key': apiKey,
'anthropic-version': '2023-06-01',
'Content-Length': Buffer.byteLength(bodyStr),
},
}
const proxyReq = https.request(options, (proxyRes) => {
res.writeHead(proxyRes.statusCode, proxyRes.headers)
proxyRes.pipe(res)
})
proxyReq.on('error', (err) => {
console.error('[Claude Proxy] Error:', err.message)
if (!res.headersSent) {
res.status(502).json({
type: 'error',
error: { type: 'proxy_error', message: err.message }
})
}
})
proxyReq.write(bodyStr)
proxyReq.end()
})
// Health check
app.get('/health', (req, res) => {
res.status(200).send('healthy')