MarkSpec — Markdown Flavor Specification
Introduction
MarkSpec is a Markdown flavor for traceable industrial documentation. It targets safety-critical and high-availability systems where requirements, traceability, and compliance documentation must live alongside code in version control.
MarkSpec is a three-layer stack:
- CommonMark — the parsing baseline.
- GFM / GLFM shared subset — platform extensions portable across GitHub and GitLab.
- MarkSpec extensions — entry authoring, captions, inline references, directives, and book structure.
Source files are pure, readable Markdown. They render correctly on GitHub and GitLab without any build step. PDF generation, traceability matrices, and reference resolution are build concerns — not format concerns.
Core vs. profile
MarkSpec splits responsibility between a core — this specification — and a profile — an external vocabulary pack that names concrete entry types and declares their attributes, relation names, and rules.
- The core defines two entry shapes (Authored and Reference), the syntax for
authoring entries, a single identity attribute (
Id:), a small universal attribute set, and the rules for discriminating shape fromId:value format. It contains no type vocabulary. - A profile declares concrete types (
requirement,test,unit,standard,dependency, …) within the two shapes, per-type attributes, traceability relations, and validation rules.
A bundled default profile ships with MarkSpec and loads by default (it can
be opted out of in .markspec.yaml). Compliance profiles (ASPICE, ISO 26262,
DO-178C, IEC 62304, MISRA-C) stack on top via an extends: chain.
This specification is the normative reference for the MarkSpec core. Profile
schemas are specified by each profile’s markspec.yaml manifest; the default
profile’s manifest is the reference for the out-of-box type vocabulary.
Part 1 — Markdown Flavor
1.1 CommonMark
MarkSpec accepts all CommonMark syntax. The following features are supported without modification:
Headings:
# Document Title
## Section
### Subsection
Paragraphs and inline formatting:
The braking system shall achieve full braking force within 150ms of driver
input. The _debounce window_ is configurable. Use **bold** for emphasis and
`debounce_input()` for code.
Block quotes:
> The system shall meet all requirements specified in ISO 26262-6 §9.4 for
> software unit testing.
Lists:
- Pressure sensor
- Speed sensor
- Temperature sensor
1. Capture raw input
2. Apply debounce filter
3. Validate plausible range
Fenced code blocks (language required):
```rust
fn debounce_input(raw: u16) -> u16 {
// implementation
}
```
Inline links:
See [ISO 26262-6](https://www.iso.org/standard/68388.html) for software-level
requirements.
Reference links:
See [ISO 26262-6] for software-level requirements.
[ISO 26262-6]: https://www.iso.org/standard/68388.html
Images (alt text required):

Hard line breaks (trailing \):
First line ends here,\
and the next line continues.
Horizontal rules:
---
HTML comments (used for directives):
<!-- markspec:ignore -->
MarkSpec restricts the following CommonMark features:
| Feature | CommonMark | MarkSpec restriction |
|---|---|---|
| Headings | ATX and setext | ATX only. |
| Code blocks | Fenced (backtick and tilde) and indented | Backtick-fenced only. |
| Emphasis | *text* and _text_ | _text_ only. |
| Strong | **text** and __text__ | **text** only. |
| List markers | -, *, + | - only. |
| Horizontal rules | ---, ***, ___ | --- only. |
| Hard line breaks | Trailing \ and trailing double-space | Trailing \ only. |
| Inline HTML | Any HTML element | Comments only (<!-- -->). No HTML elements. |
| Front matter | YAML --- blocks (not CommonMark) | YAML (---) and TOML (+++) allowed at the top of the file; schema defined in §1.3 §6 and Part 6. |
MarkSpec requires beyond CommonMark minimums:
| Requirement | Rule |
|---|---|
| First line | Must be an H1 heading. |
| H1 count | Exactly one H1 per file. Summary documents are exempt — additional H1s are part headings. |
| Heading levels | Must not skip (H2 → H4 is invalid). |
| Code fence language | Required. Use text for plain output. |
| Image alt text | Required on every image. |
1.2 GFM / GLFM shared subset
Only features supported by both GFM and GLFM are used. Platform-specific extensions are not part of the flavor.
Tables
Pipe syntax. Rows are exempt from line width limits.
| Sensor | Min | Max | Unit |
| -------- | --- | ---- | ---- |
| Pressure | 0 | 1023 | kPa |
| Speed | 0 | 255 | km/h |
Strikethrough
~~deprecated requirement~~
Task lists
- [x] Define sensor thresholds
- [ ] Validate against hardware spec
- [ ] Update traceability matrix
Footnotes
Supplementary context only — not for traceability.
The debounce window[^1] shall be configurable per sensor type.
[^1]: Debouncing eliminates transient electrical noise from raw sensor readings.
Syntax highlighting
Language identifier is required on all fenced code blocks.
```rust
fn debounce_input(raw: u16, window_ms: u32) -> u16 {
// implementation
}
```
Math
Inline and block math expressions:
The response time is $t = 150\text{ms}$ under nominal conditions.
$$
d = v \cdot t
$$
Alerts
> [!NOTE]
> This requirement derives from ISO 26262-6 §9.4.
> [!WARNING]
> Failure to debounce may lead to spurious brake activation.
> [!CAUTION]
> **ASIL-B constraint** — changes require impact analysis.
Supported types: NOTE, TIP, IMPORTANT, WARNING, CAUTION. Custom title
via bold text on the first line.
1.3 MarkSpec extensions
All extensions use valid CommonMark syntax — they render on GitHub and GitLab without tooling.
§1 Entry blocks
A list item starting with - [DISPLAY_ID] followed by a title on the same line,
and indented body content on subsequent lines. The display ID is the entry’s
human-readable identifier. The title is the rest of the first line after the
closing ].
- [DISPLAY_ID] Title
Body paragraphs.
Key: Value
Key: Value
A - [DISPLAY_ID] with no indented body is a normal list item — not an entry
block.
Example 1 — entry block:
- [SRS_BRK_0107] Sensor input debouncing
The sensor driver shall debounce raw inputs to eliminate electrical noise
before processing.
The debounce window shall be configurable per sensor type:
| Sensor type | Window (ms) | Sample rate (Hz) |
| ----------- | ----------- | ---------------- |
| Pressure | 10 | 100 |
| Speed | 5 | 200 |
| Temperature | 50 | 20 |
Id: 01HGW2Q8MNP3RSTVWXYZABCDE
type: software-requirement
Satisfies: SYS_BRK_0042
Labels: ASIL-B
Example 2 — not an entry block:
- [See documentation] for details on configuration.
No indented body. Normal list item.
Emphasis (_text_) must not appear inside entry blocks. Strong (**text**) and
inline code are allowed.
Part 2 defines the two entry shapes (Authored, Reference), the rule that discriminates them, the universal attributes that apply to both, and the profile-declared extensions layered on top.
Rendering of entry blocks (admonition-style left border, type coloring, label pills, cross-reference links) is specified in the Typography chapter, §“Entry rendering”.
§2 Attribute blocks
An attribute block is the trailing indented code block of an entry. Each
content line is a single Key: Value pair. No trailing line-continuation
characters.
The block is indented 4 spaces relative to the entry body indent (CommonMark
indented-code-block rule). Inside a Markdown list item, that means 6 spaces of
indentation before the Key; inside a source-file doc comment (no enclosing
list), 4 spaces of indentation relative to the comment content column.
Example 3 — attribute block:
- [SRS_BRK_0001] Sensor debouncing
Sensor driver shall debounce raw inputs.
Id: 01HGW2Q8MNP3RSTVWXYZABCDE
Satisfies: SYS_BRK_0042
Labels: ASIL-B
The set of valid attributes is the universal set (Part 2 §2.1) plus whatever the active profile declares for the entry’s shape and inferred type.
Generated attributes (build-time inverses of authored relations such as
Verified-by from Verifies, Cited-by from References) are computed by
tooling and never appear in source. The exact set is profile-declared.
Disambiguation from body code blocks. The trailing indented code block
qualifies as an attribute block only when every one of its content lines matches
the pattern ^[A-Z][A-Za-z-]*: (followed by a space). Otherwise it remains a
regular code block and the entry is treated as having no attribute block. Fenced
code blocks (triple-backtick fences) anywhere in the entry are body content and
never confused with the attribute block — different syntactic shape.
Trailing position is required. If body prose appears after an indented
Key: Value block, that block is not trailing — it is treated as a regular code
block with no attribute meaning. Authors must place attributes at the very end
of the entry.
Backward compatibility. During the transition, the parser also accepts the
legacy paragraph-with-trailing-\ shape. Running markspec format rewrites
legacy blocks to the canonical indented form. The legacy shape emits a
deprecation diagnostic (MSL-DEPRECATED-ATTR-001) and will be removed in a
future major release.
§3 Table captions
Emphasized paragraph starting with Table: immediately above a pipe table.
Example 4 — table with caption:
_Table: Sensor thresholds_
| Sensor | Min | Max |
| -------- | --- | ---- |
| Pressure | 0 | 1023 |
Slug: tbl.sensor-thresholds. Derived by stripping the Table: prefix, then
applying the GFM anchor algorithm (lowercase, spaces to hyphens, punctuation
stripped).
Example 5 — not a caption:
_This is just italic text._
| Column A | Column B |
| -------- | -------- |
Does not start with Table:.
§4 Figure captions and diagrams
Diagrams are embedded using standard Markdown image syntax with relative paths only:

Absolute URLs (https://…), repo-root links (/docs/…), and paths that escape
the document folder via repeated ../../ are not permitted. Relative paths keep
the document self-contained — when a folder is moved or reorganized, the diagram
travels with the document. Non-relative image references are flagged by
MSL-D008.
Diagrams are always stored as SVG files — never embedded as inline fenced
code blocks (e.g., ```mermaid). SVG renders consistently across GitHub,
GitLab, PDF, and presentation output.
Authoring recommendations by use case:
- PlantUML — simple structured diagrams: sequences, state machines, class diagrams with ~13 classes or fewer.
- draw.io (or Inkscape, Excalidraw) — advanced authoring with free-form shapes, swimlanes, complex layouts.
- Raw SVG — AI-assisted authoring, scripts, or hand-authored.
Storage conventions:
- Source embedded in the SVG →
<name>.<source>.svg(e.g.architecture.drawio.svg,unlock-sequence.plantuml.svg). - Source not embedded → source and SVG side by side (e.g.
architecture.dot+architecture.svg).
PNG is acceptable when SVG does not make sense — photographs, screenshots,
heatmaps, dense bitmap data. Use a descriptive filename with no source-format
suffix (dashboard-screenshot.png).
For sizing, visual style, and tooling details (PlantUML viewport, draw.io embedding, color palettes, stroke weights), see the Typography chapter.
Captions: an emphasized paragraph starting with Figure: immediately below
an image. Alternatively, the image alt text is the caption.
Example 6 — explicit caption:

_Figure: High-level architecture of the braking system_
Slug: fig.high-level-architecture-of-the-braking-system. Derived by stripping
the Figure: prefix, then applying the GFM anchor algorithm.
Example 7 — alt text as caption:

Slug: fig.system-overview. Explicit caption takes precedence.
§5 In-code entries
Entries can be authored in doc comments in source files. A doc comment beginning
with [DISPLAY_ID] is recognized as a MarkSpec entry. The leading - bullet is
optional in doc comments — the bracket pattern alone is sufficient.
Example 8 — Rust doc comment test entry:
#![allow(unused)]
fn main() {
/// [SWT_BRK_0107] Debounce unit test
///
/// Given a debounce window of 10ms, a transient spike shorter
/// than the window must not alter the stable output.
///
/// Id: 01HGW3R9QLP4ABCDEFGHJKMNPQ
/// Verifies: SRS_BRK_0107
/// Tests: braking_core::controller::debounce_input
/// Labels: ASIL-B
#[test]
fn swt_brk_0107_debounce_filters_noise() {
// test implementation
}
}
The doc comment declares the entry; the function body is the executable
artifact. The SWT_ prefix triggers the active profile’s type inference (e.g.,
type: unit-test under an ASPICE profile). Authors do not write type: in
source. The file path is observable as the file.path property (see Part 6).
A production unit declares what it realizes via its own doc comment. Because the
display ID is a symbolic namespace path with no declared prefix pattern, the
author writes type: explicitly:
#![allow(unused)]
fn main() {
/// [braking_core::controller::debounce_input] Debounce function
///
/// Rejects transient noise on raw sensor readings.
///
/// Id: 01HGW3D6QRST7IJKLMNOPQRSTUV
/// type: unit
/// Realizes: 01HGW2Q8MNP3RSTVWXYZABCDEF
fn debounce_input(raw: u16) -> u16 { ... }
}
Tooling extracts these doc comments to produce the same traceability output as Markdown-authored entries.
§6 Front matter
Document-level metadata is authored in YAML front matter — a ----delimited
block at the very top of the file, before the H1.
Example 8b — document with front matter:
---
document-id: 01HGW2D0DOCPQ4FGHIJKLMNOPQR
document-type: requirements
labels: [requirements, ASIL-B]
external-id: doors:VHC:SRS-BRK
---
# Braking Software Requirements
## Introduction
...
Front matter carries:
- Document identity (
document-id,document-type). - Universal attributes (
labels,external-id,supersedes,deprecated,references) — same set and semantics as entry-level universal attributes. Draft state viaDRAFTlabel; retirement viasupersedes:ordeprecated:. - A reserved
metadata:map for org-specific free-form fields. - Optional profile-declared keys (e.g., automotive
asil:). - Optional allowlisted ecosystem keys declared in
.markspec.yaml(for Hugo / Jekyll / Docusaurus interop).
Forbidden keys (Markdown-native concepts; duplication creates drift):
title, description, toc, sections, authors, author, date,
created, modified, cover, images. These live in H1, body paragraphs, or
git history — never in front matter.
Casing convention: front matter keys use kebab-case (document-id),
matching YAML ecosystem convention. Entry trailers keep Title-Case (Id:,
Satisfies:, Labels:), matching git-trailers convention.
TOML tolerance: +++-delimited TOML front matter is accepted as input (for
GitLab-flavored Markdown parity); the formatter normalizes to YAML.
Full document-structure specification: see Part 6 — Document Model.
Part 2 — Entry Shapes
Part 1 defines the format — how to write entry blocks and attribute blocks. This
part defines the core entry model: the two shapes the core recognizes, the
single Id: identity attribute, the value-format rule that discriminates shape,
the universal attributes shared between shapes, and the relationship between the
core and profile-declared type vocabulary.
MarkSpec recognizes two shapes:
| Shape | Intent | Id: value | Display ID role |
|---|---|---|---|
| Authored | Content the project authors and owns | ULID (26-char Crockford b32) | Human-readable alias |
| Reference | Citation pointing to an external work | URI (RFC 3986, scheme req.) | Slug (pandoc/BibTeX cite-key) |
Every entry carries exactly one Id: attribute. Its value format determines
the shape:
Id: 01HGW2P4KFR7ABCDEFGHJKMNPQ # ULID → Authored
Id: urn:iso:std:iso:26262:-6:ed-2 # URI → Reference
Id: pkg:cargo/serde@1.0.0 # URI (purl) → Reference
Id: doi:10.1109/IEEESTD.2008.4610935 # URI → Reference
The two value formats are visually disjoint — a ULID has no scheme; a URI must
carry a scheme followed by :. A bare slug (no scheme, not a ULID) is rejected
as an Id: value.
Concrete types (requirement, test, unit, standard, dependency,
hazard, …) are declared by the active profile, not by the core. The core has
no TYPE vocabulary and no family enum.
2.1 Universal attributes
The following attributes apply to every family:
| Attribute | Type | Required | Description |
|---|---|---|---|
Labels | tag-list | no | Classification tags (includes DRAFT marker) |
References | citation | no | External reference citations with locator |
External-id | external-id | no | Cross-system identifier(s) |
Supersedes | id | no | Same-shape entry this one replaces |
Superseded-by | id | — | Generated inverse of Supersedes |
Deprecated | string | no | Retirement reason (non-replacement case) |
Draft state is carried by the DRAFT label — a plain universal tag with no
exclusive-group semantics. Authors set Labels: DRAFT on entries that are
merged but not yet authoritative.
Retirement is structural, expressed two ways:
- Replacement — successor entry authors
Supersedes: <predecessor-id>; the predecessor automatically gains the generatedSuperseded-by:inverse. - Non-replacement — entry authors
Deprecated: "<free-text reason>"(e.g., “Feature cut from scope in v3.0”).
An entry is retired when either signal is present. The two are complementary
(a replacement may still carry a Deprecated: reason for additional context).
Tooling emits severity-tiered diagnostics when a relation target is retired or
draft: DRAFT target → info, retired target → warning, unresolved target →
error. There is no DEPRECATED / WITHDRAWN label — retirement lives in
Supersedes and Deprecated. Supersedes operates within a shape: an Authored
entry supersedes an Authored entry; a Reference entry supersedes a Reference
entry.
See §2.5 for attribute value types (multi-line repeat vs CSV, canonical form).
2.2 Authored entries
An Authored entry is a content unit the project authors and owns: a requirement, a test, a rule, a component, a code unit, a hardware part, a glossary term, a hazard, a design decision. Its identity is project-local and machine-generated.
Display ID: any non-empty, project-unique string. The core does not
constrain format. Profiles tighten by declaring per-type display-id-pattern:
templates; the active profile determines what shape Authored-entry display IDs
take in a given project. Common conventions:
SRS_BRK_0107 ← typed prefix + scope + number
braking_core::controller::debounce_input ← symbolic namespace path
REQ-042, NOTE-007 ← simple typed prefix + number
Identity: Id: carries a bare 26-character ULID
(^[0-9A-HJKMNP-TV-Z]{26}$). Assigned by markspec format, never
hand-authored, immutable once assigned. The ULID is the stable identity; the
display ID is a renumberable alias that resolves through the ULID for
cross-references.
Type: profile-declared. Normally inferred by the active profile from the
display-ID prefix (SRS_BRK_0107 → type: software-requirement under an ASPICE
profile that declares
software-requirement: display-id-pattern: "SRS_{scope}_{n:04d}"). An explicit
type: attribute in source overrides inference and is required when the display
ID matches no declared pattern.
Profile-declared attributes: every attribute beyond the universal set (§2.1) is declared by the active profile. Examples — declared by an automotive ASPICE profile, not by the core:
Derived-from,Satisfies,Allocated-toon requirementsVerifies,Tests,Test-levelon testsRealizes,Depends-on,Part-of,Element-kindon units / artifactsASIL,Safety-goal,Risk-classon hazards / safety-relevant entries
The core does not define any of these names. They live in the profile manifest
and are validated against the profile’s attributes: / traceability:
declarations.
Example 9 — inferred type (SRS prefix):
- [SRS_BRK_0107] Sensor input debouncing
The sensor driver shall debounce raw inputs to eliminate electrical noise
before processing.
Id: 01HGW2Q8MNP3RSTVWXYZABCDE
Derived-from: SYS_BRK_0042
Labels: ASIL-B
Example 10 — explicit type override (symbolic path):
- [braking_core::controller::debounce_input] Debounce function
Rejects transient noise on raw sensor readings using a configurable window.
Id: 01HGW3D6QRST7IJKLMNOPQRSTUV
type: unit
Realizes: 01HGW2Q8MNP3RSTVWXYZABCDE
The author writes type: unit because no display-id-pattern matches a
symbolic namespace path. Realizes: references the upstream entry by its ULID
identity (the stable handle), not by display ID.
2.3 Reference entries
A Reference entry is a bibliographic citation of an external artifact: a standard, a regulation, a paper, an RFC, a corporate specification, a package dependency, a hardware part from an external catalog.
Display ID (slug): matches ^[A-Za-z]([A-Za-z0-9._/-]*[A-Za-z0-9])?$
(pandoc/BibTeX-style cite-key, restricted to a portable character set). Common
conventions:
ISO-26262-6 ← ISO 26262-6:2018
ISO/IEC-15504 ← ISO/IEC 15504
DO-178C ← RTCA DO-178C
RFC-2119 ← IETF RFC
serde ← Rust crate (dependency)
smith2021 ← academic citation
Both [@ISO-26262-6] and [ISO-26262-6] are accepted in the entry header; the
leading @ is pandoc-citation sugar and is stripped during parsing. Inline
pandoc citations [@ISO-26262-6] in prose resolve to the matching Reference
entry.
Identity: Id: carries a URI per RFC 3986 — any scheme, but the scheme is
required (urn:, doi:, pkg:, https:, isbn:, …):
Id: urn:iso:std:iso:26262:-6:ed-2 ← URN (preferred for standards)
Id: doi:10.1109/IEEESTD.2008.4610935 ← DOI (preferred for papers)
Id: pkg:cargo/serde@1.0.0 ← purl (Package URL, for dependencies)
Id: isbn:9780132350884 ← ISBN
Id: https://www.rfc-editor.org/rfc/rfc2119
Author-provided, not tooling-generated. A bare slug (no scheme) is rejected as
an Id: value — the slug lives in the display ID, not in Id:.
Type: profile-declared. Normally inferred by the active profile from the URI
scheme or the display ID (Id: pkg:cargo/... → type: dependency;
Id: urn:iso:... → type: standard). An explicit type: overrides inference.
Body is optional for Reference entries — a minimal entry may consist of
display ID, title, and Id: only.
Universal attributes (§2.1) apply, except that References: is not
applicable to Reference entries (a Reference entry does not itself cite other
Reference entries via References:; the replacement relation is expressed via
the universal Supersedes).
Profile-declared attributes: profiles may declare convenience attributes for
Reference entries — Reference-url: (HTTPS navigation link when different from
the canonical Id:), Reference-document: (canonical citation string),
License: (SPDX license expression for dependencies), etc.
Example 11 — normative standard:
- [@ISO-26262-6] ISO 26262 Part 6
Road vehicles — Functional safety — Part 6: Software level.
Id: urn:iso:std:iso:26262:-6:ed-2
Reference-url: https://www.iso.org/standard/68383.html
Reference-document: ISO 26262-6:2018
Labels: functional-safety, automotive
Example 12 — dependency (purl):
- [serde] serde Rust serialization framework
Id: pkg:cargo/serde@1.0.0
License: Apache-2.0 OR MIT
2.4 Shape discrimination
The shape of an entry is determined by the value format of its Id:
attribute. An entry has exactly one Id:.
if Id matches ULID regex (^[0-9A-HJKMNP-TV-Z]{26}$) → Authored
if Id is a scheme-qualified URI (RFC 3986) → Reference
otherwise → validation error
Properties of this rule:
- Disjoint — ULIDs and URIs do not overlap. A ULID has no scheme; a URI
requires a scheme followed by
:. No value matches both. - Complete — every well-formed
Id:value is either a ULID or a URI; the two exhaust the accepted formats. - Independent of display ID — shape is decided by the
Id:value, not by the display-ID format. - Independent of document context — shape is intrinsic to the entry, not dependent on which document it appears in.
- Independent of profile — shape resolution completes without consulting any profile.
When a new entry is authored without an Id: attribute, markspec format
classifies it using a heuristic on the display ID and the document directive
(see Part 3), then either mints a fresh ULID (Authored) or prompts for a URI
(Reference). Once Id: is assigned, the shape is fixed by the value’s format.
2.5 Attribute value types
Every attribute declares a value type that determines which input forms the parser accepts and which form the formatter produces.
| Type | Cardinality | Multi-line repeat | CSV on one line | Description |
|---|---|---|---|---|
id | single | — | — | Display ID or slug |
id-list | repeatable | ✓ | ✓ | Multiple identifiers |
uri | single | — | — | URI per RFC 3986 (URN, DOI, HTTPS URL) |
url | single | — | — | HTTPS navigation link |
path | single | — | — | Filesystem path |
path-or-id | single | — | — | Filesystem path or element display ID |
enum | single | — | — | One value from a closed vocabulary |
tag-list | repeatable | ✓ | ✓ | Free-form tags |
text | single | — | — | Free-form single-line text |
citation | repeatable | ✓ | ✗ | Slug + optional free-text locator (locator may contain ,) |
external-id | repeatable | ✓ | ✓ | scheme:value qualified identifier |
integer | single | — | — | Whole number |
date | single | — | — | ISO 8601 date (YYYY-MM-DD) |
boolean | single | — | — | true or false |
Multi-line repeat (git-trailers canonical form):
Derived-from: SYS_BRK_0042
Derived-from: SYS_BRK_0043
Labels: ASIL-B
Labels: safety
CSV on one line (accepted when no value contains a comma):
Derived-from: SYS_BRK_0042, SYS_BRK_0043
Labels: ASIL-B, safety
The formatter rewrites every repeatable attribute to multi-line form. CSV is an accepted input but never a canonical output.
CSV is forbidden for the citation type. References values may carry
free-text locators like §9.4, Table 7 that would be ambiguous in CSV.
Part 3 — Directives
Directives are HTML comments starting with markspec:. They are invisible on
GitHub and GitLab. A markspec: token at the start of a line inside an HTML
comment begins a directive. Everything until the next markspec: or --> is
the payload.
3.1 Syntax
Example 13 — single directive:
<!-- markspec:glossary -->
Example 14 — multiple directives with multiline payload:
<!--
markspec:deck
markspec:references https://safety.company.io/registry
-->
Continuation lines without markspec: are part of the previous directive’s
payload.
Parsing rules:
- Scan each HTML comment for lines starting with
markspec:. - Token after
markspec:is the directive name. - Remainder of line is the start of the payload.
- Lines not starting with
markspec:are payload continuation. - A new
markspec:line or-->terminates the payload. - Range directives closed by
<!-- markspec:end NAME -->.
3.2 Document directives
Placed in the first HTML comment after the H1 heading.
| Directive | Payload | Context |
|---|---|---|
markspec:glossary | none | doc |
markspec:summary | none | doc |
markspec:deck | none | deck |
markspec:specs | none | doc |
markspec:tests | none | doc |
markspec:elements | none | doc |
markspec:references | registry URL | both |
markspec:paginate | none | deck |
Type directives (glossary, summary, deck) are mutually exclusive.
Family-hint directives (specs, tests, elements, references without a
payload) hint at the predominant entry family in the document — used by
markspec format to classify new entries before they carry an identity
attribute. references (with a URL payload) can coexist with any type
directive. doc is the default — no directive for it. Multiple
markspec:references directives with URL payloads declare multiple upstream
registries; order matters, with an implicit fallback to RefHub.
Document-level retirement uses the deprecated: front-matter key (or
supersedes: for replacement retirement), not a directive. See §6.2.
glossary and summary are auto-detected from filename (GLOSSARY.md,
SUMMARY.md). Family-hint directives are auto-detected from filename:
tests.md implies markspec:tests, elements.md implies markspec:elements,
references.md implies markspec:references. The directive is needed when the
file has a different name. deck is never auto-detected — it always requires an
explicit directive.
Example 15 — deck with pagination:
# Architecture Review
<!--
markspec:deck
markspec:paginate
-->
---
## System Boundaries
...
Example 16 — deprecated glossary:
---
document-type: glossary
supersedes: 01HGW2D0GLOSPQ4FGHIJKLMNOPQ # platform-glossary.md document-id
---
# Legacy Terms
Or for a glossary with no successor:
---
document-type: glossary
deprecated: "Archived after platform migration; content no longer maintained."
---
# Legacy Terms
Example 17 — upstream registries:
# Braking Controller
<!--
markspec:references https://safety.company.io/registry
markspec:references https://driftsys.github.io/refhub
-->
Each markspec:references declares one upstream registry. Order matters —
registries are searched first to last. RefHub is the implicit final fallback
even if not declared.
3.3 Inline directives
Placed anywhere in the document body.
| Directive | Payload | Closing | Context |
|---|---|---|---|
markspec:break | page, column | — | both |
markspec:columns | count (2, 3) | markspec:end columns | both |
markspec:section | section name | — | deck |
markspec:notes | free text | inside comment | deck |
markspec:disable | MSL rule ID(s) | markspec:end disable | both |
markspec:disable-next-line | MSL rule ID(s) | — | both |
markspec:ignore | none | markspec:end ignore | both |
markspec:disable opens a range closed by markspec:end disable.
markspec:disable-next-line suppresses rules for the next line only.
markspec:ignore skips all MarkSpec processing — content inside the range is
treated as plain Markdown with no requirement parsing, reference resolution, or
MSL validation.
Example 18 — multi-column layout:
<!-- markspec:columns 2 -->
Content in the first column.
<!-- markspec:break column -->
Content in the second column.
<!-- markspec:end columns -->
In decks, markspec:break column works within a slide without the
markspec:columns range — slide boundaries are the implicit region.
Example 19 — slide section:
---
<!-- markspec:section Architecture -->
## System Boundaries
A high-level view of the braking subsystem
---
## Component Design
...
The section name appears in slide footers until the next markspec:section.
Example 20 — speaker notes:
<!--
markspec:notes
Mention the 150ms response time requirement from
STK_BRK_0001. The debounce window was determined
by bench testing with the Bosch sensor module.
-->
Notes are entirely inside an HTML comment. The --> closes the payload — no
markspec:end notes needed.
Example 21 — page break:
<!-- markspec:break page -->
Example 22 — lint suppression:
<!-- markspec:disable MSL-R011 -->
- [SRS_BRK_0108] Some _legacy_ requirement
<!-- markspec:end disable -->
<!-- markspec:disable-next-line MSL-R011 -->
- [SRS_BRK_0109] Another _legacy_ one
Part 4 — Book Structure
A MarkSpec book organizes MarkSpec files into a single navigable document. The
book is defined by a source directory containing a SUMMARY.md at its root.
4.1 Layout
The book source directory contains the SUMMARY.md and all files it references.
Directory structure is free-form — the SUMMARY.md defines the navigation, not
the file tree.
Example 23 — book source directory:
src/
├── SUMMARY.md
├── GLOSSARY.md
├── overview.md
├── product/
│ ├── stakeholder-requirements.md
│ └── software-requirements/
│ ├── README.md
│ └── braking.md
├── architecture/
│ └── system-architecture.md
└── guide/
└── getting-started.md
4.2 Summary
The summary is a SUMMARY.md file at the root of the book source directory. It
is a manually authored table of contents — not a generated file tree. The author
decides what appears and in what order.
The core structure is a nested list of links. Each link is a chapter. Nesting creates sub-chapters.
Example 24 — SUMMARY.md without parts:
# Braking Controller
- [Overview](overview.md)
- [Requirements](requirements.md)
- [Architecture](architecture.md)
- [Getting Started](getting-started.md)
- [Glossary](GLOSSARY.md)
Optional elements:
- Unnested links — links outside a list. Front matter (before the first
---) and back matter (after the last---). Rendered without numbering. - Part headings — H1 headings (
# Part Name) label groups of chapters. Rendered as unclickable section dividers. - Separators —
---separates front matter, body, and back matter.
Example 25 — SUMMARY.md with front matter, parts, back matter, and annexes:
# Braking Controller
[Overview](overview.md) [Introduction](introduction.md)
---
# Product
- [Stakeholder Requirements](product/stakeholder-requirements.md)
- [System Requirements](product/system-requirements.md)
- [Software Requirements](product/software-requirements/README.md)
- [Braking](product/software-requirements/braking.md)
- [Steering](product/software-requirements/steering.md)
- [Diagnostics](product/software-requirements/diagnostics.md)
# Architecture
- [System Architecture](architecture/system-architecture.md)
- [Software Architecture](architecture/software-architecture.md)
- [Interface Contracts](architecture/interface-contracts.md)
- [Decisions](architecture/decisions/README.md)
- [ADR-001: Documentation Format](architecture/decisions/adr-001.md)
# Guide
- [Getting Started](guide/getting-started.md)
- [Configuration](guide/configuration.md)
- [Troubleshooting](guide/troubleshooting.md)
# Verification
- [Traceability Matrix](verification/traceability-matrix.md)
- [Test Reports](verification/test-reports.md)
---
# Annexes
- [Color Palettes](annexes/color-palettes.md)
- [Coding Standards](annexes/coding-standards.md)
---
[Glossary](GLOSSARY.md) [Contributing](CONTRIBUTING.md)
[Changelog](CHANGELOG.md) [License](LICENSE.md)
The first H1 is the book title. Subsequent H1s are part headings. Front matter
(unnested links before the first ---) introduces the book. Back matter
(unnested links after the last ---) is reference and administrative content.
Both render without numbering.
Rules:
- The first H1 is the book title.
- Summary documents are exempt from the single-H1 rule — additional H1s are part headings.
---separates front matter, body, and back matter.- Every file referenced must exist.
- Empty links (
- [Title]()) are not allowed. - The summary is committed and human-authored — tooling may validate it but does not generate it.
4.3 Glossary
The glossary is a GLOSSARY.md file (or any file with a markspec:glossary
directive). It uses heading-based structure.
Example 26 — glossary:
# Glossary
## A
### ASIL
Automotive Safety Integrity Level. Risk classification defined by [ISO 26262]
ranging from QM (quality managed, no safety relevance) to D (highest
criticality). The level is determined by the [HARA] process.
### ASPICE
Automotive SPICE. A process assessment model for the automotive industry derived
from [ISO/IEC 15504].
## H
### HARA
Hazard Analysis and Risk Assessment. Systematic process defined in [ISO 26262]
Part 3 for identifying hazards and assigning [ASIL] levels.
<!-- Internal references -->
[ASIL]: #asil
[ASPICE]: #aspice
[HARA]: #hara
<!-- External references -->
[ISO 26262]: https://www.iso.org/standard/68383.html
[ISO/IEC 15504]: https://www.iso.org/standard/60555.html
Structure rules:
- H1 for the title.
- H2 for letter groupings.
- H3 for terms — alphabetically sorted within each group.
- Link reference definitions at the end of the file — internal cross-links first, external references second.
Part 5 — Inline References
Inline references resolve content entities across all documents in the project.
They use double braces: {{namespace.id}}.
5.1 Syntax
Example 27 — spec and test references:
This module implements {{spec.SRS_BRK_0107}}. Verified by {{test.SWT_BRK_0107}}.
Example 28 — reference to a standard:
Derived from {{ref.ISO-26262-6}}.
Example 29 — figure, table, and heading references:
See {{fig.system-overview}} and {{tbl.sensor-thresholds}}. Refer to
{{h.requirement-format}} for the full syntax.
On GitHub/GitLab: the braces render as plain text — the ID is human-readable. At build time: resolved to links.
5.2 Namespaces
| Namespace | References | ID source |
|---|---|---|
spec | Spec entries (any TYPE) | Display ID |
test | Test entries | Display ID |
element | Element entries | Display ID |
ref | Reference entries, registry chain | Slug |
fig | Figures | Slug from caption |
tbl | Tables | Slug from caption |
h | Headings | GFM anchor |
Slugs use the GFM algorithm: lowercase, spaces to hyphens, punctuation stripped.
5.3 Rules
- Exactly two braces:
{{and}}. - The first period separates namespace from ID (e.g.,
{{ref.ISO-26262-6}}→ namespaceref, IDISO-26262-6). - No whitespace inside braces.
- No sections (
{{#}}), inverted sections ({{^}}), or partials ({{>}}). - Every reference must resolve at build time.
- References are never committed in resolved form.
- References inside fenced code blocks are not resolved — they render as literal text.
Part 6 — Document Model
6.1 Project properties
Shared across all documents. Every property always resolves. The project.yaml
schema is defined at https://driftsys.github.io/schemas/project/v1.json.
| Property | Source | Fallback |
|---|---|---|
| project | project.yaml → name | Repo directory name |
| repository | project.yaml → repository | git remote get-url origin |
| version | project.yaml → version | git describe |
| license | project.yaml → license | proprietary |
license: defaults to proprietary — absence of a declared license means all
rights reserved. Use SPDX identifiers when specifying a license.
6.2 Document attributes and properties
Per-file metadata splits into two tiers (mirroring the entry model):
- Attributes — authored by the author in YAML front matter.
- Properties — observed by tooling (filename, git history, filesystem).
Document attributes (authored in front matter)
| Attribute | Type | Required | Description |
|---|---|---|---|
document-id | id | no | Document ULID — 01H… 26-char Crockford base32 |
document-type | enum | no | Overrides filename/directive detection (see §6.3) |
labels | tag-list | no | Classification tags (includes DRAFT marker) |
external-id | external-id | no | Cross-system identifier (scheme:value) |
supersedes | id | no | document-id of a document this one replaces |
deprecated | string | no | Retirement reason (non-replacement case) |
references | citation | no | External reference citations with optional locator |
metadata | map | no | Org free-form metadata, never validated |
Document lifecycle mirrors entry lifecycle: DRAFT label for work in progress;
supersedes: for replacement retirement; deprecated: for non-replacement
retirement. There is no separate status: front-matter key. There is no
DEPRECATED / WITHDRAWN label.
All attribute value types follow the 14-type system from §2.6. Profiles may
declare additional keys; projects may allowlist SSG-ecosystem keys in
.markspec.yaml → frontMatter.allowedKeys (see §9.1).
Document properties (observed)
| Property | Source | Fallback |
|---|---|---|
| title | H1 heading | Filename stem |
| revision | Merge-to-main count | 0 |
| authors | project.yaml | Git unique commit authors |
| created | Git first commit timestamp | File system creation time |
| modified | Git last merge commit timestamp | File system modification time |
These are never authored in front matter — title:, author:, date:,
description:, toc:, cover: in front matter are errors (see §8.5 MSL-D001).
The H1, first paragraph, git history, and filesystem are the authoritative
sources.
revision: starts at 0. Increments on each merged PR/MR that modifies the
file. Commits within a branch do not count.
authors: project.yaml recommended — Git history is fragile across moves
and migrations.
draft / retirement: structural signals, not derived. Documents carry DRAFT
via labels: and/or deprecated: via front matter. Replacement is via
supersedes:. No branch-derived lifecycle inference.
6.3 Document types
| Type | Detection | Description |
|---|---|---|
doc | default | Any Markdown file |
glossary | GLOSSARY.md or directive | Heading-based term definitions |
summary | SUMMARY.md or directive | Book table of contents |
references | references.md or directive | Reference-entry collection |
deck | directive only | Slide deck (--- = slide breaks) |
code | file extension | Source files with doc comments |
Profiles may declare additional document types and bind them to per-type
collections (e.g., requirements.md, tests.md mapped to specific
profile-declared Authored types). The core ships only the generic types above.
Heading rules by type:
- doc — one H1, no skipped levels.
- glossary — one H1 (title), H2 (letter groups), H3 (terms).
- summary — first H1 is the book title, additional H1s are part headings. Exempt from single-H1.
- references and other profile-declared collection types — one H1, standard heading rules.
- deck — one H1 (deck title).
---creates slide breaks. H2 headings start each slide. Heading hierarchy is per-slide — H3/H4 within a slide are valid regardless of other slides.
6.4 Content entities
| Entity | Source | ID |
|---|---|---|
| Authored | Authored entry blocks (ULID-valued Id:) | Display ID |
| Reference | Reference entry blocks (URI-valued Id:) | Slug |
| fig | Figure captions, alt text | Slug |
| tbl | Table captions | Slug |
| h | Headings | GFM anchor |
The active profile classifies Authored and Reference entries into specific types
(requirement, test, unit, standard, dependency, …). The core
distinguishes only the two shapes; everything finer is profile-declared.
6.5 References
References (standards, regulations, external specifications) are resolved
through a resolution chain. Projects declare upstream registries in
project.yaml via the references field (project-wide) and per-file via
markspec:references directives. Resolution order: local project → declared
dependencies (in order) → declared references (in order) → per-file
markspec:references directives → RefHub (implicit fallback).
{{ref.ID}} inline references and Derived-from: attribute values are
validated against the resolution chain at build time.
6.6 Rule activation
Entry rules (MSL-R*) activate on any file containing - [DISPLAY_ID] entry
blocks. Traceability rules (MSL-T*) activate on entries carrying an identity
attribute. Glossary rules (MSL-G*) activate only on glossary documents.
Summary rules (MSL-S*) activate only on summary documents.
Part 7 — Formatting Rules
Fixed rules:
| Rule | Value |
|---|---|
| Line endings | lf |
| Emphasis | _text_ (underscores) |
| Strong | **text** (asterisks) |
| List marker | - (dashes) |
| List indent | 2 spaces |
| Code fences | backticks, language required |
| Trailing whitespace | removed |
| Final newline | single \n |
| Table columns | aligned, padded (tables exempt from line width) |
| Horizontal rules | --- |
| Reference definitions | end of file, alphabetical within groups |
Configurable rules (with defaults):
| Rule | Default | Options |
|---|---|---|
| Line width | 80 | any positive integer |
| Prose wrap | always | always, preserve |
MarkSpec normalization
- Attribute blocks — sorted to canonical order, trailing backslashes normalized.
- Reference definitions — moved to end of file, sorted alphabetically within groups.
- Alerts — markers uppercased, spacing normalized.
- Front matter — YAML form; keys sorted to canonical order (core keys, then
profile-declared, then
metadata:, then allowlisted ecosystem keys); forbidden keys removed with an info diagnostic (MSL-D001). See §6.
Part 8 — Lint Rules
8.1 Severity
| Severity | CI behavior |
|---|---|
| error | Fails the build |
| warning | Reported, does not fail |
| notice | Verbose mode only |
8.2 Entry format (MSL-R)
| ID | Severity | Rule |
|---|---|---|
MSL-R001 | error | Entry block: - [DISPLAY_ID] with indented body (Reference-entry body is optional). |
MSL-R002 | error | Display ID is non-empty; matches the active profile’s display-id-pattern: for its inferred type when one applies. |
MSL-R003 | error | Exactly one Id: attribute per entry. |
MSL-R004 | error | Id: value well-formed: bare ULID (^[0-9A-HJKMNP-TV-Z]{26}$) for Authored entries; scheme-qualified URI (RFC 3986) for Reference entries. |
MSL-R005 | error | ULID unique across repository. |
MSL-R006 | error | Display ID unique within project and registry chain. |
MSL-R007 | warning | When a profile declares a display-id-pattern: for the entry’s inferred type, the display ID matches it. |
MSL-R008 | error | Slug-shaped display ID (no scheme, not a ULID) on a Reference entry must match the slug regex. |
MSL-R009 | warning | Sequence number > 0 in patterned display IDs. |
MSL-R010 | warning | Unknown attributes (not in core universal set, not declared by active profile). Generated attributes must not appear in source. |
MSL-R011 | error | No emphasis inside entry blocks. |
MSL-R012 | warning | Canonical attribute order. Auto-fixed. |
MSL-R013 | warning | Sequential numbering expected within a scope. |
MSL-R001 and MSL-R011 apply to all entry blocks. MSL-R002–R010 apply to entries
carrying an Id: attribute.
8.3 Traceability (MSL-T)
| ID | Severity | Rule |
|---|---|---|
MSL-T001 | error | Satisfies: target must resolve to an existing spec entry. |
MSL-T004 | warning | Derived-from: target must resolve to an existing spec entry. |
MSL-T005 | error | References: slug must resolve to an existing reference entry. |
MSL-T006 | error | Allocated-to: target must resolve to an existing element entry. |
MSL-T007 | error | Realizes: target (on elements) must resolve to an existing spec entry. |
MSL-T008 | error | Verifies: target (on tests) must resolve to an existing spec entry. |
MSL-T009 | error | Tests: target (on tests) must resolve to an existing element entry. |
MSL-T010 | error | Part-of: target must resolve to an existing element entry. |
MSL-T011 | error | Depends-on: target must resolve to an existing element entry. |
MSL-T012 | error | Supersedes: target must resolve to an existing same-family entry. |
MSL-T013 | tiered | Link target is non-active: DRAFT label=info; Superseded-by: set or Deprecated: set=warning. |
MSL-T014 is reserved for a future registry-chain check on References:
(warning severity) when reference resolution via upstream registries lands.
Type resolution and per-type attribute validation (spec §1.3, §1.6):
| ID | Severity | Rule |
|---|---|---|
MSL-T020 | error | Type: value is neither a core type nor a profile-declared type. |
MSL-T021 | warning | Core type inferred (display-ID prefix, URI scheme, or discriminating attribute); declare Type: explicitly to silence. |
MSL-T022 | warning | Attribute is core-known but not valid on the entry’s resolved type. |
MSL-T023 | error | Type: value looks like a profile-declared type but no profile is loaded (core-only mode). |
MSL-T024 | warning | Entry carries a type-specific attribute but its core type could not be resolved (no Type:, no profile, no inferable signal). |
Profile-declared enum attributes (e.g., ASIL, Test-level, Element-kind)
are validated against the vocabulary declared in the active profile’s manifest.
The core defines no enum vocabularies of its own.
Direction and level-crossing rules (e.g., “acceptance tests verify stakeholder requirements”) are profile concerns, not core concerns.
8.4 References (MSL-M)
| ID | Severity | Rule |
|---|---|---|
MSL-M001 | error | Every {{namespace.id}} must resolve. |
MSL-M002 | error | Namespace: spec, test, element, ref, fig, tbl, h. |
MSL-M003 | error | No sections, inverted sections, or partials. |
8.5 Document structure (MSL-D)
| ID | Severity | Rule |
|---|---|---|
MSL-D001 | error | Front matter keys must be core, profile-declared, metadata, or allowlisted in .markspec.yaml. Forbidden keys (title, description, toc, authors, author, date, created, modified, cover, images, sections) are errors with auto-fix to remove. |
MSL-D002 | warning | Footnotes must not contain requirement IDs. |
MSL-D003 | notice | Non-standard alert types. |
MSL-D004 | warning | Caption format: _Table:_ above, _Figure:_ below. |
MSL-D005 | warning | SVGs: viewBox required, no fixed width/height. |
MSL-D006 | configurable | Inline links vs reference-style links. Controlled by referenceLinks config: none (no check), warn (prefer reference-style), enforce (require reference-style). |
MSL-D007 | warning | Reference definitions at end of document, alphabetical within groups. Auto-fixed. |
MSL-D008 | error | Image paths must be relative and stay within the document folder. Absolute URLs (https://...), repo-root links (/...), and paths escaping via ../../ are not permitted (ADR-003). |
8.6 Glossary (MSL-G)
| ID | Severity | Rule |
|---|---|---|
MSL-G001 | error | H1 title, H2 letter groups, H3 terms. |
MSL-G002 | warning | Terms sorted within letter groups. |
MSL-G003 | warning | Link references at end of file, alphabetical within groups (internal then external). |
MSL-G004 | error | Cross-links reference existing headings. |
8.7 Summary (MSL-S)
| ID | Severity | Rule |
|---|---|---|
MSL-S001 | error | Every link target must reference an existing file. |
MSL-S002 | error | No empty links (- [Title]()). |
MSL-S003 | error | No duplicate file paths. |
MSL-S004 | warning | Markdown files in the source directory not referenced in summary. |
Part 9 — Configuration
9.1 Schema
.markspec.yaml:
referenceLinks: warn # none | warn | enforce
frontMatter:
allowedKeys:
- layout # Hugo
- permalink # Jekyll
- sidebar_position # Docusaurus
- draft
- aliases
.markspec.toml:
referenceLinks = "warn"
[frontMatter]
allowedKeys = ["layout", "permalink", "sidebar_position", "draft", "aliases"]
| Property | Type | Default | Values |
|---|---|---|---|
referenceLinks | string | "warn" | none, warn, enforce |
frontMatter.allowedKeys | string[] | [] | Top-level front-matter keys to accept beyond core / profile / metadata. Preserved verbatim, not validated. |
All formatting rules are fixed. There is no formatter choice — dprint is the formatter.
9.2 CI
lint:
steps:
- name: Format check
run: markspec format --check
- name: Lint
run: markspec validate
9.3 Editor integration
- Format-on-save: dprint (
dprint.vscode). - Diagnostics: markdownlint (David Anson) for generic rules; MarkSpec LSP
(
markspec-lsp) for MSL rules.
Annex A — Formatter Compatibility
dprint is the MarkSpec formatter. This table maps MarkSpec rules to dprint settings and equivalent Prettier settings for teams migrating from Prettier.
| Behavior | dprint | Prettier (reference) |
|---|---|---|
| Emphasis | emphasisKind: underscores | _text_ (default) |
| Strong | strongKind: asterisks | **text** (default) |
| Lists | unorderedListKind: dashes | - (default) |
| Table alignment | aligned | aligned |
| Trailing whitespace | removed | removed |
| Final newline | ensured | ensured |
| Line width | lineWidth: 80 | printWidth: 80 |
| Prose wrap | textWrap: "always" | proseWrap: "always" |
| Line endings | newLineKind: "lf" | endOfLine: "lf" |
| Indent | (global) 2 | tabWidth: 2 |
Prettier’s proseWrap defaults to "preserve". MarkSpec requires "always".
Annex B — dprint.json
{
"$schema": "https://dprint.dev/schemas/v0.json",
"incremental": true,
"markdown": {
"lineWidth": 80,
"textWrap": "always",
"newLineKind": "lf",
"emphasisKind": "underscores",
"strongKind": "asterisks",
"unorderedListKind": "dashes"
},
"includes": ["**/*.md"],
"excludes": [
"**/node_modules",
"**/target",
"**/dist",
"**/build",
"CHANGELOG.md"
],
"plugins": [
"https://plugins.dprint.dev/markdown-0.20.0.wasm"
]
}
Annex C — .prettierrc (migration reference)
Prettier is not the MarkSpec formatter. This config is provided for teams migrating from Prettier to dprint, to verify equivalent output during transition.
{
"$schema": "https://json.schemastore.org/prettierrc",
"printWidth": 80,
"proseWrap": "always",
"tabWidth": 2,
"useTabs": false,
"endOfLine": "lf",
"embeddedLanguageFormatting": "auto",
"overrides": [
{
"files": ["CHANGELOG.md"],
"options": {
"proseWrap": "preserve"
}
}
]
}
Annex D — .markdownlint.yaml (standalone)
heading-increment: true
heading-style: { style: atx }
no-missing-space-atx: true
no-multiple-space-atx: true
blanks-around-headings: true
heading-start-left: true
no-duplicate-heading: { siblings_only: true }
single-title: true # override to false for SUMMARY.md
no-trailing-punctuation: true
first-line-heading: true
ul-style: { style: dash }
list-indent: true
ul-indent: { indent: 2 }
ol-prefix: { style: ordered }
list-marker-space: true
blanks-around-lists: true
no-trailing-spaces: true
no-hard-tabs: true
no-multiple-blanks: true
single-trailing-newline: true
line-length:
line_length: 80 # from .markspec.yaml lineWidth
tables: false
code_blocks: false
headings: false
blanks-around-fences: true
fenced-code-language: true
code-block-style: { style: fenced }
code-fence-style: { style: backtick }
no-emphasis-as-heading: false
no-space-in-emphasis: true
emphasis-style: { style: underscore }
strong-style: { style: asterisk }
no-bare-urls: true
no-space-in-links: true
no-empty-links: true
no-alt-text: true
link-fragments: true
link-image-reference-definitions: true
link-image-style: true
no-inline-html: true
no-multiple-space-blockquote: true
no-blanks-blockquote: false
hr-style: { style: "---" }
required-headings: false
proper-names: false
Annex E — .markdownlint-dprint.yaml
Formatting rules disabled (→ dprint).
heading-increment: true
heading-style: { style: atx }
no-missing-space-atx: true
no-multiple-space-atx: true
blanks-around-headings: false # → dprint
heading-start-left: true
no-duplicate-heading: { siblings_only: true }
single-title: true # override to false for SUMMARY.md
no-trailing-punctuation: true
first-line-heading: true
ul-style: { style: dash }
list-indent: true
ul-indent: { indent: 2 }
ol-prefix: { style: ordered }
list-marker-space: true
blanks-around-lists: false # → dprint
no-trailing-spaces: true # safety net
no-hard-tabs: true
no-multiple-blanks: false # → dprint
single-trailing-newline: false # → dprint
line-length:
line_length: 80 # from .markspec.yaml lineWidth
tables: false
code_blocks: false
headings: false
blanks-around-fences: false # → dprint
fenced-code-language: true
code-block-style: { style: fenced }
code-fence-style: { style: backtick }
no-emphasis-as-heading: false
no-space-in-emphasis: true
emphasis-style: { style: underscore } # safety net
strong-style: { style: asterisk } # safety net
no-bare-urls: true
no-space-in-links: true
no-empty-links: true
no-alt-text: true
link-fragments: true
link-image-reference-definitions: true
link-image-style: true
no-inline-html: true
no-multiple-space-blockquote: true
no-blanks-blockquote: false
hr-style: { style: "---" }
required-headings: false
proper-names: false
Annex F — In-Code Entries by Language
A doc comment starting with [TYPE_XYZ_NNNN] (with or without a leading -) is
recognized as a MarkSpec entry. The following examples show the same entry in
each supported language.
Rust
#![allow(unused)]
fn main() {
/// [SWT_BRK_0107] Debounce unit test
///
/// Given a 10ms debounce window, a 5ms noise spike
/// must not alter the stable output.
///
/// Id: 01HGW3R9QLP4ABCDEFGHJKMNPQ
/// Verifies: SRS_BRK_0107
/// Tests: braking_core::controller::debounce_input
/// Labels: ASIL-B
#[test]
fn swt_brk_0107_debounce_filters_noise() {
// test implementation
}
}
Kotlin
/**
* [SWT_BRK_0107] Debounce unit test
*
* Given a 10ms debounce window, a 5ms noise spike
* must not alter the stable output.
*
* Id: 01HGW3R9QLP4ABCDEFGHJKMNPQ
* Verifies: SRS_BRK_0107
* Tests: braking_core::controller::debounce_input
* Labels: ASIL-B
*/
@Test
fun `swt_brk_0107 debounce filters noise`() {
// test implementation
}
C++ (Doxygen)
/// [SWT_BRK_0107] Debounce unit test
///
/// Given a 10ms debounce window, a 5ms noise spike
/// must not alter the stable output.
///
/// Id: 01HGW3R9QLP4ABCDEFGHJKMNPQ
/// Verifies: SRS_BRK_0107
/// Tests: braking_core::controller::debounce_input
/// Labels: ASIL-B
auto debounce_input(uint16_t raw) -> uint16_t;
C (Doxygen)
/**
* [SWT_BRK_0107] Debounce unit test
*
* Given a 10ms debounce window, a 5ms noise spike
* must not alter the stable output.
*
* Id: 01HGW3R9QLP4ABCDEFGHJKMNPQ
* Verifies: SRS_BRK_0107
* Tests: braking_core::controller::debounce_input
* Labels: ASIL-B
*/
void debounce_input(uint16_t* raw);
Java (JDK 23+)
/// [SWT_BRK_0107] Debounce unit test
///
/// Given a 10ms debounce window, a 5ms noise spike
/// must not alter the stable output.
///
/// Id: 01HGW3R9QLP4ABCDEFGHJKMNPQ
/// Verifies: SRS_BRK_0107
/// Tests: braking_core::controller::debounce_input
/// Labels: ASIL-B
@Test
void swt_brk_0107_debounce_filters_noise() {
// test implementation
}
Java (legacy Javadoc)
Trailing backslashes are omitted — they render as literal characters in Javadoc. Attributes are on consecutive lines.
/**
* [SWT_BRK_0107] Debounce unit test
*
* Given a 10ms debounce window, a 5ms noise spike
* must not alter the stable output.
*
* Id: 01HGW3R9QLP4ABCDEFGHJKMNPQ
* Verifies: SRS_BRK_0107
* Tests: braking_core::controller::debounce_input
* Labels: ASIL-B
*/
@Test
void swt_brk_0107_debounce_filters_noise() {
// test implementation
}
Language support summary
| Language | Doc syntax | Markdown native? |
|---|---|---|
| Rust | /// | yes |
| Kotlin | /** */ KDoc | yes |
| C++ | /// Doxygen | yes (since 1.8) |
| C | /** */ Doxygen | yes (since 1.8) |
| Java 23+ | /// (JEP 467) | yes |
| Java (legacy) | /** */ Javadoc | no (HTML) |
MarkSpec AST Extensions
Status (2026-04-23): partially updated. The
entryKindfield has been renamed toshapewith valuesidentified | referencedper ADR-009 and ADR-002. Entry-identity uses a singleId:attribute (ULID or URI). Examples below reflect these changes. The AST node interface (MsEntry) has not yet been updated in the actual codebase — the parser produces flatEntryobjects, not mdast extension nodes. This document describes the planned AST layer.
This document specifies the MarkSpec abstract syntax tree extensions. The input is a standard mdast tree produced by a CommonMark parser (remark). The MarkSpec transform walks the tree and promotes recognized patterns into extension nodes. Unrecognized nodes pass through unchanged.
The transform is a single post-processing pass over an already-parsed mdast tree. It does not modify the parser grammar — it pattern-matches on existing node types.
Conventions
- mdast refers to the Markdown Abstract Syntax Tree specification.
- Node types use
camelCasewith anmsprefix (e.g.,msEntry). - Extension nodes replace the original mdast nodes in the tree. The original children are redistributed into the extension node’s fields.
- All position information (
positionfield) is preserved from the original nodes. - Extension nodes are
Parentnodes — they can be walked by any mdast-compatible visitor. Unknown node types are skipped by standard tools but readable by MarkSpec tools.
§1 Entry block — msEntry
Detection
The transform inspects every list node in the tree. For each listItem child,
it applies the following decision procedure:
listItem
│
├─ Parent list is ordered? → Skip.
├─ Parent list is nested (depth > 1)? → Skip.
├─ First child is not a paragraph? → Skip.
│
├─ First inline of paragraph is a link node? → Inline link. Skip.
│ (mdast type: link — e.g., [text](url))
│
├─ First inline of paragraph is a linkReference → Reference link. Skip.
│ with referenceType "full" or "collapsed"?
│ (mdast type: linkReference — e.g., [text][ref] or [text][])
│
├─ First inline of paragraph is a linkReference → Shortcut ref link
│ with referenceType "shortcut", AND a matching resolved by
│ definition node exists in the document? definition. Skip.
│ (e.g., [text] with [text]: url elsewhere)
│
├─ Bracket content matches /^\[[ xX]\]/ → Task list item. Skip.
│ (GFM checkbox)
│
├─ listItem has no children beyond the first → No body. Skip.
│ paragraph? (single paragraph, no continuation)
│
├─ Bracket content matches typed entry pattern → msEntry (identified)
│ /^[A-Z]{2,}_[A-Z]{2,12}_\d{3,4}$/
│
├─ Bracket content matches reference entry pattern → msEntry (referenced)
│ /^[A-Za-z0-9-]+$/ AND document type is
│ "references"
│
└─ Otherwise → Normal listItem.
Skip.
“Has body” means the listItem contains children beyond the opening
paragraph. In mdast terms: listItem.children.length > 1, or the first
paragraph is followed by additional block-level content (paragraphs,
blockquotes, code blocks, etc.) at the list item’s indentation level.
“Depth > 1” means the list node’s parent chain includes another
listItem. Entry blocks must be top-level list items — nested list items are
never promoted.
Link resolution relies on the mdast tree. A CommonMark parser resolves
[text] shortcut references against definition nodes in the same document. If
the parser produced a linkReference node (of any referenceType) whose
identifier matches a definition node, the bracket content is a link — not an
entry candidate.
Node type
interface MsEntry extends mdast.Parent {
type: "msEntry";
shape: "identified" | "referenced";
displayId: string;
title: MsEntryTitle;
body: mdast.BlockContent[];
attributes: MsAttribute[];
}
interface MsEntryTitle extends mdast.Parent {
type: "msEntryTitle";
children: mdast.PhrasingContent[];
}
interface MsAttribute {
key: string;
value: string;
position: mdast.Position;
}
Fields:
| Field | Source |
|---|---|
shape | "identified" if display ID matches typed pattern, else "referenced" |
displayId | Text content inside [...] brackets |
title | Inline content after the closing ] on the first line |
body | All block-level children between title and attribute block |
attributes | Parsed from the trailing indented code block (code mdast node) |
Attribute block extraction
The trailing block of the listItem body is inspected. The attribute block is a
trailing indented code mdast node (4-space indent relative to body indent in
CommonMark) whose content lines each match:
Key: Value
Where Key matches [A-Z][A-Za-z-]* and : is the separator. The block
qualifies as an attribute block only when every content line matches.
If the trailing code node is an attribute block, it is removed from body and
its lines are parsed into MsAttribute entries. Otherwise, attributes is
empty and the code node stays in body (as a regular code block).
Backward compatibility. During the transition, the parser also accepts the
legacy form: a trailing paragraph whose text content is Key: Value lines
joined by hard line breaks (the trailing \ form). When the legacy shape is
recognized, the parser emits MSL-DEPRECATED-ATTR-001.
Examples
Input mdast (simplified):
list (unordered, depth 0)
listItem
paragraph
text "[SRS_BRK_0001] Sensor debouncing"
paragraph
text "The sensor driver shall debounce..."
code (indented)
"Id: 01HGW2Q8MNP3RSTVWXYZABCDEF\nSatisfies: SYS_BRK_0042\nLabels: ASIL-B"
Output:
msEntry (identified)
displayId: "SRS_BRK_0001"
title
text "Sensor debouncing"
body
paragraph
text "The sensor driver shall debounce..."
attributes
{ key: "Id", value: "01HGW2Q8MNP3RSTVWXYZABCDEF" }
{ key: "Satisfies", value: "SYS_BRK_0042" }
{ key: "Labels", value: "ASIL-B" }
Not promoted — inline link:
- [See documentation](https://example.com) for details.
mdast: listItem > paragraph > link. First inline is a link node → skip.
Not promoted — shortcut reference link with definition:
- [CommonMark] is the baseline grammar.
[CommonMark]: https://commonmark.org
mdast: listItem > paragraph > linkReference (shortcut). A definition node
with identifier commonmark exists → skip.
Not promoted — nested list item:
- Parent item
- [SRS_BRK_0002] This is nested
Body text.
Id: 01HGW2R9QNP4ABCDEFGHJKMNPQ
The inner list has depth > 1 (parent chain includes a listItem) → skip.
Not promoted — no body:
- [SRS_BRK_0003] Title only, no indented content
listItem.children.length === 1 (single paragraph) → skip.
§2 Attribute block — msAttributeBlock
Attribute blocks are always extracted as part of msEntry detection (§1). They
do not exist as standalone nodes — they are a structural component of an entry
block.
When an entry block is detected, the trailing code mdast node is tested for
the attribute pattern. If every content line matches Key: Value, it is
consumed into the msEntry.attributes array and removed from the tree.
Otherwise, the code node stays in body as a regular code block.
The trailing position is required: if any block-level content appears after the
candidate code node, it does not qualify as an attribute block.
Legacy shape. During the transition, a trailing paragraph with
Key: Value lines separated by hard line breaks is also recognized. This
triggers MSL-DEPRECATED-ATTR-001. The legacy form will be removed in a future
major release.
§3 Table caption — msTableCaption
Detection
A paragraph node containing a single emphasis child whose text content starts
with Table: and is immediately followed by a table sibling node.
“Immediately followed” means the table is the next sibling in the parent’s
children array — no intervening block nodes.
Node type
interface MsTableCaption extends mdast.Parent {
type: "msTableCaption";
slug: string;
caption: mdast.PhrasingContent[];
table: mdast.Table;
}
Fields:
| Field | Source |
|---|---|
slug | tbl. + GFM anchor of caption text after Table: prefix |
caption | Inline content after stripping Table: prefix |
table | The sibling table node, reparented under this node |
The msTableCaption node replaces both the caption paragraph and the table
node in the parent’s children array.
Example
_Table: Sensor thresholds_
| Sensor | Min | Max |
| -------- | --- | ---- |
| Pressure | 0 | 1023 |
Output:
msTableCaption
slug: "tbl.sensor-thresholds"
caption: [text "Sensor thresholds"]
table: (the pipe table node)
Not promoted:
_This is just italic text._
| Column A | Column B |
| -------- | -------- |
Emphasis text does not start with Table: → both nodes unchanged.
§4 Figure caption — msFigureCaption
Detection
An image node followed by a paragraph containing a single emphasis child
whose text content starts with Figure:. Alternatively, an image node with
non-empty alt text and no explicit caption paragraph.
“Followed by” means the caption paragraph is the next sibling after the image
node — no intervening block nodes.
Node type
interface MsFigureCaption extends mdast.Parent {
type: "msFigureCaption";
slug: string;
caption: mdast.PhrasingContent[];
image: mdast.Image;
}
Fields:
| Field | Source |
|---|---|
slug | fig. + GFM anchor of caption text |
caption | Explicit: inline content after Figure:. Alt: alt text. |
image | The image node, reparented under this node |
Explicit caption takes precedence over alt text.
Example

_Figure: High-level architecture of the braking system_
Output:
msFigureCaption
slug: "fig.high-level-architecture-of-the-braking-system"
caption: [text "High-level architecture of the braking system"]
image: (the image node)
§5 Directive — msDirective
Detection
An html node (HTML comment) whose content contains one or more lines starting
with markspec:.
Node type
interface MsDirective extends mdast.Literal {
type: "msDirective";
directives: MsDirectiveEntry[];
}
interface MsDirectiveEntry {
name: string;
payload: string;
position: mdast.Position;
}
Fields:
| Field | Source |
|---|---|
directives | One entry per markspec: line in the comment |
name | Token after markspec: (e.g., deck, references) |
payload | Remainder of line + continuation lines |
Parsing rules
- Scan the
htmlnode value for lines starting withmarkspec:. - Token after
markspec:is the directive name. - Remainder of line is the start of the payload.
- Subsequent lines not starting with
markspec:are payload continuation. - A new
markspec:line or end of comment (-->) terminates the payload.
Example
<!--
markspec:deck
markspec:references https://safety.company.io/registry
-->
Output:
msDirective
directives:
{ name: "deck", payload: "" }
{ name: "references", payload: "https://safety.company.io/registry" }
Range directives (markspec:columns, markspec:disable, markspec:ignore)
produce a start msDirective and are closed by a separate msDirective node
containing markspec:end NAME. The transform does not pair them into a single
range node — range matching is a validation concern, not a parse concern.
§6 Inline reference — msInlineRef
Detection
A text node containing {{namespace.id}} patterns. The text node is split
into alternating text and msInlineRef nodes.
References inside code and inlineCode nodes are not detected — they render
as literal text.
Node type
interface MsInlineRef extends mdast.Literal {
type: "msInlineRef";
namespace: string;
refId: string;
}
Fields:
| Field | Source |
|---|---|
namespace | Text before the first . inside {{}} |
refId | Text after the first . inside {{}} |
Example
This module implements {{req.SRS_BRK_0107}}.
Output:
paragraph
text "This module implements "
msInlineRef
namespace: "req"
refId: "SRS_BRK_0107"
text "."
Transform order
The transform processes the tree in a single depth-first pass, in this order:
- Directives (§5) — HTML comments →
msDirective. Must run first so thatmarkspec:ignoreranges can suppress subsequent transforms. - Entry blocks (§1) — list items →
msEntry. Depends on linkdefinitionnodes being present (they are never removed). - Table captions (§3) — emphasis + table pairs →
msTableCaption. - Figure captions (§4) — image + emphasis pairs →
msFigureCaption. - Inline references (§6) —
{{...}}in text nodes →msInlineRef.
Steps 3 and 4 are independent and could run in either order. Step 5 runs last
because it operates on text nodes inside any parent — including inside
msEntry body content.
Non-goals
- In-code entries (doc comments in source files) are handled by a separate
source parser (
core/parser/source.ts), not by the mdast transform. The source parser produces the sameMsEntrydata structure but extracts it from tree-sitter ASTs, not mdast. - Validation (MSL rules) is not part of the AST transform. The transform produces the extended tree; the validator inspects it.
- Formatting (ULID stamping, attribute normalization) operates on the extended tree but is a separate pass.