Skip to content

Commit fe83be5

Browse files
authored
Add parallel merge plan (#171)
1 parent 345f9db commit fe83be5

File tree

1 file changed

+100
-0
lines changed

1 file changed

+100
-0
lines changed

openspec-parallel-merge-plan.md

Lines changed: 100 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,100 @@
1+
# OpenSpec Parallel Delta Remediation Plan
2+
3+
## Problem Summary
4+
- Active changes apply requirement-level replacements when archiving. When two changes touch the same requirement, the second archive overwrites the first and silently drops scenarios (e.g., Windsurf vs. Kilo Code slash command updates).
5+
- The archive workflow (`src/core/archive.ts:191` and `src/core/archive.ts:501`) rebuilds main specs by replacing entire requirement blocks with the content contained in the change delta. The delta format (`src/core/parsers/requirement-blocks.ts:113`) has no notion of base versions or scenario-level operations.
6+
- The tooling cannot detect divergence between the change author’s starting point and the live spec, so parallel development corrupts the source of truth without warning.
7+
8+
## Observed Failure Mode
9+
- Change A (`add-windsurf-workflows`) adds a Windsurf scenario under `Slash Command Configuration`.
10+
- Change B (`add-kilocode-workflows`) adds a Kilo Code scenario to the same requirement, starting from the pre-Windsurf spec.
11+
- After Change A archives, the main spec contains both scenarios.
12+
- When Change B archives, `buildUpdatedSpec` sees a `MODIFIED` block for `Slash Command Configuration` and replaces the requirement with the four-scenario variant shipped in that change. Because that file never learned about Windsurf, the Windsurf scenario disappears.
13+
- There is no warning, diff, or conflict indicator—the archive completes successfully, and the source-of-truth spec now omits a shipped scenario.
14+
15+
## Root Causes
16+
1. **Replace-only semantics.** `buildUpdatedSpec` performs hash-map substitution of requirement blocks and cannot merge or compare individual scenarios (`src/core/archive.ts:455`-`src/core/archive.ts:526`).
17+
2. **Missing base fingerprint.** Changes do not persist the requirement content they were authored against, so the archive step cannot tell if the live spec diverged.
18+
3. **Single-level granularity.** The delta language only understands requirements. Even if we introduced scenario-level parsing, we would still lose sibling edits without an accompanying merge strategy.
19+
4. **Lack of conflict UX.** The CLI never forces contributors to reconcile parallel updates. There is no equivalent of `git merge`, `git rebase`, or conflict markers.
20+
21+
## Design Objectives
22+
- Preserve every approved scenario regardless of archive order.
23+
- Detect and block speculative archives when the live spec diverges from the author’s base.
24+
- Provide a deterministic, reviewable conflict resolution flow that mirrors source-control best practices.
25+
- Keep the authoring experience ergonomic: deltas should remain human-editable markdown.
26+
- Support incremental adoption so existing repositories can roll forward without breaking active work.
27+
28+
## Proposed Fix: Layered Remediation
29+
30+
### Phase 0 – Stop the Bleeding (Detection & Guardrails)
31+
1. **Persist requirement fingerprints alongside each change.**
32+
- When scaffolding or validating a change, capture the current requirement body for every `MODIFIED`/`REMOVED`/`RENAMED` entry and write it to `changes/<id>/meta.json`.
33+
- Store a stable hash (e.g., SHA-256) of the base requirement content and the raw text itself for later merges.
34+
2. **Validate fingerprints during archive.**
35+
- Before `buildUpdatedSpec` mutates specs, recompute the requirement hash from the live spec.
36+
- If the hash differs from the stored base, abort and instruct the user to rebase. This makes the destructive path impossible.
37+
3. **Surface intent in CLI output.**
38+
- Show which requirements are stale, when they diverged, and which change last touched them.
39+
4. **Document interim manual mitigation.**
40+
- Update `openspec/AGENTS.md` and docs so contributors know to rerun `openspec change sync` (see Phase 1) whenever another change lands.
41+
42+
_Outcome:_ We prevent data loss immediately while we work on a richer merge story.
43+
44+
### Phase 1 – Add a Rebase Workflow (Author-Side Merge)
45+
1. **Introduce `openspec change sync <id>` (or `rebase`).**
46+
- Reads the stored base snapshot, the current spec, and the author’s delta.
47+
- Performs a 3-way merge per requirement. A naive diff3 on markdown lines is acceptable initially because we already operate on requirement-sized chunks.
48+
- If the merge is clean, rewrite the `MODIFIED` block with the merged text and refresh the stored fingerprint.
49+
- On conflict, write conflict markers inside the change delta (similar to Git) and require the author to hand-edit before re-running validation.
50+
2. **Enrich validator messages.**
51+
- `openspec validate` should flag unresolved conflict markers or fingerprint mismatches so errors appear early in the workflow.
52+
3. **Improve diff tooling.**
53+
- Extend `openspec diff` to compare change deltas against the live spec and highlight pending merges.
54+
4. **Optional:** Offer a `--rewrite-scenarios` helper that merges bullet lists of scenarios to reduce manual editing noise.
55+
56+
_Outcome:_ Contributors can safely reconcile their work with the latest spec before archiving, restoring true parallel development.
57+
58+
### Phase 2 – Increase Delta Granularity
59+
1. **Extend the delta language with scenario-level directives.**
60+
- Allow `## MODIFIED Requirements` + `## ADDED Scenarios` / `## MODIFIED Scenarios` sections nested under the requirement header.
61+
- Backed by stable scenario identifiers (explicit IDs or generated hashes) stored in `meta.json`. This lets the system reason about individual scenarios.
62+
2. **Teach the parser to understand nested operations.**
63+
- Update `parseDeltaSpec` to emit scenario-level operations in addition to requirement blocks.
64+
- Update `buildUpdatedSpec` (or its replacement) to merge scenario lists, preserving order while inserting new entries in a deterministic fashion.
65+
3. **Automate migration.**
66+
- Provide a one-time command that inspects each existing spec, injects scenario IDs, and rewrites in-flight change deltas into the richer format.
67+
4. **Continue to rely on the Phase 1 rebase flow for conflicts when two changes edit the same scenario body or description.**
68+
69+
_Outcome:_ Most concurrent updates become commutative, drastically reducing the odds of human merges.
70+
71+
### Phase 3 – Structured Spec Graph (Long-Term)
72+
1. **Define stable requirement IDs.**
73+
- Embed `Requirement ID: <uuid>` markers in specs so renames and moves are trackable.
74+
- This enables future features like cross-capability references and better diff visualizations.
75+
2. **Model spec edits as operations over an AST.**
76+
- Build an intermediate representation (IR) for requirements/scenarios/metadata.
77+
- Use operational transforms or CRDT-like techniques to guarantee merge associativity.
78+
3. **Integrate with Git directly.**
79+
- Offer optional `openspec branch` scaffolding that aligns spec changes with Git branches, letting teams leverage Git’s conflict editor for the markdown IR.
80+
81+
_Outcome:_ OpenSpec graduates from replace-based updates to a resilient, intent-preserving spec management platform.
82+
83+
## Migration & Product Impacts
84+
- **Backfill metadata:** add hashes for all active changes and the current main specs during the initial rollout.
85+
- **CLI UX:** new commands (`change sync`, enhanced `archive`) require documentation, help text, and release notes.
86+
- **Docs & AGENTS updates:** reinforce the rebase workflow and explain conflict resolution to AI assistants.
87+
- **Testing:** introduce fixtures covering divergent requirement fingerprints and merge resolution logic.
88+
- **Telemetry (optional):** log fingerprint mismatches so we can see how often teams hit conflicts after the rollout.
89+
90+
## Open Questions / Risks
91+
- How should we order scenarios when multiple changes insert at different points? (Consider optional `position` metadata or deterministic alphabetical fallbacks.)
92+
- What is the graceful failure mode if contributors delete the `meta.json` file? (CLI should recreate fingerprints on demand.)
93+
- Do we need to support offline authors who cannot easily re-run the sync command before archiving? (Potential `--accept-outdated` escape hatch for emergencies.)
94+
- How will archived historical changes be handled? We may need a migration script to embed fingerprints retroactively so re-validation succeeds.
95+
96+
## Immediate Next Steps
97+
1. Prototype fingerprint capture during `openspec change validate` and block archive on mismatches.
98+
2. Ship `openspec change sync` with line-based diff3 merging and conflict markers.
99+
3. Update contributor docs and AI instructions to mandate running `sync` before archiving.
100+
4. Plan the scenario-level delta extension and migration path as a follow-up RFC.

0 commit comments

Comments
 (0)