The project is in a healthy, maintained state
BDD story testing library for Ruby. Tests and documentation from the same code.
2005
2006
2007
2008
2009
2010
2011
2012
2013
2014
2015
2016
2017
2018
2019
2020
2021
2022
2023
2024
2025
2026
 Dependencies

Development

~> 5.0
~> 13.0
~> 1.50
~> 3.13
 Project Readme

executable-stories

Executable stories without Cucumber across JavaScript/TypeScript and non-JS test frameworks, with generated documentation and report outputs.

Why not Cucumber?

This project Cucumber
Write TypeScript Write Gherkin feature files
Steps are inline functions Steps matched by regex
Normal variables and closures World object and shared state
Docs generated from test runs Separate documentation pipeline

One source of truth. Code that executes. Docs that do not lie.

What you get

  • Scenario API built on your test runner's native primitives
  • given(), when(), then(), and(), but() helpers that register real tests
  • Reporters/formatters that generate Markdown, HTML, JUnit XML, and Cucumber outputs from test metadata
  • Output readable by developers and stakeholders

If a test is skipped, failed, or todo, the docs reflect that.

Packages

Package Ecosystem Install / Usage
executable-stories-jest Jest 30+ npm i -D executable-stories-jest
executable-stories-vitest Vitest 4+ npm i -D executable-stories-vitest
executable-stories-playwright Playwright 1.58+ npm i -D executable-stories-playwright
executable-stories-cypress Cypress 13+ npm i -D executable-stories-cypress
executable-stories-ruby Ruby / Minitest Ruby gem/package in repo
executable-stories-go Go testing Go module in repo
executable-stories-rust Rust Rust crate in repo
executable-stories-pytest Python / pytest Python package in repo
executable-stories-junit5 Kotlin / JUnit 5 JVM module in repo
executable-stories-xunit C# / xUnit .NET package in repo
executable-stories-formatters Cross-runner formatter CLI npm i -D executable-stories-formatters
executable-stories-react React StoryReport renderer npm i executable-stories-react
executable-stories-mcp Read-only MCP behavior tools npm i -D executable-stories-mcp
executable-stories-init JS/TS onboarding CLI npm i -D executable-stories-init
executable-stories-demo Demo site/report tooling workspace package
eslint-plugin-executable-stories-vitest ESLint plugin (Vitest) npm i -D eslint-plugin-executable-stories-vitest
eslint-plugin-executable-stories-jest ESLint plugin (Jest) npm i -D eslint-plugin-executable-stories-jest
eslint-plugin-executable-stories-playwright ESLint plugin (Playwright) npm i -D eslint-plugin-executable-stories-playwright
eslint-config Shared ESLint config workspace package

Example apps: apps/jest-example, apps/vitest-example, apps/playwright-example, apps/cypress-example, apps/vite-plus-example, apps/junit5-example (Java 21, Maven; verification: pnpm run verify:junit5), apps/pytest-example (Python 3.12+, pytest; verification: pnpm run verify:pytest), apps/go-example (Go 1.22+; verification: pnpm run verify:go), apps/rust-example (Rust 1.75+; verification: pnpm run verify:rust), apps/xunit-example (.NET 8, xUnit; verification: pnpm run verify:xunit).

Features matrix

The matrix below covers the JS/TS adapters. The same story structure and doc model are mirrored across the Go, Ruby, Rust, pytest, JUnit 5 (Kotlin), and xUnit (C#) adapters — see the cross-language parity policy.

Feature Jest Vitest Playwright Cypress
API story.init() + story.given / story.when / story.then; top-level step helpers also exported story.init(task) + story.given / story.when / story.then; no top-level then export story.init(testInfo) + story.given / story.when / story.then; top-level step helpers also exported story.init() + story.given / story.when / story.then; top-level step helpers also exported
Step modifiers .skip .only .todo .fails .concurrent .skip .only .todo .fails .concurrent .skip .only .fixme .todo .fail .slow .skip .only .todo .fails .concurrent
Scenario modifiers story.skip story.only story.skip story.only story.skip story.only story.fixme story.slow story.skip story.only
Output modes Colocated, aggregated, mixed Colocated, aggregated, mixed Colocated, aggregated, mixed Colocated, aggregated, mixed
Rich step docs ✅ note, kv, code, table, link, section, mermaid, screenshot, video, runtime, custom ✅ same ✅ same ✅ same
Embedded HTML (story.html) ✅ path / url / content → sandboxed iframe ✅ path / url / content → sandboxed iframe ✅ same (local files inlined at capture time) ✅ path / url / content → sandboxed iframe
Scenario options tags, meta, ticket, traceUrlTemplate tags, meta, ticket, traceUrlTemplate tags, meta, ticket, traceUrlTemplate tags, meta, ticket, traceUrlTemplate
OTel trace link ✅ auto-detect via @opentelemetry/api ✅ same ✅ same — (browser env)
OTel trace waterfall ✅ via autotel task.meta.otelSpans ✅ via autotel otel-spans annotation
Attach story to plain it/test story.init() inside test() story.init(task) with it(..., ({ task }) => ...) story.init(testInfo) inside test() story.init() or doc.story("Title") inside it()
Step callbacks story.given('text', () => ...) on all steps ✅ same ✅ same ✅ same
AAA aliases arrange/act/assert, setup/context, etc. arrange/act/assert, setup/context, etc. arrange/act/assert, setup/context, etc. arrange/act/assert, setup/context, etc.
CLI collate
CI detection (formatter CLI) ✅ (report meta: branch, commit, build URL)
Notifications (formatter CLI) Slack, Teams, webhook; --notify same same same
Run history (formatter CLI) --history-file → flakiness, stability, perf in HTML same same same
GitHub Actions summary
Custom doc renderers

For per-framework behaviour and guarantees (entry point, mental model, modifiers, framework-native attach), see: Jest — Developer experience, Vitest — Developer experience, Playwright — Developer experience, Cypress.

Details and reporter options: see each package's README.

OTel trace link is also supported in the non-JS adapters: Go (WithTraceUrlTemplate), Python (trace_url_template), Kotlin/JUnit5 (traceUrlTemplate parameter or env var), Rust (with_trace_url_template, requires otel feature), and C#/xUnit (Story.WithTraceUrlTemplate() or env var). All adapters auto-detect an active span and inject trace ID docs bidirectionally. Set OTEL_TRACE_URL_TEMPLATE (with {traceId} placeholder) to generate clickable trace links in reports.

Step timing (startTimer/endTimer) is supported in all non-JS adapters: Go (s.StartTimer()/s.EndTimer(token)), Python (story.start_timer()/story.end_timer(token)), Kotlin/JUnit5 (Story.startTimer()/Story.endTimer(token)), Rust (story.start_timer()/story.end_timer(token)), and C#/xUnit (Story.StartTimer()/Story.EndTimer(token)). The JS adapters record step timing automatically via story.fn() / story.expect() wrappers and step callbacks.

Quick example

Jest (story.init() plus step markers):

import { expect, it } from '@jest/globals';
import { story } from 'executable-stories-jest';

it('User logs in', () => {
  story.init();
  story.given('user is on login page');
  story.when('user submits valid credentials');
  story.then('user sees the dashboard', () => {
    expect(true).toBe(true); // or real assertion
  });
});

Playwright uses the same story.given / story.when / story.then style, but pass testInfo to story.init(testInfo).

Cypress (call story.init() at the start of each it, then use step markers; see Cypress README).

Vitest (story.init(task); no top-level then):

import { expect, it } from 'vitest';
import { story } from 'executable-stories-vitest';

it('User logs in', ({ task }) => {
  story.init(task);
  story.given('user is on login page');
  story.when('user submits valid credentials');
  story.then('user sees the dashboard', () => {
    expect(true).toBe(true);
  });
});

Playwright step callbacks can use fixtures: given("...", async ({ page }) => { await page.goto("/login"); });

Generated Markdown:

### User logs in

- **Given** user is on login page
- **When** user submits valid credentials
- **Then** user sees the dashboard

Getting started

  1. Install the package for your test runner
  2. Add the reporter to your config
  3. Run your tests
  4. Open the generated Markdown

The reporter writes reports/raw-run.json; turn it into any report format with the CLI:

npx --package executable-stories-formatters executable-stories format reports/raw-run.json --format html,markdown

See each package's README for detailed setup instructions.

Agent workflows: Publish StoryReport JSON and a scenario index from CI — see the agent artifact contract and MCP server guide. Package roles: package map. Cross-language parity policy: parity matrix.

Living documentation site

Generate a multi-page Astro site from your stories — one page per story file plus a scenario Explorer. It auto-picks-up: add a *.story.test.ts and re-run, its page appears; delete it and the page is pruned. Zero per-test wiring.

The executable-stories CLI ships in the executable-stories-formatters package (install it, or invoke via npx --package executable-stories-formatters executable-stories …).

npx --package executable-stories-formatters executable-stories init-astro site   # scaffold the Astro/Starlight site
# vitest.config: createStoryReporter({ rawRunPath: 'reports/raw-run.json' })
pnpm test                                      # writes reports/raw-run.json (auto-includes all stories)
npx --package executable-stories-formatters executable-stories build-docs reports/raw-run.json --site-dir site
cd site && npm install && npm run build        # static dist/

build-docs is the headline command — format --format astro is a lower-level primitive that emits a single aggregated page, not a site.

Development

From the repo root: pnpm quality runs build, lint, type-check, and test for all packages.

For contributor and AI agent guidance (conventions, framework APIs, ESLint plugins, verification), see AGENTS.md. CLAUDE.md is a symlink to the same file. Example apps in apps/ use the workspace packages. JUnit 5, pytest, Go, Rust, Ruby, and xUnit example apps are not part of pnpm quality. When Java 21 and Maven are available (e.g. in the devcontainer), run pnpm run verify:junit5 to run junit5-example. When Python 3.12+ is available, run pnpm run verify:pytest to run pytest-example. When Go 1.22+ is available, run pnpm run verify:go to run go-example. When Rust is available, run pnpm run verify:rust to run rust-example. When Ruby and Bundler are available, run pnpm run verify:ruby for executable-stories-ruby. When .NET 8 is available, run pnpm run verify:xunit to run xunit-example.

Formatters standalone binary

The executable-stories-formatters package (CLI for generating reports from test results JSON) supports filtering by source file (--include / --exclude), CI auto-detection (GitHub Actions, GitLab, CircleCI, Azure DevOps, Buildkite, Jenkins, Travis) so reports include branch, commit, and build links, notifications (Slack, Teams, or generic webhook with optional HMAC signing; --notify always|on-failure|never), and run history (--history-file) for flakiness, stability, and performance trends in the HTML report. See formatters README. The HTML report highlights step parameters (quoted strings and numbers) for readability. The package can be built as a single standalone binary with Bun:

cd packages/executable-stories-formatters && bun run compile

This produces an executable-stories binary in that package directory. CI builds the binary for the runner platform and uploads it as an artifact (executable-stories-linux-x64). The Release workflow builds multi-platform binaries (linux-x64, linux-arm64, darwin-x64, darwin-arm64, windows-x64) and uploads them as the formatters-binaries artifact.

License

MIT