-
-
Notifications
You must be signed in to change notification settings - Fork 29
feat(samples): add sample-module-setup skill #70
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Merged
Merged
Changes from all commits
Commits
File filter
Filter by extension
Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
There are no files selected for viewing
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,76 @@ | ||
| --- | ||
| name: sample-module-setup | ||
| description: Sets up BMad Samples module in a project. Use when the user requests to 'install samples module', 'configure BMad Samples', or 'setup BMad Samples'. | ||
| --- | ||
|
|
||
| # Module Setup | ||
|
|
||
| ## Overview | ||
|
|
||
| Installs and configures a BMad module into a project. Module identity (name, code, version) comes from `./assets/module.yaml`. Collects user preferences and writes them to three files: | ||
|
|
||
| - **`{project-root}/_bmad/config.yaml`** — shared project config: core settings at root (e.g. `output_folder`, `document_output_language`) plus a section per module with metadata and module-specific values. User-only keys (`user_name`, `communication_language`) are **never** written here. | ||
| - **`{project-root}/_bmad/config.user.yaml`** — personal settings intended to be gitignored: `user_name`, `communication_language`, and any module variable marked `user_setting: true` in `./assets/module.yaml`. These values live exclusively here. | ||
| - **`{project-root}/_bmad/module-help.csv`** — registers module capabilities for the help system. | ||
|
|
||
| Both config scripts use an anti-zombie pattern — existing entries for this module are removed before writing fresh ones, so stale values never persist. | ||
|
|
||
| `{project-root}` is a **literal token** in config values — never substitute it with an actual path. It signals to the consuming LLM that the value is relative to the project root, not the skill root. | ||
|
|
||
| ## On Activation | ||
|
|
||
| 1. Read `./assets/module.yaml` for module metadata and variable definitions (the `code` field is the module identifier) | ||
| 2. Check if `{project-root}/_bmad/config.yaml` exists — if a section matching the module's code is already present, inform the user this is an update | ||
| 3. Check for per-module configuration at `{project-root}/_bmad/sam/config.yaml` and `{project-root}/_bmad/core/config.yaml`. If either file exists: | ||
| - If `{project-root}/_bmad/config.yaml` does **not** yet have a section for this module: this is a **fresh install**. Inform the user that installer config was detected and values will be consolidated into the new format. | ||
| - If `{project-root}/_bmad/config.yaml` **already** has a section for this module: this is a **legacy migration**. Inform the user that legacy per-module config was found alongside existing config, and legacy values will be used as fallback defaults. | ||
| - In both cases, per-module config files and directories will be cleaned up after setup. | ||
|
|
||
| If the user provides arguments (e.g. `accept all defaults`, `--headless`, or inline values like `user name is BMad, I speak Swahili`), map any provided values to config keys, use defaults for the rest, and skip interactive prompting. Still display the full confirmation summary at the end. | ||
|
|
||
| ## Collect Configuration | ||
|
|
||
| Ask the user for values. Show defaults in brackets. Present all values together so the user can respond once with only the values they want to change (e.g. "change language to Swahili, rest are fine"). Never tell the user to "press enter" or "leave blank" — in a chat interface they must type something to respond. | ||
|
|
||
| **Default priority** (highest wins): existing new config values > legacy config values > `./assets/module.yaml` defaults. When legacy configs exist, read them and use matching values as defaults instead of `module.yaml` defaults. Only keys that match the current schema are carried forward — changed or removed keys are ignored. | ||
|
|
||
| **Core config** (only if no core keys exist yet): `user_name` (default: BMad), `communication_language` and `document_output_language` (default: English — ask as a single language question, both keys get the same answer), `output_folder` (default: `{project-root}/_bmad-output`). Of these, `user_name` and `communication_language` are written exclusively to `config.user.yaml`. The rest go to `config.yaml` at root and are shared across all modules. | ||
|
|
||
| **Module config**: Read each variable in `./assets/module.yaml` that has a `prompt` field. Ask using that prompt with its default value (or legacy value if available). | ||
|
|
||
| ## Write Files | ||
|
|
||
| Write a temp JSON file with the collected answers structured as `{"core": {...}, "module": {...}}` (omit `core` if it already exists). Then run both scripts — they can run in parallel since they write to different files: | ||
|
|
||
| ```bash | ||
| python3 ./scripts/merge-config.py --config-path "{project-root}/_bmad/config.yaml" --user-config-path "{project-root}/_bmad/config.user.yaml" --module-yaml ./assets/module.yaml --answers {temp-file} --legacy-dir "{project-root}/_bmad" | ||
| python3 ./scripts/merge-help-csv.py --target "{project-root}/_bmad/module-help.csv" --source ./assets/module-help.csv --legacy-dir "{project-root}/_bmad" --module-code sam | ||
| ``` | ||
|
|
||
| Both scripts output JSON to stdout with results. If either exits non-zero, surface the error and stop. The scripts automatically read legacy config values as fallback defaults, then delete the legacy files after a successful merge. Check `legacy_configs_deleted` and `legacy_csvs_deleted` in the output to confirm cleanup. | ||
|
|
||
| Run `./scripts/merge-config.py --help` or `./scripts/merge-help-csv.py --help` for full usage. | ||
|
|
||
| ## Create Output Directories | ||
|
|
||
| After writing config, create any output directories that were configured. For filesystem operations only (such as creating directories), resolve the `{project-root}` token to the actual project root and create each path-type value from `config.yaml` that does not yet exist — this includes `output_folder` and any module variable whose value starts with `{project-root}/`. The paths stored in the config files must continue to use the literal `{project-root}` token; only the directories on disk should use the resolved paths. Use `mkdir -p` or equivalent to create the full path. | ||
|
|
||
| ## Cleanup Legacy Directories | ||
|
|
||
| After both merge scripts complete successfully, remove the installer's package directories. Skills and agents in these directories are already installed at `.claude/skills/` — the `_bmad/` directory should only contain config files. | ||
|
|
||
| ```bash | ||
| python3 ./scripts/cleanup-legacy.py --bmad-dir "{project-root}/_bmad" --module-code sam --also-remove _config --skills-dir "{project-root}/.claude/skills" | ||
| ``` | ||
|
|
||
| The script verifies that every skill in the legacy directories exists at `.claude/skills/` before removing anything. Directories without skills (like `_config/`) are removed directly. If the script exits non-zero, surface the error and stop. Missing directories (already cleaned by a prior run) are not errors — the script is idempotent. | ||
|
|
||
| Check `directories_removed` and `files_removed_count` in the JSON output for the confirmation step. Run `./scripts/cleanup-legacy.py --help` for full usage. | ||
|
|
||
| ## Confirm | ||
|
|
||
| Use the script JSON output to display what was written — config values set (written to `config.yaml` at root for core, module section for module values), user settings written to `config.user.yaml` (`user_keys` in result), help entries added, fresh install vs update. If legacy files were deleted, mention the migration. If legacy directories were removed, report the count and list (e.g. "Cleaned up 106 installer package files from sam/, core/, \_config/ — skills are installed at .claude/skills/"). Then display the `module_greeting` from `./assets/module.yaml` to the user. | ||
|
|
||
| ## Outcome | ||
|
|
||
| Once the user's `user_name` and `communication_language` are known (from collected input, arguments, or existing config), use them consistently for the remainder of the session: address the user by their configured name and communicate in their configured `communication_language`. |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,16 @@ | ||
| module,skill,display-name,menu-code,description,action,args,phase,after,before,required,output-location,outputs | ||
| BMad Samples,sample-module-setup,Setup Samples Module,SS,"Install or update BMad Samples module config and help entries.",configure,"{-H: headless mode}|{inline values: skip prompts with provided values}",anytime,,,false,{project-root}/_bmad,config.yaml and config.user.yaml | ||
| BMad Samples,bmad-agent-code-coach,Code Coaching,TC,"Personal coding coaching and mentoring — code review, pair programming, and learning paths.",activate,,anytime,,,false,,coaching session | ||
| BMad Samples,bmad-agent-creative-muse,Creative Muse,TM,"Creative brainstorming, problem-solving, and storytelling companion.",activate,,anytime,,,false,,creative session | ||
| BMad Samples,bmad-agent-diagram-reviewer,Diagram Review,DR,"Review architecture diagrams for gaps, ambiguities, and missing connections.",diagram-review,,anytime,,,false,,review findings | ||
| BMad Samples,bmad-agent-dream-weaver,Dream Capture,DL,"Capture and log a dream through guided conversation.",dream-log,,anytime,,,false,,journal entry | ||
| BMad Samples,bmad-agent-dream-weaver,Dream Interpretation,DI,"Analyze a dream for symbolism, meaning, and personal connections.",dream-interpret,,anytime,sam:dream-log,,false,,interpretation | ||
| BMad Samples,bmad-agent-dream-weaver,Recall Training,RT,"Dream recall exercises and progress tracking.",recall-training,,anytime,,,false,,coaching updates | ||
| BMad Samples,bmad-agent-dream-weaver,Lucid Coaching,LC,"Progressive lucid dreaming training and milestone tracking.",lucid-coach,,anytime,sam:recall-training,,false,,coaching updates | ||
| BMad Samples,bmad-agent-dream-weaver,Dream Seeding,DS,"Pre-sleep dream incubation — plant themes and intentions.",dream-seed,,anytime,,,false,,seed log entry | ||
| BMad Samples,bmad-agent-dream-weaver,Pattern Discovery,PD,"Surface recurring themes, symbols, and emotional patterns across dreams.",pattern-discovery,,anytime,sam:dream-log,,false,,pattern analysis | ||
| BMad Samples,bmad-agent-dream-weaver,Dream Query,DQ,"Search dream history by symbol, emotion, date, or keyword.",dream-query,,anytime,sam:dream-log,,false,,search results | ||
| BMad Samples,bmad-agent-dream-weaver,Save Memory,SM,"Save current session context to memory.",save-memory,,anytime,,,false,,memory checkpoint | ||
| BMad Samples,bmad-agent-sentinel,Sentinel Coaching,TS,"Strategic coaching — risk radar, decision stress-testing, and execution accountability.",activate,,anytime,,,false,,coaching session | ||
| BMad Samples,bmad-excalidraw,Create Diagram,XD,"Create Excalidraw diagrams through guided or autonomous workflows.",diagram-generation,"{-H: headless mode}|{--yolo: quick generation}",anytime,,,false,sample_output_folder,excalidraw file | ||
| BMad Samples,bmad-excalidraw,Validate Diagram,XV,"Validate an Excalidraw file's structure and report issues.",validate,,anytime,sam:diagram-generation,,false,,validation report |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,13 @@ | ||
| code: sam | ||
| name: "BMad Samples" | ||
| description: "Demo module showcasing sample BMad agents, workflows, and skills" | ||
| module_version: 1.0.0 | ||
| default_selected: false | ||
| module_greeting: > | ||
| The BMad Samples module is ready! Explore the sample agents and skills to see what's possible with BMad. | ||
| For questions, suggestions and support - check us on Discord at https://discord.gg/gk8jAdXWmj | ||
|
|
||
| sample_output_folder: | ||
| prompt: "Where should sample output (diagrams, reports) be saved?" | ||
| default: "{project-root}/_bmad-output" | ||
| result: "{project-root}/{value}" |
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
| Original file line number | Diff line number | Diff line change |
|---|---|---|
| @@ -0,0 +1,259 @@ | ||
| #!/usr/bin/env python3 | ||
| # /// script | ||
| # requires-python = ">=3.9" | ||
| # dependencies = [] | ||
| # /// | ||
| """Remove legacy module directories from _bmad/ after config migration. | ||
|
|
||
| After merge-config.py and merge-help-csv.py have migrated config data and | ||
| deleted individual legacy files, this script removes the now-redundant | ||
| directory trees. These directories contain skill files that are already | ||
| installed at .claude/skills/ (or equivalent) — only the config files at | ||
| _bmad/ root need to persist. | ||
|
|
||
| When --skills-dir is provided, the script verifies that every skill found | ||
| in the legacy directories exists at the installed location before removing | ||
| anything. Directories without skills (like _config/) are removed directly. | ||
|
|
||
| Exit codes: 0=success (including nothing to remove), 1=validation error, 2=runtime error | ||
| """ | ||
|
|
||
| import argparse | ||
| import json | ||
| import shutil | ||
| import sys | ||
| from pathlib import Path | ||
|
|
||
|
|
||
| def parse_args(): | ||
| parser = argparse.ArgumentParser( | ||
| description="Remove legacy module directories from _bmad/ after config migration." | ||
| ) | ||
| parser.add_argument( | ||
| "--bmad-dir", | ||
| required=True, | ||
| help="Path to the _bmad/ directory", | ||
| ) | ||
| parser.add_argument( | ||
| "--module-code", | ||
| required=True, | ||
| help="Module code being cleaned up (e.g. 'bmb')", | ||
| ) | ||
| parser.add_argument( | ||
| "--also-remove", | ||
| action="append", | ||
| default=[], | ||
| help="Additional directory names under _bmad/ to remove (repeatable)", | ||
| ) | ||
| parser.add_argument( | ||
| "--skills-dir", | ||
| help="Path to .claude/skills/ — enables safety verification that skills " | ||
| "are installed before removing legacy copies", | ||
| ) | ||
| parser.add_argument( | ||
| "--verbose", | ||
| action="store_true", | ||
| help="Print detailed progress to stderr", | ||
| ) | ||
| return parser.parse_args() | ||
|
|
||
|
|
||
| def find_skill_dirs(base_path: str) -> list: | ||
| """Find directories that contain a SKILL.md file. | ||
|
|
||
| Walks the directory tree and returns the leaf directory name for each | ||
| directory containing a SKILL.md. These are considered skill directories. | ||
|
|
||
| Returns: | ||
| List of skill directory names (e.g. ['bmad-agent-builder', 'bmad-builder-setup']) | ||
| """ | ||
| skills = [] | ||
| root = Path(base_path) | ||
| if not root.exists(): | ||
| return skills | ||
| for skill_md in root.rglob("SKILL.md"): | ||
| skills.append(skill_md.parent.name) | ||
| return sorted(set(skills)) | ||
|
|
||
|
|
||
| def verify_skills_installed( | ||
| bmad_dir: str, dirs_to_check: list, skills_dir: str, verbose: bool = False | ||
| ) -> list: | ||
| """Verify that skills in legacy directories exist at the installed location. | ||
|
|
||
| Scans each directory in dirs_to_check for skill folders (containing SKILL.md), | ||
| then checks that a matching directory exists under skills_dir. Directories | ||
| that contain no skills (like _config/) are silently skipped. | ||
|
|
||
| Returns: | ||
| List of verified skill names. | ||
|
|
||
| Raises SystemExit(1) if any skills are missing from skills_dir. | ||
| """ | ||
| all_verified = [] | ||
| missing = [] | ||
|
|
||
| for dirname in dirs_to_check: | ||
| legacy_path = Path(bmad_dir) / dirname | ||
| if not legacy_path.exists(): | ||
| continue | ||
|
|
||
| skill_names = find_skill_dirs(str(legacy_path)) | ||
| if not skill_names: | ||
| if verbose: | ||
| print( | ||
| f"No skills found in {dirname}/ — skipping verification", | ||
| file=sys.stderr, | ||
| ) | ||
| continue | ||
|
|
||
| for skill_name in skill_names: | ||
| installed_path = Path(skills_dir) / skill_name | ||
| if installed_path.is_dir(): | ||
| all_verified.append(skill_name) | ||
| if verbose: | ||
| print( | ||
| f"Verified: {skill_name} exists at {installed_path}", | ||
| file=sys.stderr, | ||
| ) | ||
| else: | ||
| missing.append(skill_name) | ||
| if verbose: | ||
| print( | ||
| f"MISSING: {skill_name} not found at {installed_path}", | ||
| file=sys.stderr, | ||
| ) | ||
|
|
||
| if missing: | ||
| error_result = { | ||
| "status": "error", | ||
| "error": "Skills not found at installed location", | ||
| "missing_skills": missing, | ||
| "skills_dir": str(Path(skills_dir).resolve()), | ||
| } | ||
| print(json.dumps(error_result, indent=2)) | ||
| sys.exit(1) | ||
|
|
||
| return sorted(set(all_verified)) | ||
|
|
||
|
|
||
| def count_files(path: Path) -> int: | ||
| """Count all files recursively in a directory.""" | ||
| count = 0 | ||
| for item in path.rglob("*"): | ||
| if item.is_file(): | ||
| count += 1 | ||
| return count | ||
|
|
||
|
|
||
| def cleanup_directories( | ||
| bmad_dir: str, dirs_to_remove: list, verbose: bool = False | ||
| ) -> tuple: | ||
| """Remove specified directories under bmad_dir. | ||
|
|
||
| Returns: | ||
| (removed, not_found, total_files_removed) tuple | ||
| """ | ||
| removed = [] | ||
| not_found = [] | ||
| total_files = 0 | ||
|
|
||
| for dirname in dirs_to_remove: | ||
| target = Path(bmad_dir) / dirname | ||
| if not target.exists(): | ||
| not_found.append(dirname) | ||
| if verbose: | ||
| print(f"Not found (skipping): {target}", file=sys.stderr) | ||
| continue | ||
|
|
||
| if not target.is_dir(): | ||
| if verbose: | ||
| print(f"Not a directory (skipping): {target}", file=sys.stderr) | ||
| not_found.append(dirname) | ||
| continue | ||
|
|
||
| file_count = count_files(target) | ||
| if verbose: | ||
| print( | ||
| f"Removing {target} ({file_count} files)", | ||
| file=sys.stderr, | ||
| ) | ||
|
|
||
| try: | ||
| shutil.rmtree(target) | ||
| except OSError as e: | ||
| error_result = { | ||
| "status": "error", | ||
| "error": f"Failed to remove {target}: {e}", | ||
| "directories_removed": removed, | ||
| "directories_failed": dirname, | ||
| } | ||
| print(json.dumps(error_result, indent=2)) | ||
| sys.exit(2) | ||
|
|
||
| removed.append(dirname) | ||
| total_files += file_count | ||
|
|
||
| return removed, not_found, total_files | ||
|
|
||
|
|
||
| def main(): | ||
| args = parse_args() | ||
|
|
||
| bmad_dir = args.bmad_dir | ||
| module_code = args.module_code | ||
|
|
||
| # Build the list of directories to remove | ||
| dirs_to_remove = [module_code, "core"] + args.also_remove | ||
| # Deduplicate while preserving order | ||
| seen = set() | ||
| unique_dirs = [] | ||
| for d in dirs_to_remove: | ||
| if d not in seen: | ||
| seen.add(d) | ||
| unique_dirs.append(d) | ||
| dirs_to_remove = unique_dirs | ||
|
|
||
| if args.verbose: | ||
| print(f"Directories to remove: {dirs_to_remove}", file=sys.stderr) | ||
|
|
||
| # Safety check: verify skills are installed before removing | ||
| verified_skills = None | ||
| if args.skills_dir: | ||
| if args.verbose: | ||
| print( | ||
| f"Verifying skills installed at {args.skills_dir}", | ||
| file=sys.stderr, | ||
| ) | ||
| verified_skills = verify_skills_installed( | ||
| bmad_dir, dirs_to_remove, args.skills_dir, args.verbose | ||
| ) | ||
|
|
||
| # Remove directories | ||
| removed, not_found, total_files = cleanup_directories( | ||
| bmad_dir, dirs_to_remove, args.verbose | ||
| ) | ||
|
|
||
| # Build result | ||
| result = { | ||
| "status": "success", | ||
| "bmad_dir": str(Path(bmad_dir).resolve()), | ||
| "directories_removed": removed, | ||
| "directories_not_found": not_found, | ||
| "files_removed_count": total_files, | ||
| } | ||
|
|
||
| if args.skills_dir: | ||
| result["safety_checks"] = { | ||
| "skills_verified": True, | ||
| "skills_dir": str(Path(args.skills_dir).resolve()), | ||
| "verified_skills": verified_skills, | ||
| } | ||
| else: | ||
| result["safety_checks"] = None | ||
|
|
||
| print(json.dumps(result, indent=2)) | ||
|
|
||
|
|
||
| if __name__ == "__main__": | ||
| main() | ||
Oops, something went wrong.
Oops, something went wrong.
Add this suggestion to a batch that can be applied as a single commit.
This suggestion is invalid because no changes were made to the code.
Suggestions cannot be applied while the pull request is closed.
Suggestions cannot be applied while viewing a subset of changes.
Only one suggestion per line can be applied in a batch.
Add this suggestion to a batch that can be applied as a single commit.
Applying suggestions on deleted lines is not supported.
You must change the existing code in this line in order to create a valid suggestion.
Outdated suggestions cannot be applied.
This suggestion has been applied or marked resolved.
Suggestions cannot be applied from pending reviews.
Suggestions cannot be applied on multi-line comments.
Suggestions cannot be applied while the pull request is queued to merge.
Suggestion cannot be applied right now. Please check back later.
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
The
--skills-dirsafety check treats “installed” asPath(...).is_dir(), which can pass even if the directory exists but doesn’t actually contain the expected skill contents (e.g., missingSKILL.md). That could allow removing the legacy copy even when the installed copy is incomplete.Severity: low
🤖 Was this useful? React with 👍 or 👎, or 🚀 if it prevented an incident/outage.