Skip to main content

Command-line interface for ContentRX — content-design linter for Figma, code, and CI.

Project description

ContentRX CLI

contentrx-cli is a thin command-line client for the ContentRX content-standards checker. It posts your strings to the ContentRX API and prints violations — the kind of thing you'd call from a pre-commit hook, a CI job, or an ad-hoc terminal check while writing UI copy.

pip install contentrx-cli
export CONTENTRX_API_KEY=cx_...      # mint at https://contentrx.io/dashboard
contentrx "Click here to learn more"

The CLI ships with no third-party dependencies — stdlib only. Install time is seconds, the attack surface is whatever Python itself already includes.


Authentication

All requests need an API key. Generate one in the dashboard:

https://contentrx.io/dashboardGenerate key (or Rotate)

The key is shown exactly once. Store it somewhere you trust. The CLI reads it from the CONTENTRX_API_KEY environment variable — no global config file, no per-directory .contentrxrc.

export CONTENTRX_API_KEY=cx_a1b2c3...

If the key is missing or expired the CLI prints a short, actionable error pointing back at the dashboard URL.

Pointing at a non-default backend

Self-hosted or test-deployment users can override the API URL:

export CONTENTRX_API_URL=https://contentrx-staging.example.com

Usage

Single check

contentrx "Save changes"
✓ PASS
  Content type: button_cta
  Active-voice verb. Specific action. Clean.

With hints:

contentrx --content-type error_message "Something went wrong."
contentrx --moment destructive_action "Delete forever?"

JSON output

For piping into other tools or inspecting the raw response:

contentrx --json "Save changes"

Batch check

Text file, one string per line:

cat > strings.txt <<'EOF'
Save changes
Click here to learn more
Something went wrong.
EOF

contentrx --batch strings.txt

JSON with per-string hints:

cat > strings.json <<'EOF'
[
  {"text": "Save changes", "content_type": "button_cta"},
  {"text": "Delete forever?", "moment": "destructive_action"}
]
EOF

contentrx --batch strings.json

Verbose mode

Adds latency + month-to-date usage from the API response:

contentrx -v "Save changes"

Scenarios

Three concrete things you can do with the CLI today.

1. Ad-hoc check while writing copy

You're writing a button label for a hero section. Is "Click here to learn more" actually good?

$ contentrx --content-type button_cta "Click here to learn more" FAIL
  Moment: decision_point
  2 violations:
    ACT-02 [block]  Vague CTA  "Click here" doesn't name the destination.
                    Suggestion: Lead with verb+object: "Learn more" or "Read the docs".
    CLR-03 [warn]   Redundant phrasing — "to learn more" duplicates the button's purpose.

$ contentrx --content-type button_cta "Read the docs" PASS
  Moment: decision_point
  Verb-first. Object named. Length appropriate for a CTA.

Takes <2 seconds per check. Free tier gives 10 checks/month for exactly this kind of exploration.

2. Linting an i18n catalog during CI

You have locales/en.json with every user-facing string. Lint it on every PR that touches it:

# .github/workflows/copy-lint.yml
on:
  pull_request:
    paths: ['locales/en.json']

jobs:
  lint:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4
      - uses: actions/setup-python@v5
        with:
          python-version: '3.12'
      - run: pip install contentrx-cli
      - name: Convert en.json → ContentRX batch format
        run: |
          jq -r 'to_entries | map({text: .value}) | tostring' locales/en.json > strings.json
      - name: Lint
        env:
          CONTENTRX_API_KEY: ${{ secrets.CONTENTRX_API_KEY }}
        run: contentrx --batch strings.json

Exit code 1 on any violation fails the job and blocks the PR. Engineers get a terminal output they can scan in 10 seconds.

3. Pre-commit hook on a specific file

You only care about the canonical strings file, not every edit. One hook, scoped to that file:

# .pre-commit-config.yaml
repos:
  - repo: local
    hooks:
      - id: contentrx
        name: ContentRX
        entry: contentrx --batch
        language: system
        files: ^content/strings\.json$
        pass_filenames: true

Runs only when content/strings.json changes. Catches violations before they leave your laptop.


Exit codes

Stable across versions — safe to pin on in CI.

Code Meaning
0 All checks passed
1 At least one check failed (violations present)
2 Usage error (argparse / invalid arguments / bad batch file)
3 Missing or invalid CONTENTRX_API_KEY
4 Monthly quota exhausted
5 Rate limit exceeded
6 Network / upstream error

CI examples

GitHub Actions

name: Content lint
on:
  pull_request:
    paths: ['src/**/*.tsx', 'src/**/*.ts']

jobs:
  lint:
    runs-on: ubuntu-latest
    steps:
      - uses: actions/checkout@v4
      - uses: actions/setup-python@v5
        with:
          python-version: '3.12'
      - run: pip install contentrx-cli
      - name: Lint changed UI strings
        env:
          CONTENTRX_API_KEY: ${{ secrets.CONTENTRX_API_KEY }}
        run: |
          contentrx --batch strings.json

(For AST-based extraction of strings from TSX/JSX, use the contentrx-action GitHub Action instead of rolling your own extraction. Shipped separately in github-action/.)

pre-commit

# .pre-commit-config.yaml
repos:
  - repo: local
    hooks:
      - id: contentrx
        name: ContentRX
        entry: contentrx --batch
        language: system
        files: ^content/strings\.json$

Development

cd cli-client
pip install -e ".[dev]"
pytest tests/

Tests mock the HTTP layer so they run offline without an API key.

License

In-monorepo source: Functional Source License, FSL-1.1-MIT (auto-converts to MIT after the FSL grant period). The published contentrx-cli package on PyPI ships under MIT. See LICENSE.

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

contentrx_cli-0.4.0.tar.gz (27.2 kB view details)

Uploaded Source

Built Distribution

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

contentrx_cli-0.4.0-py3-none-any.whl (20.5 kB view details)

Uploaded Python 3

File details

Details for the file contentrx_cli-0.4.0.tar.gz.

File metadata

  • Download URL: contentrx_cli-0.4.0.tar.gz
  • Upload date:
  • Size: 27.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.13

File hashes

Hashes for contentrx_cli-0.4.0.tar.gz
Algorithm Hash digest
SHA256 17082f8f5c4f9e1a69ed0dbbea546cc8deaa97dbf3c62b17aee43ed2fdd6cf33
MD5 eceddd11b637047fdd89051f58e49860
BLAKE2b-256 6917fc70d1f0353569ec3c23b184b1d608380ccce82c8f186b3326df27f4298d

See more details on using hashes here.

Provenance

The following attestation bundles were made for contentrx_cli-0.4.0.tar.gz:

Publisher: publish_cli.yml on thenewforktimes/ContentRX

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

File details

Details for the file contentrx_cli-0.4.0-py3-none-any.whl.

File metadata

  • Download URL: contentrx_cli-0.4.0-py3-none-any.whl
  • Upload date:
  • Size: 20.5 kB
  • Tags: Python 3
  • Uploaded using Trusted Publishing? Yes
  • Uploaded via: twine/6.1.0 CPython/3.13.13

File hashes

Hashes for contentrx_cli-0.4.0-py3-none-any.whl
Algorithm Hash digest
SHA256 5c586384309d5406620b6de1e5bca81c480ba80d380770934b853b19e87e4c3f
MD5 e4b5a11efea3e938a8643ed4814c4e15
BLAKE2b-256 1c96637e7582288c1ccb41e0deb7022c90fc8b2b8c461d4015a2e38c49977b4a

See more details on using hashes here.

Provenance

The following attestation bundles were made for contentrx_cli-0.4.0-py3-none-any.whl:

Publisher: publish_cli.yml on thenewforktimes/ContentRX

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