Skip to main content

Project-scoped Railway CLI — reads RAILWAY_TOKEN from .env.local, no login needed

Project description

railguey

railguey

Project-scoped Railway CLI.
Reads RAILWAY_TOKEN from each project's .env.local — no railway login needed.

PyPI Tests Python License


railguey is for teams and businesses that need reliable Railway deployments. It is not the simplest way to deploy — Railway's built-in GitHub app is simpler. But railguey is more reliable, because it draws a cleaner engineering boundary.

Why not just use Railway's GitHub App?

Railway's GitHub App is fast to set up: connect your repo, push to main, and your service deploys. For prototyping, that speed is genuinely great. But speed of setup and quality of engineering are different things.

The GitHub App bundles five responsibilities into one opaque chain: watch for code changes, authenticate to GitHub, receive a webhook, clone the repo, build and deploy. When the chain works, it feels like magic. When it doesn't — and it has broken four times in four months — there is no observability, no retry, and no notification. Your push goes in. Nothing comes out. You find out when a customer does.

git push
   |
   v
GitHub webhook -> Railway GitHub App -> clone/build/deploy
   |                    |                     |
   +-- missed event     +-- auth drift        +-- silent trigger failure

Every decision diamond in this diagram is a place where the chain can silently break. Missed webhooks, lost build triggers, GitHub App auth failures — all produce the same result: nothing happens, and nobody tells you.

This isn't a bug. It's an architectural choice. Railway chose to own the entire pipeline from push to deploy, which means every failure in GitHub's webhook delivery becomes Railway's problem — and yours.

How railguey fixes this

railguey separates concerns. GitHub Actions watches your repo (GitHub watching GitHub — the thing it was built for). railguey handles the deploy via Railway's API using project-scoped tokens. Railway builds and runs your service (the thing it was built for). Each system does one job.

git push
   |
   v
GitHub Actions -> tests -> railguey -> Railway API -> build/deploy
       |             |        |              |
       +-- visible   +-- fail +-- nonzero    +-- deployment status

If CI fails, GitHub tells you. If the deploy fails, the CLI returns an error. If the service is unhealthy, railguey doctor catches it. Every step is observable, retryable, and owned by the system best suited to do it.

Fast delivery and good engineering aren't opposites — but Railway's GitHub App trades the second for the first. railguey gives you both.

Doc What it covers
WHY-NOT-RAILWAY-APP.md The architectural argument — why coupling CI/CD triggering with deployment is a design flaw, not just a bug
WHY-RAILGUEY.md The evidence — four incidents, community reports, and what the project-token pattern does differently
WHY-RAILGUEY.md#case-study Real-world case study — ghost GitHub repo links silently blocked env var operations across 5 services

When to use railguey

  • You manage Railway services from agents, local shells, or CI and want reliable deploys, rollbacks, and logs without railway login
  • You run multiple Railway projects and want one auth pattern across local dev, CI/CD, and AI tooling
  • Deploy reliability matters — production services, client projects, anything where a silently missed deploy costs you
  • You're already using GitHub Actions and want Railway deploys in the same pipeline as your tests

When NOT to use railguey

  • Quick demos and hobby projects. Railway's GitHub app is genuinely convenient for push-and-forget deploys. If you're prototyping and don't care about deploy reliability, the built-in integration is fine.
  • You only deploy once in a while from the terminal. If manual dashboard checks are fine, the Railway CLI with a project token is enough.
  • You're happy with the dashboard. If you deploy once a week and check status manually, railguey adds complexity you don't need.

Known limitations

  • All tools depend on Railway's Backboard GraphQL API, which isn't officially documented. The schema could change without notice.
  • No Railway CLI required. The core deploy tools use Railway's GraphQL/API surfaces with project-scoped tokens. The legacy CLI backend has been removed.
  • One token per project. Project-scoped tokens can't query across projects. If you manage 10 projects, you need 10 .env.local files in 10 workspaces. This is by design (isolation), but it's more setup than a user-level login.

Install

Requires Python 3.10+. No Railway CLI needed.

pip install railguey

Or run without installing:

uvx railguey --help
Install from source
git clone https://github.com/eidos-agi/railguey.git
cd railguey
pip install -e .

CLI usage

pip install railguey gives you the railguey command with the core deploy and diagnostics tools as subcommands:

railguey status ~/repos/my-app
railguey logs ~/repos/my-app cerebro --lines 50
railguey deploy ~/repos/my-app web
railguey deployments ~/repos/my-app cerebro --limit 5
railguey doctor ~/repos/my-app
railguey variables ~/repos/my-app web
railguey service-info ~/repos/my-app cerebro

Every command takes a workspace path — the directory containing .env.local with RAILWAY_TOKEN.

First deploy of a fresh service

A brand-new Railway service needs both a serviceCreate mutation (with environmentId so the per-env instance materializes) AND a source upload to be deployable. railguey collapses this to one verb:

railguey service-bootstrap /path/to/repo my-service

Under the hood: serviceCreate with the project-token's environmentId (Railway requires this), then a gzipped tarball of the workspace POSTed to https://backboard.railway.com/project/{p}/environment/{e}/up?serviceId={s} — both project-token-only, no GitHub-Railway link, no account-level auth.

The tarball respects .gitignore, .dockerignore, and .railwayignore (90% of .gitignore semantics — no negation, no nested **); .git, node_modules, .venv, and *.cache directories are always excluded. Hard-cap of 256MB; use .railwayignore to trim larger workspaces.

For subsequent deploys, use railguey upload-source (just upload — service already exists) or railguey deploy (redeploy from existing source).

Commands

Core CLI commands, all token-based. No Railway CLI required.

Command What it does
railguey status Project overview — all services, deploy status, domains
railguey services List services with IDs
railguey logs Fetch recent deploy or build logs (with optional filter)
railguey deploy Trigger a deploy from linked source
railguey redeploy Redeploy latest deployment (rebuilds from source)
railguey restart Restart latest deployment (no rebuild, fast)
railguey variables List env vars for a service
railguey variable-set Set an env var (triggers redeploy)
railguey domain Generate a railway.app domain or add a custom domain
railguey environment-create Create a new environment (staging, preview, etc.)
railguey deployments Deployment history with IDs, statuses, timestamps, rollback eligibility
railguey rollback Roll back to a specific deployment
railguey service-info Full service config — build/start commands, healthcheck, region, replicas
railguey http-logs HTTP request logs — status codes, latency, paths
railguey deployment-logs Logs for a specific deployment by ID (deploy or build, with filter)
railguey unlink-repo Disconnect a service from GitHub repo linking
railguey service-create Create a service bound to the project token's environment
railguey upload-source Tarball workspace + POST to Railway /up — deploy via project token
railguey service-bootstrap One-call first deploy: service-create + upload-source
railguey service-delete Delete a service from the Railway project (irreversible)
railguey doctor Audit a workspace for deployment best practices

railguey doctor checks:

  1. RAILWAY_TOKEN exists in .env.local
  2. .env.local is in .gitignore
  3. GitHub Actions deploy workflow exists with token-based CI/CD
  4. No services linked to GitHub repos

Every command requires a workspace parameter — the absolute path to a project directory that has a .env.local (or .env) containing RAILWAY_TOKEN.

Example

CLI:

railguey logs ~/repos/my-app web --lines 50

Python library:

from railguey.lib import tools
result = await tools.logs("/Users/you/repos/my-app", "web", lines=50)

Both paths read /Users/you/repos/my-app/.env.local, extract the token, resolve the Railway project/service, and fetch logs through Railway's API.

The project-token pattern

Railway lets you create project-scoped tokens — API keys that authenticate to a single project without any user login. These tokens work the same way everywhere:

Context How the token is used
Local dev .env.localrailway logs, railway up, etc.
Agents using railguey Call the CLI with a workspace path
GitHub Actions CI/CD Repository secret → RAILWAY_TOKEN env var
Any CI system Same — export the token, run railway up

One mechanism. No OAuth. No repo linking. No webhook fragility.

GitHub Actions deploy workflow (copy-paste)

Add RAILWAY_TOKEN as a repository secret, then:

# .github/workflows/deploy.yml
name: Deploy to Railway

on:
  push:
    branches: [main]

jobs:
  deploy:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4

      - name: Install Railway CLI
        run: curl -fsSL https://railway.com/install.sh | sh

      - name: Deploy
        env:
          RAILWAY_TOKEN: ${{ secrets.RAILWAY_TOKEN }}
        run: railway up --service ${{ vars.RAILWAY_SERVICE }} --detach

Set RAILWAY_SERVICE as a repository variable. More examples in examples/.

Token discovery

  1. Looks for RAILWAY_TOKEN= in {workspace}/.env.local
  2. Falls back to {workspace}/.env
  3. Raises a clear error if not found

Supports bare values, single-quoted, and double-quoted values.

License

MIT

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

railguey-0.3.0.tar.gz (347.5 kB view details)

Uploaded Source

Built Distribution

If you're not sure about the file name format, learn more about wheel file names.

railguey-0.3.0-py3-none-any.whl (45.5 kB view details)

Uploaded Python 3

File details

Details for the file railguey-0.3.0.tar.gz.

File metadata

  • Download URL: railguey-0.3.0.tar.gz
  • Upload date:
  • Size: 347.5 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for railguey-0.3.0.tar.gz
Algorithm Hash digest
SHA256 0e40ee5382186595af9b100939a82d736d7e1f96bdd297aeb35113d18ff98bb1
MD5 397f77acdb9e1aa9f6652c5db6462833
BLAKE2b-256 5b274d304e4492bc1c56dc11cfc2c1d81b771cda8bb389061824bc39906894e5

See more details on using hashes here.

Provenance

The following attestation bundles were made for railguey-0.3.0.tar.gz:

Publisher: publish.yml on eidos-agi/railguey

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

File details

Details for the file railguey-0.3.0-py3-none-any.whl.

File metadata

  • Download URL: railguey-0.3.0-py3-none-any.whl
  • Upload date:
  • Size: 45.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.12

File hashes

Hashes for railguey-0.3.0-py3-none-any.whl
Algorithm Hash digest
SHA256 793c6579cda5e0ae5608811559fe6e992c1debcf126ed04af879615fa9851714
MD5 d42b584f426fafec7cb9ad5bbc53f522
BLAKE2b-256 72fdba1082901f37c0c4424d43913360b70e5c461060e41726c5d40b0eae9418

See more details on using hashes here.

Provenance

The following attestation bundles were made for railguey-0.3.0-py3-none-any.whl:

Publisher: publish.yml on eidos-agi/railguey

Attestations: Values shown here reflect the state when the release was signed and may no longer be current.

Supported by

AWS Cloud computing and Security Sponsor Datadog Monitoring Depot Continuous Integration Fastly CDN Google Download Analytics Pingdom Monitoring Sentry Error logging StatusPage Status page