Back to Copilotkit

showcase-aimock Railway service reference

showcase/aimock/RAILWAY.md

1.61.213.9 KB
Original Source

showcase-aimock Railway service reference

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.

1. What this service is

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).

2. Railway identity

FieldValue
Service nameshowcase-aimock
Service ID<service-id> (see Notion plan, section 9)
Project nameshowcase
Project ID<project-id> (see Notion plan, section 9)
Environmentproduction
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-scoped RAILWAY_TOKEN (stored in the DevOps showcase 1Password item) against the Railway GraphQL API with an Authorization: Bearer <token> header. The Railway CLI session token is not authorized for mutations on the showcase project — the account-scoped token is the working path.

To look these up live from Railway GraphQL with a valid account token:

graphql
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:

graphql
query {
  service(id: "<service-id>") {
    id
    name
    projectId
    serviceInstances {
      edges {
        node {
          environmentId
          startCommand
          source {
            image
            repo
          }
          domains {
            serviceDomains {
              domain
            }
            customDomains {
              domain
            }
          }
        }
      }
    }
  }
}

3. Runtime image

  • Image: 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.
  • Base aimock version: tracks ghcr.io/copilotkit/aimock:latest. Pin the base tag in the Dockerfile if you need to freeze it for showcase stability.
  • Available platforms: linux/amd64, linux/arm64 (Railway pulls amd64).

4. Fixture sources

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=3 long-lived Chromium processes with a global BROWSER_POOL_MAX_CONTEXTS=40 context 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 — tune BROWSER_POOL_MAX_CONTEXTS to bound contention, or reduce FEATURE_CONCURRENCY in e2e-deep.ts / max_concurrency in e2e-deep.yml if a single probe needs throttling.

5. Start command

Railway overrides Docker ENTRYPOINT. When startCommand is set, Railway runs it as the container's command and the image's ENTRYPOINT is ignored. That means the full node /app/dist/cli.js bin invocation must appear explicitly in startCommand — flag-only invocations fail at boot with The executable --proxy-only could not be found. This was discovered during the Phase 2 deploy when an initial flag-only startCommand was rejected.

sh
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 --fixtures flags must point at the three baked-in subdirectories, not the /fixtures parent. A single --fixtures /fixtures does 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/d6 explicitly.

Flag-by-flag:

FlagValuePurpose
node /app/dist/cli.jsExplicit bin invocation — required because Railway's startCommand overrides ENTRYPOINT.
--proxy-onlyForward unmatched requests to upstream providers instead of failing.
--provider-openaihttps://api.openai.comUpstream URL for OpenAI passthrough.
--provider-anthropichttps://api.anthropic.comUpstream URL for Anthropic passthrough.
--provider-geminihttps://generativelanguage.googleapis.comUpstream 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-loadFail-loud on schema errors at boot.
--host0.0.0.0Bind all interfaces so Railway can route to the container.
--port4010Hardcoded 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.

6. Environment variables

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.

7. How to reconstruct

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.

  1. 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.

  2. 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:

    graphql
    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.

  3. 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.

  4. No env vars needed for default setup (see section 6).

  5. 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.

  6. Deploy with serviceInstanceDeployV2 (do NOT use serviceInstanceRedeploy — it replays the last snapshot, which may predate the image/startCommand change):

    graphql
    mutation {
      serviceInstanceDeployV2(
        serviceId: "<service-id>"
        environmentId: "<environment-id>"
      )
    }
    
  7. Verify (find the current public domain via Railway GraphQL's domains field or the service's Railway dashboard):

    sh
    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).

  8. Update any showcase services whose OPENAI_BASE_URL points at the old URL, if the domain changed during reconstruction.

8. The Dockerfile is LIVE

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:

dockerfile
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).