Skip to main content

RAC — lint and diff product requirements written in Markdown.

Project description

RAC (Requirements-as-Code)

Lint, diff, and analyse product requirements from the command line

Product requirements are often trapped in documents, making them difficult to review, validate, and track over time.

RAC brings software engineering workflows to product requirements.

Write requirements in Markdown. Store them in Git. Validate them, compare versions, and analyse change over time.

Markdown
    ↓
Product Model
    ↓
Validation
    ↓
Diffing
    ↓
Portfolio Analysis
    ↓
AI Review (future)

Why RAC?

Engineers have mature tooling for code:

  • Linters
  • Code review
  • Diffs
  • Static analysis
  • Version control

Product requirements typically have none of these.

RAC applies the same principles to product requirements.

Validate

rac validate bond_dashboard.md

Compare

rac diff bond_dashboard_v1.md bond_dashboard_v2.md

Analyse

rac stats ./features

Example

Create a requirement file:

# Bond Dashboard

## Problem

Retail investors struggle to understand interest-rate exposure.

## Requirements

- [REQ-001] User can view portfolio holdings
- [REQ-002] User can view portfolio duration
- [REQ-003] User can view portfolio yield

## Success Metrics

- Monthly Active Users
- Dashboard Views

## Risks

- Inaccurate market data

Validate it:

rac validate bond_dashboard.md

Output:

PASS

Now compare two versions:

rac diff bond_dashboard_v1.md bond_dashboard_v2.md

Output:

Added Requirements

+ REQ-004 View projected yield forecast

Modified Requirements

~ REQ-002

Before:
User can view portfolio duration

After:
User can view and compare portfolio duration

RAC compares product changes, not just text changes.


Installation

Using pip

pip install requirements-as-code

Using uv

uv tool install requirements-as-code

Verify installation:

rac --help

Quick Start

Create a file:

touch feature.md

Add requirements:

# Trade Alerts

## Problem

Investors miss important market movements.

## Requirements

- [REQ-001] User can create a trade alert
- [REQ-002] User can edit a trade alert
- [REQ-003] User can delete a trade alert

Validate:

rac validate feature.md

Compare versions:

rac diff old.md new.md

Philosophy

RAC follows a few simple principles.

Markdown First

Requirements should remain easy to write and review.

RAC uses Markdown as the source format.

No proprietary editors.

No custom file formats.

Git Native

Requirements should work naturally inside:

  • GitHub
  • GitLab
  • VS Code
  • Cursor
  • Claude Code

AI Optional

RAC should be useful without AI.

The foundation is:

  • structure
  • validation
  • diffing
  • analysis

AI is an enhancement, not a dependency.

Product Model

Internally, RAC converts Markdown into a structured Product Model.

Markdown
    ↓
Parser
    ↓
Feature Model
    ↓
Validation
    ↓
Diffing
    ↓
Stats

This enables reliable analysis without relying on fragile text processing.


Markdown Specification

Every feature is represented by a single Markdown file.

Example:

# Feature Title

## Problem

Problem statement.

## Requirements

- [REQ-001] Requirement text
- [REQ-002] Requirement text

## Success Metrics

- Metric 1

## Risks

- Risk 1

Required Sections

Required:

  • # Title
  • ## Problem
  • ## Requirements

Optional (recommended):

  • ## Success Metrics
  • ## Risks

Commands

Validate

Validate a requirement file.

rac validate feature.md

Checks:

  • Required sections exist
  • Requirement IDs are valid
  • Requirement IDs are unique
  • Requirement text is not empty

Warnings:

  • Missing risks
  • Missing success metrics
  • Duplicate requirement text
  • Ambiguous wording

Diff

Compare two versions of a feature.

rac diff old.md new.md

Detects:

  • Added requirements
  • Removed requirements
  • Modified requirements
  • Added metrics
  • Removed metrics
  • Added risks
  • Removed risks

Requirements are matched by ID.


Stats

Portfolio-level analysis. Recursively scans a directory for *.md files (skipping dotted folders like .git), parses and validates each one, and aggregates the totals. Files that fail validation are still counted and are listed separately rather than skipped silently.

rac stats ./features

Example output:

Portfolio Overview
==================

Features: 12
Requirements: 87
Metrics: 24
Risks: 18

Quality
=======

Features Missing Metrics: 2
  - Trade Alerts
  - Watchlists
Features Missing Risks: 3
  - Trade Alerts
  - Watchlists
  - Onboarding
Average Requirements Per Feature: 7.3
Largest Feature: Bond Dashboard (16 requirements)

Requirements by Feature
=======================

Bond Dashboard      16
Trade Alerts        11
Watchlists           8
Onboarding           3

Invalid Features (1)
  ./features/draft.md — missing-title, missing-requirements

Counts span all parsed files; a feature with only warnings (e.g. no metrics) still counts as valid. Invalid files are listed at the end so they are never silently skipped.

Add --json for machine-readable output. stats exits 0 when the directory has at least one valid feature, 1 if none are valid, and 2 if the path is not a directory. (A --strict flag for failing on any invalid file — handy in CI — is planned.)


Ingest

Convert an existing document into Markdown so it can enter the RAC workflow. Ingestion only converts and preserves structure — it does not decide whether the result is a valid RAC artifact (that is the job of future inspect / normalize commands).

rac ingest spec.docx              # print converted Markdown (preview)
rac ingest spec.docx -o spec.md   # write it to a file
rac ingest spec.docx -o spec.md --force   # overwrite an existing file
rac ingest spec.docx --json       # { source, converter, output, markdown }

Conversion is powered by MarkItDown, installed via optional extras — split by format so you only pull the readers you need:

Extra Adds Formats
ingest markitdown[docx] DOCX, HTML, Markdown
ingest-pdf markitdown[pdf] + PDF
ingest-office markitdown[pptx,xlsx,xls] + PPTX, XLSX, XLS
ingest-all everything above all supported formats
pip install "requirements-as-code[ingest]"       # DOCX + HTML + Markdown
pip install "requirements-as-code[ingest-all]"    # everything

HTML and Markdown need no extra (HTML is built into MarkItDown; Markdown is a pass-through). If a file's reader isn't installed, rac ingest tells you exactly which extra to install. Converters live behind a DocumentConverter abstraction, so new sources can be added without changing the CLI.

ingest exits 0 on success, 1 if a recognized document fails to convert, and 2 for usage errors (file not found, unsupported type, missing ingest extra, or an existing output file without --force).


Review (Planned)

AI-assisted product review.

rac review feature.md

Potential checks:

  • Missing requirements
  • Missing risks
  • Ambiguity
  • Product concerns
  • Engineering concerns

RAC will use the user's configured AI provider rather than requiring hosted infrastructure.


Roadmap

v0.1

  • Markdown parser
  • Product Model (AST)
  • Validation
  • Diffing
  • CLI

v0.2

  • Portfolio statistics
  • Quality metrics
  • Repository-wide analysis

v0.3

  • AI review
  • Provider abstraction
  • Git-aware workflows

v1.0

  • Product intelligence
  • Daily product briefs
  • VS Code integration

Contributing

Contributions, ideas, and feedback are welcome.

The project is intentionally focused on one goal:

Treat product requirements like code.


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

requirements_as_code-0.3.1.tar.gz (38.8 kB view details)

Uploaded Source

Built Distribution

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

requirements_as_code-0.3.1-py3-none-any.whl (21.3 kB view details)

Uploaded Python 3

File details

Details for the file requirements_as_code-0.3.1.tar.gz.

File metadata

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

File hashes

Hashes for requirements_as_code-0.3.1.tar.gz
Algorithm Hash digest
SHA256 33ce053b89c1839971ba22717ea581236e702ede8b3994af93963fd2d22a22bd
MD5 4a4ff4cc70ba02e5358475df1d721f01
BLAKE2b-256 771c219631a7a1fb0335f619c67cbfde13e4ce79d3f875033375e216f61d0051

See more details on using hashes here.

Provenance

The following attestation bundles were made for requirements_as_code-0.3.1.tar.gz:

Publisher: python-publish.yml on tcballard/requirements-as-code

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

File details

Details for the file requirements_as_code-0.3.1-py3-none-any.whl.

File metadata

File hashes

Hashes for requirements_as_code-0.3.1-py3-none-any.whl
Algorithm Hash digest
SHA256 fe572093fbb4316eb03979c30d2d3c36fb0bc33cd4210d88a997779da5e02362
MD5 acd1697a046372334839c01b2914d93a
BLAKE2b-256 187529c3154be1132bb29ce354b5a472ff49c93fdc52b005a838e41551001721

See more details on using hashes here.

Provenance

The following attestation bundles were made for requirements_as_code-0.3.1-py3-none-any.whl:

Publisher: python-publish.yml on tcballard/requirements-as-code

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