showcase/aimock/RAILWAY.md
Tagline: authoritative backup of the showcase-aimock Railway service config
(image, startCommand, baked-in fixtures, env vars) and the from-scratch recreate
recipe. Concrete IDs / domains live in the Notion plan (section 9), not in
this public repo.
This document persists the Railway service configuration for showcase-aimock
in the repo so the service can be reconstructed from scratch if Railway state
is ever lost. All runtime config (image, startCommand, env vars) lives only in
Railway — this file is the authoritative backup.
Where the concrete IDs live. Because this repo is public, concrete Railway service/project/environment IDs and the current public domain are not stored here. They live in the internal Notion plan (see section 9) and can be queried live from the Railway GraphQL API with a valid account token. Everywhere below you see
<service-id>,<project-id>,<environment-id>, or<public-domain>, substitute the current value from one of those sources.
showcase-aimock is a shared mock LLM server that 14+ CopilotKit showcase
services route to via OPENAI_BASE_URL. It runs the @copilotkit/aimock
container in proxy-only mode and serves fixture-driven responses so demos work
deterministically without burning provider tokens. Unmatched requests fall
through to real upstream providers (OpenAI, Anthropic, Gemini).
| Field | Value |
|---|---|
| Service name | showcase-aimock |
| Service ID | <service-id> (see Notion plan, section 9) |
| Project name | showcase |
| Project ID | <project-id> (see Notion plan, section 9) |
| Environment | production |
| Environment ID | <environment-id> (see Notion plan, section 9) |
| Public domain | <public-domain> (see Notion plan, or Railway dashboard) |
Auth for
showcase-project mutations. Use an account-scopedRAILWAY_TOKEN(stored in the DevOpsshowcase1Password item) against the Railway GraphQL API with anAuthorization: Bearer <token>header. The Railway CLI session token is not authorized for mutations on theshowcaseproject — the account-scoped token is the working path.
To look these up live from Railway GraphQL with a valid account token:
query {
# List services under the `showcase` project to find the ID.
projects {
edges {
node {
id
name
services {
edges {
node {
id
name
}
}
}
}
}
}
}
Then drill into the specific service:
query {
service(id: "<service-id>") {
id
name
projectId
serviceInstances {
edges {
node {
environmentId
startCommand
source {
image
repo
}
domains {
serviceDomains {
domain
}
customDomains {
domain
}
}
}
}
}
}
}
showcase-aimock, built by .github/workflows/showcase_build.yml
from showcase/aimock/Dockerfile. The Dockerfile is FROM ghcr.io/copilotkit/aimock:latest (the upstream aimock image published from
CopilotKit/aimock) and bakes the fixture tree into the image (see
section 4) — that baked image is what Railway deploys, not the bare upstream
image.ghcr.io/copilotkit/aimock:latest. Pin the base
tag in the Dockerfile if you need to freeze it for showcase stability.linux/amd64, linux/arm64 (Railway pulls amd64).Fixtures are baked into the image at build time, not fetched remotely. The
showcase/aimock/Dockerfile copies three fixture directories from this repo
into the image:
shared/ → /fixtures/shared/ — common.json shared responses plus
smoke.json (the minimal "OK" ping used for health verification).d4/ → /fixtures/d4/ — per-slug fixtures for the D4 demos.d6/ → /fixtures/d6/ — per-slug fixtures for the D6 demos. The
showcase/aimock/d6/<slug>/ tree is the source of truth for these.The container loads these baked-in directories at boot (see the --fixtures
flags in section 5). There are no remote fixture URLs and no boot-time fetch —
the old d5-all.json / feature-parity.json / remote-smoke.json bundles
no longer exist (d5-all.json was a one-time migration source that was split
into the per-slug d6/ tree).
To update fixtures, edit the files under showcase/aimock/{shared,d4,d6}/ and
rebuild the image (a push touching showcase/aimock/** triggers
showcase_build.yml). Changes land on the next Railway deploy of the rebuilt
image.
showcase-harness browser-pool budget. The harness runs
BROWSER_POOL_BROWSERS=3long-lived Chromium processes with a globalBROWSER_POOL_MAX_CONTEXTS=40context cap (D6 peak 32 + D5 peak 8). D5 e2e-deep alone runs up to 4 services x 2 features = 8 concurrent contexts (~2.4 GB peak). The binding constraint is the PID ceiling of 1000, not memory, so contexts (not processes) are the scaling knob — tuneBROWSER_POOL_MAX_CONTEXTSto bound contention, or reduceFEATURE_CONCURRENCYine2e-deep.ts/max_concurrencyine2e-deep.ymlif a single probe needs throttling.
Railway overrides Docker ENTRYPOINT. When
startCommandis set, Railway runs it as the container's command and the image'sENTRYPOINTis ignored. That means the fullnode /app/dist/cli.jsbin invocation must appear explicitly instartCommand— flag-only invocations fail at boot withThe executable --proxy-only could not be found.This was discovered during the Phase 2 deploy when an initial flag-only startCommand was rejected.
node /app/dist/cli.js \
--proxy-only \
--provider-openai https://api.openai.com \
--provider-anthropic https://api.anthropic.com \
--provider-gemini https://generativelanguage.googleapis.com \
--fixtures /fixtures/shared \
--fixtures /fixtures/d4 \
--fixtures /fixtures/d6 \
--validate-on-load \
--host 0.0.0.0 \
--port 4010
The
--fixturesflags must point at the three baked-in subdirectories, not the/fixturesparent. A single--fixtures /fixturesdoes NOT recurse into the subdirectories — it loads nothing useful and every request falls through to the proxy. Pass each of/fixtures/shared,/fixtures/d4, and/fixtures/d6explicitly.
Flag-by-flag:
| Flag | Value | Purpose |
|---|---|---|
node /app/dist/cli.js | — | Explicit bin invocation — required because Railway's startCommand overrides ENTRYPOINT. |
--proxy-only | — | Forward unmatched requests to upstream providers instead of failing. |
--provider-openai | https://api.openai.com | Upstream URL for OpenAI passthrough. |
--provider-anthropic | https://api.anthropic.com | Upstream URL for Anthropic passthrough. |
--provider-gemini | https://generativelanguage.googleapis.com | Upstream URL for Gemini passthrough. |
--fixtures (×3 dirs) | /fixtures/{shared,d4,d6} | Repeatable flag; each loads one baked-in fixture directory at boot. Point at the subdirectories, not the /fixtures parent. |
--validate-on-load | — | Fail-loud on schema errors at boot. |
--host | 0.0.0.0 | Bind all interfaces so Railway can route to the container. |
--port | 4010 | Hardcoded listen port — matches the legacy wrapper container convention and the fixed |
Railway domain routing. Railway injects $PORT but the image defaults align with 4010. |
If adopting $PORT interpolation in the future, both startCommand and any
upstream OPENAI_BASE_URL env vars pointing at this service stay unchanged —
Railway routes the public domain to whatever port the container listens on.
None are required for the default configuration. Notes:
AIMOCK_ALLOW_PRIVATE_URLS=1 would only be needed if fixtures were loaded
from private URLs (RFC1918, loopback, etc.). Not applicable here — fixtures
are baked into the image and loaded from local directories, not over the
network.PORT is injected by Railway but not read by the current startCommand
(port is hardcoded to 4010). Harmless.If the Railway service is ever lost, recreate with the following recipe.
Substitute <service-id>, <environment-id>, and <public-domain> with the
concrete values from the Notion plan (section 9) or by querying Railway
GraphQL directly.
Create a new service in the showcase project, production environment.
Easiest path is the Railway UI (New Service → Docker Image), but the
GraphQL serviceCreate mutation works too.
Set source.image to the showcase-aimock image published by
.github/workflows/showcase_build.yml (the baked image from
showcase/aimock/Dockerfile, which contains the fixture tree — see section 3) via serviceInstanceUpdate:
mutation {
serviceInstanceUpdate(
serviceId: "<service-id>"
environmentId: "<environment-id>"
input: { source: { image: "<showcase-aimock-image-ref>" } }
) {
id
}
}
Deploying the bare upstream ghcr.io/copilotkit/aimock instead will boot
with no fixtures baked in — every request falls through to the proxy.
Set startCommand to the block in section 5 (join with spaces, escape as
needed) via the same serviceInstanceUpdate mutation with
input: { startCommand: "..." }. Remember Railway's startCommand overrides
the image's Docker ENTRYPOINT, so the full node /app/dist/cli.js bin
invocation must appear explicitly in the command string.
No env vars needed for default setup (see section 6).
Generate a public domain (serviceDomainCreate mutation, or the UI's
"Generate Domain" button). The historical domain pattern is
showcase-aimock-production.<railway-edge> — the current domain is in the
Notion plan (section 9) and visible in the Railway dashboard.
Deploy with serviceInstanceDeployV2 (do NOT use serviceInstanceRedeploy
— it replays the last snapshot, which may predate the image/startCommand
change):
mutation {
serviceInstanceDeployV2(
serviceId: "<service-id>"
environmentId: "<environment-id>"
)
}
Verify (find the current public domain via Railway GraphQL's domains
field or the service's Railway dashboard):
curl -sS -X POST https://<public-domain>/v1/chat/completions \
-H "Content-Type: application/json" \
-H "Authorization: Bearer test" \
-d '{"model":"gpt-4","messages":[{"role":"user","content":"ping"}]}'
Expect the smoke fixture's "OK" response. If proxy-fallthrough to OpenAI
fires instead, the smoke fixture did not load — confirm the deployed image
is the baked showcase-aimock image and that the --fixtures flags point
at /fixtures/{shared,d4,d6} (not the bare /fixtures parent).
Update any showcase services whose OPENAI_BASE_URL points at the old
URL, if the domain changed during reconstruction.
The Dockerfile in this directory is not dead code — it is the image that
Railway deploys. .github/workflows/showcase_build.yml builds it (matrix entry
showcase-aimock, with dockerfile: showcase/aimock/Dockerfile and context
showcase/aimock) and publishes the showcase-aimock image. The Dockerfile is
FROM ghcr.io/copilotkit/aimock:latest and bakes the fixture tree into the
image:
FROM ghcr.io/copilotkit/aimock:latest
# Depth-organized fixture directories
COPY shared/ /fixtures/shared/
COPY d4/ /fixtures/d4/
COPY d6/ /fixtures/d6/
Do not remove it — deleting it would strip the baked-in fixtures and the deployed mock would serve nothing (all requests would fall through to the proxy).
CopilotKit/aimock repo CHANGELOGshowcase/aimock/{shared,d4,d6}/,
which triggers showcase_build.yml to rebuild the showcase-aimock image;
changes take effect on the next Railway deploy of the rebuilt image.