.claude/skills/containerize-apps/SKILL.md
Containerizes applications with impact-aware Dockerfiles and docker-compose configurations. This skill should be used when containerizing projects for Docker, creating Dockerfiles, docker-compose files, or preparing applications for Kubernetes deployment. It performs impact analysis first (env vars, network topology, auth/CORS), then generates properly configured container configs. Invokes the impact-analyzer subagent for comprehensive project scanning.
npx skillsauth add anasahmed07/doit containerize-appsInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
This skill containerizes applications with impact analysis first, ensuring Docker configs work correctly with authentication, networking, and environment configuration. It generates Dockerfiles, docker-compose files, and documents required changes.
1. Impact Analysis → Scan project for containerization requirements
2. Blueprint Selection → Choose appropriate Dockerfile patterns
3. Configuration Gen → Generate Dockerfiles + docker-compose
4. Impact Documentation → Document required code/config changes
5. Optional: Gordon → Validate with Docker AI (if available)
Before generating ANY container configuration, invoke the impact-analyzer subagent:
Use Task tool with:
subagent_type: "impact-analyzer"
prompt: |
Analyze this project for containerization requirements.
Scan for:
1. Environment variables (build-time vs runtime)
2. Localhost/127.0.0.1 references that need Docker service names
3. Auth/CORS configurations (Better Auth trustedOrigins, FastAPI CORS)
4. Service dependencies and startup order
5. Ports used by each service
Return structured findings for containerization.
Wait for the analysis report before proceeding.
The report will identify:
Based on project analysis, select appropriate blueprints from assets/:
| Project Type | Blueprint | Key Considerations |
|--------------|-----------|-------------------|
| FastAPI/Python | Dockerfile.fastapi | uv, multi-stage, non-root |
| Next.js | Dockerfile.nextjs | standalone output, NEXT_PUBLIC_* |
| Python Service | Dockerfile.python | Generic Python app |
| MCP Server | Dockerfile.mcp | Based on Python service |
For each service, generate Dockerfile using blueprint + impact analysis:
Customization points:
# From impact analysis - replace these:
CMD ["uvicorn", "{{MODULE_PATH}}:app", ...] # Module path from project
EXPOSE {{PORT}} # Port from analysis
ENV {{ENV_VARS}} # Runtime env vars
ARG {{BUILD_ARGS}} # Build-time vars (NEXT_PUBLIC_*)
Generate compose file with proper networking:
# Network topology from impact analysis
services:
web:
build:
context: ./web-dashboard
args:
# BROWSER: baked into JS bundle, runs on user's machine
- NEXT_PUBLIC_API_URL=http://localhost:8000
- NEXT_PUBLIC_SSO_URL=http://localhost:3001
environment:
- NODE_ENV=production
# SERVER: read at runtime, runs inside container
- SERVER_API_URL=http://api:8000
- SERVER_SSO_URL=http://sso-platform:3001
ports:
- "3000:3000"
depends_on:
api:
condition: service_healthy
api:
build:
context: ./packages/api
environment:
- DATABASE_URL=${DATABASE_URL} # From .env (external Neon)
- FRONTEND_URL=http://web:3000 # Docker service name!
- CORS_ORIGINS=http://localhost:3000,http://web:3000
ports:
- "8000:8000"
healthcheck:
test: ["CMD", "curl", "-f", "http://localhost:8000/health"]
interval: 30s
timeout: 10s
retries: 3
start_period: 40s
# Add other services based on analysis...
Generate environment template for Docker:
# External services (keep actual values)
DATABASE_URL=postgresql://[email protected]/...
# Docker networking (use service names)
API_URL=http://api:8000
SSO_URL=http://sso:3001
FRONTEND_URL=http://web:3000
# Secrets (in production, use Docker secrets or K8s secrets)
BETTER_AUTH_SECRET=your-secret-here
Generate a CONTAINERIZATION.md documenting what needs to change in the codebase:
# Containerization Impact Report
## Required Code Changes
### 1. Auth Configuration (sso-platform/src/lib/auth.ts)
Add Docker service names to trustedOrigins:
```typescript
trustedOrigins: [
"http://localhost:3000",
"http://localhost:8000",
"http://web:3000", // ADD: Docker frontend
"http://api:8000", // ADD: Docker backend
]
Update CORS origins:
origins = [
"http://localhost:3000",
"http://web:3000", # ADD: Docker frontend
os.getenv("FRONTEND_URL", "http://localhost:3000"),
]
DATABASE_URL: Keep as-is (external Neon)NEXT_PUBLIC_API_URL: Must be build ARG, set to http://api:8000BETTER_AUTH_URL: Runtime ENV, set to http://sso:3001
---
## Step 5: Optional Gordon Validation
If Docker Desktop with Gordon is available, suggest validation:
```bash
# Validate Dockerfile
cat packages/api/Dockerfile | docker ai "Rate this Dockerfile for production use"
# Or use Docker Desktop UI
# Click ✨ icon → "Review my Dockerfile"
Note: Gordon CLI can't read files directly. Use piping or Desktop UI.
See assets/Dockerfile.fastapi for:
Customize:
CMD module path (e.g., taskflow_api.main:app)EXPOSE portSee assets/Dockerfile.nextjs for:
Customize:
See assets/docker-compose.template.yml for:
Problem: Dockerfile may not use latest features or build checks
Solution: Always add # syntax=docker/dockerfile:1 as the first line:
# syntax=docker/dockerfile:1
FROM python:3.13-slim
...
This ensures:
docker build --check for lintingProblem: Browser runs on host (needs localhost), server runs in container (needs service names) Solution: Use DIFFERENT variable names - no confusion:
build:
args:
- NEXT_PUBLIC_API_URL=http://localhost:8000 # Browser only
environment:
- SERVER_API_URL=http://api:8000 # Server only
Code change: Server-side routes use process.env.SERVER_API_URL || process.env.NEXT_PUBLIC_API_URL
This is cleaner than using same variable with different values.
Problem: localhost refers to container, not host or other containers Solution: Use Docker service names (api, web, sso) for server-side
Problem: wget http://localhost:3000 fails with IPv6 resolution
Solution: Always use 127.0.0.1 instead of localhost in healthchecks:
healthcheck:
test: ["CMD", "wget", "--spider", "http://127.0.0.1:3000/"] # NOT localhost!
Problem: Code detects Neon vs local postgres incorrectly with Docker service names
Solution: Add sslmode=disable to local postgres URLs:
DATABASE_URL=postgresql://postgres:postgres@postgres:5432/db?sslmode=disable
Code can check: url.includes("sslmode=disable") → local postgres
Problem: Better Auth rejects requests from unknown origins Solution: Add Docker service names to trustedOrigins BEFORE building
Problem: Frontend starts before API is ready
Solution: Use depends_on with condition: service_healthy
Problem: Container marked unhealthy before app starts
Solution: Use start_period in health check (e.g., 40s)
Problem: pgAdmin rejects .local domains
Solution: Use valid email like [email protected]
Problem: playwright (300MB+) in dependencies bloats image
Solution: Keep test tools in devDependencies, ensure postgres driver is in dependencies
Problem: FastMCP's transport security rejects Docker service names
Error: 421 Misdirected Request - Invalid Host header
Cause: MCP SDK defaults to allowed_hosts=["127.0.0.1:*", "localhost:*", "[::1]:*"]
Solution: Configure transport security to allow Docker container names:
from mcp.server.transport_security import TransportSecuritySettings
transport_security = TransportSecuritySettings(
allowed_hosts=[
"127.0.0.1:*",
"localhost:*",
"[::1]:*",
"mcp-server:*", # Docker container name
"0.0.0.0:*",
],
)
mcp = FastMCP(..., transport_security=transport_security)
Problem: MCP /mcp endpoint returns 406 on GET requests
Solution: Add a separate /health endpoint via ASGI middleware:
class HealthMiddleware:
def __init__(self, app): self.app = app
async def __call__(self, scope, receive, send):
if scope["type"] == "http" and scope["path"] == "/health":
response = JSONResponse({"status": "healthy"})
await response(scope, receive, send)
return
await self.app(scope, receive, send)
Problem: Drizzle db:push drops tables not in its schema (including API tables)
Root Cause: If API creates tables, then Drizzle runs, Drizzle drops them
Solution: Startup order must be:
references/startup-script-pattern.mdProblem: SQLModel.metadata.create_all() doesn't create tables
Cause: Models not imported before create_all() runs
Solution: Explicitly import all models in database.py:
# MUST import before create_all()
from .models import User, Task, Project # noqa: F401
Problem: JWT validated against wrong SSO instance's keys
Error: Key not found - token kid: ABC, available kids: ['XYZ']
Cause: Logged in via local SSO, but Docker API validates against Docker SSO
Solution: Clear browser cookies and login fresh through Docker stack
Problem: uv pip install fails with network timeout
Error: Failed to download distribution due to network timeout
Solution: Increase timeout in Dockerfile:
RUN UV_HTTP_TIMEOUT=120 uv pip install --system --no-cache -r pyproject.toml
After containerization, verify:
impact-analysis.md - Detailed impact analysis guidenetwork-topology.md - Docker/K8s networking patternsauth-containerization.md - Better Auth + Docker guideDockerfile.fastapi - FastAPI blueprintDockerfile.nextjs - Next.js blueprintDockerfile.python - Generic Python blueprintdocker-compose.template.yml - Compose templatedevelopment
Use when building real-time communication systems with WebSockets or Socket.IO. Invoke for bidirectional messaging, horizontal scaling with Redis, presence tracking, room management.
development
Review UI code for Web Interface Guidelines compliance. Use when asked to "review my UI", "check accessibility", "audit design", "review UX", or "check my site against best practices".
development
React and Next.js performance optimization guidelines from Vercel Engineering. This skill should be used when writing, reviewing, or refactoring React/Next.js code to ensure optimal performance patterns. Triggers on tasks involving React components, Next.js pages, data fetching, bundle optimization, or performance improvements.
tools
Guide for creating effective skills. This skill should be used when users want to create a new skill (or update an existing skill) that extends Claude's capabilities with specialized knowledge, workflows, or tool integrations.