@intentsolutionsio/skill-creator 5.0.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (39) hide show
  1. package/.claude-plugin/plugin.json +17 -0
  2. package/README.md +55 -0
  3. package/package.json +38 -0
  4. package/scripts/validate-skill.py +1132 -0
  5. package/skills/agent-creator/SKILL.md +305 -0
  6. package/skills/agent-creator/references/anthropic-agent-spec.md +89 -0
  7. package/skills/skill-creator/SKILL.md +267 -0
  8. package/skills/skill-creator/agents/analyzer.md +279 -0
  9. package/skills/skill-creator/agents/comparator.md +207 -0
  10. package/skills/skill-creator/agents/grader.md +228 -0
  11. package/skills/skill-creator/assets/eval_review.html +146 -0
  12. package/skills/skill-creator/eval-viewer/generate_review.py +471 -0
  13. package/skills/skill-creator/eval-viewer/viewer.html +1325 -0
  14. package/skills/skill-creator/references/advanced-eval-workflow.md +320 -0
  15. package/skills/skill-creator/references/anthropic-comparison.md +93 -0
  16. package/skills/skill-creator/references/ard-template.md +47 -0
  17. package/skills/skill-creator/references/creation-guide.md +305 -0
  18. package/skills/skill-creator/references/errors-template.md +27 -0
  19. package/skills/skill-creator/references/examples-template.md +40 -0
  20. package/skills/skill-creator/references/frontmatter-spec.md +531 -0
  21. package/skills/skill-creator/references/implementation-template.md +42 -0
  22. package/skills/skill-creator/references/output-patterns.md +193 -0
  23. package/skills/skill-creator/references/prd-template.md +55 -0
  24. package/skills/skill-creator/references/schemas.md +430 -0
  25. package/skills/skill-creator/references/source-of-truth.md +658 -0
  26. package/skills/skill-creator/references/validation-rules.md +528 -0
  27. package/skills/skill-creator/references/workflows.md +233 -0
  28. package/skills/skill-creator/scripts/__init__.py +0 -0
  29. package/skills/skill-creator/scripts/aggregate_benchmark.py +401 -0
  30. package/skills/skill-creator/scripts/generate_report.py +326 -0
  31. package/skills/skill-creator/scripts/improve_description.py +247 -0
  32. package/skills/skill-creator/scripts/package_skill.py +136 -0
  33. package/skills/skill-creator/scripts/quick_validate.py +103 -0
  34. package/skills/skill-creator/scripts/run_eval.py +344 -0
  35. package/skills/skill-creator/scripts/run_loop.py +329 -0
  36. package/skills/skill-creator/scripts/utils.py +47 -0
  37. package/skills/skill-creator/scripts/validate-skill.py +87 -0
  38. package/skills/skill-creator/templates/agent-template.md +99 -0
  39. package/skills/skill-creator/templates/skill-template.md +122 -0
@@ -0,0 +1,136 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Skill Packager - Creates a distributable .skill file of a skill folder
4
+
5
+ Usage:
6
+ python utils/package_skill.py <path/to/skill-folder> [output-directory]
7
+
8
+ Example:
9
+ python utils/package_skill.py skills/public/my-skill
10
+ python utils/package_skill.py skills/public/my-skill ./dist
11
+ """
12
+
13
+ import fnmatch
14
+ import sys
15
+ import zipfile
16
+ from pathlib import Path
17
+ from scripts.quick_validate import validate_skill
18
+
19
+ # Patterns to exclude when packaging skills.
20
+ EXCLUDE_DIRS = {"__pycache__", "node_modules"}
21
+ EXCLUDE_GLOBS = {"*.pyc"}
22
+ EXCLUDE_FILES = {".DS_Store"}
23
+ # Directories excluded only at the skill root (not when nested deeper).
24
+ ROOT_EXCLUDE_DIRS = {"evals"}
25
+
26
+
27
+ def should_exclude(rel_path: Path) -> bool:
28
+ """Check if a path should be excluded from packaging."""
29
+ parts = rel_path.parts
30
+ if any(part in EXCLUDE_DIRS for part in parts):
31
+ return True
32
+ # rel_path is relative to skill_path.parent, so parts[0] is the skill
33
+ # folder name and parts[1] (if present) is the first subdir.
34
+ if len(parts) > 1 and parts[1] in ROOT_EXCLUDE_DIRS:
35
+ return True
36
+ name = rel_path.name
37
+ if name in EXCLUDE_FILES:
38
+ return True
39
+ return any(fnmatch.fnmatch(name, pat) for pat in EXCLUDE_GLOBS)
40
+
41
+
42
+ def package_skill(skill_path, output_dir=None):
43
+ """
44
+ Package a skill folder into a .skill file.
45
+
46
+ Args:
47
+ skill_path: Path to the skill folder
48
+ output_dir: Optional output directory for the .skill file (defaults to current directory)
49
+
50
+ Returns:
51
+ Path to the created .skill file, or None if error
52
+ """
53
+ skill_path = Path(skill_path).resolve()
54
+
55
+ # Validate skill folder exists
56
+ if not skill_path.exists():
57
+ print(f"❌ Error: Skill folder not found: {skill_path}")
58
+ return None
59
+
60
+ if not skill_path.is_dir():
61
+ print(f"❌ Error: Path is not a directory: {skill_path}")
62
+ return None
63
+
64
+ # Validate SKILL.md exists
65
+ skill_md = skill_path / "SKILL.md"
66
+ if not skill_md.exists():
67
+ print(f"❌ Error: SKILL.md not found in {skill_path}")
68
+ return None
69
+
70
+ # Run validation before packaging
71
+ print("🔍 Validating skill...")
72
+ valid, message = validate_skill(skill_path)
73
+ if not valid:
74
+ print(f"❌ Validation failed: {message}")
75
+ print(" Please fix the validation errors before packaging.")
76
+ return None
77
+ print(f"✅ {message}\n")
78
+
79
+ # Determine output location
80
+ skill_name = skill_path.name
81
+ if output_dir:
82
+ output_path = Path(output_dir).resolve()
83
+ output_path.mkdir(parents=True, exist_ok=True)
84
+ else:
85
+ output_path = Path.cwd()
86
+
87
+ skill_filename = output_path / f"{skill_name}.skill"
88
+
89
+ # Create the .skill file (zip format)
90
+ try:
91
+ with zipfile.ZipFile(skill_filename, 'w', zipfile.ZIP_DEFLATED) as zipf:
92
+ # Walk through the skill directory, excluding build artifacts
93
+ for file_path in skill_path.rglob('*'):
94
+ if not file_path.is_file():
95
+ continue
96
+ arcname = file_path.relative_to(skill_path.parent)
97
+ if should_exclude(arcname):
98
+ print(f" Skipped: {arcname}")
99
+ continue
100
+ zipf.write(file_path, arcname)
101
+ print(f" Added: {arcname}")
102
+
103
+ print(f"\n✅ Successfully packaged skill to: {skill_filename}")
104
+ return skill_filename
105
+
106
+ except Exception as e:
107
+ print(f"❌ Error creating .skill file: {e}")
108
+ return None
109
+
110
+
111
+ def main():
112
+ if len(sys.argv) < 2:
113
+ print("Usage: python utils/package_skill.py <path/to/skill-folder> [output-directory]")
114
+ print("\nExample:")
115
+ print(" python utils/package_skill.py skills/public/my-skill")
116
+ print(" python utils/package_skill.py skills/public/my-skill ./dist")
117
+ sys.exit(1)
118
+
119
+ skill_path = sys.argv[1]
120
+ output_dir = sys.argv[2] if len(sys.argv) > 2 else None
121
+
122
+ print(f"📦 Packaging skill: {skill_path}")
123
+ if output_dir:
124
+ print(f" Output directory: {output_dir}")
125
+ print()
126
+
127
+ result = package_skill(skill_path, output_dir)
128
+
129
+ if result:
130
+ sys.exit(0)
131
+ else:
132
+ sys.exit(1)
133
+
134
+
135
+ if __name__ == "__main__":
136
+ main()
@@ -0,0 +1,103 @@
1
+ #!/usr/bin/env python3
2
+ """
3
+ Quick validation script for skills - minimal version
4
+ """
5
+
6
+ import sys
7
+ import os
8
+ import re
9
+ import yaml
10
+ from pathlib import Path
11
+
12
+ def validate_skill(skill_path):
13
+ """Basic validation of a skill"""
14
+ skill_path = Path(skill_path)
15
+
16
+ # Check SKILL.md exists
17
+ skill_md = skill_path / 'SKILL.md'
18
+ if not skill_md.exists():
19
+ return False, "SKILL.md not found"
20
+
21
+ # Read and validate frontmatter
22
+ content = skill_md.read_text()
23
+ if not content.startswith('---'):
24
+ return False, "No YAML frontmatter found"
25
+
26
+ # Extract frontmatter
27
+ match = re.match(r'^---\n(.*?)\n---', content, re.DOTALL)
28
+ if not match:
29
+ return False, "Invalid frontmatter format"
30
+
31
+ frontmatter_text = match.group(1)
32
+
33
+ # Parse YAML frontmatter
34
+ try:
35
+ frontmatter = yaml.safe_load(frontmatter_text)
36
+ if not isinstance(frontmatter, dict):
37
+ return False, "Frontmatter must be a YAML dictionary"
38
+ except yaml.YAMLError as e:
39
+ return False, f"Invalid YAML in frontmatter: {e}"
40
+
41
+ # Define allowed properties
42
+ ALLOWED_PROPERTIES = {'name', 'description', 'license', 'allowed-tools', 'metadata', 'compatibility'}
43
+
44
+ # Check for unexpected properties (excluding nested keys under metadata)
45
+ unexpected_keys = set(frontmatter.keys()) - ALLOWED_PROPERTIES
46
+ if unexpected_keys:
47
+ return False, (
48
+ f"Unexpected key(s) in SKILL.md frontmatter: {', '.join(sorted(unexpected_keys))}. "
49
+ f"Allowed properties are: {', '.join(sorted(ALLOWED_PROPERTIES))}"
50
+ )
51
+
52
+ # Check required fields
53
+ if 'name' not in frontmatter:
54
+ return False, "Missing 'name' in frontmatter"
55
+ if 'description' not in frontmatter:
56
+ return False, "Missing 'description' in frontmatter"
57
+
58
+ # Extract name for validation
59
+ name = frontmatter.get('name', '')
60
+ if not isinstance(name, str):
61
+ return False, f"Name must be a string, got {type(name).__name__}"
62
+ name = name.strip()
63
+ if name:
64
+ # Check naming convention (kebab-case: lowercase with hyphens)
65
+ if not re.match(r'^[a-z0-9-]+$', name):
66
+ return False, f"Name '{name}' should be kebab-case (lowercase letters, digits, and hyphens only)"
67
+ if name.startswith('-') or name.endswith('-') or '--' in name:
68
+ return False, f"Name '{name}' cannot start/end with hyphen or contain consecutive hyphens"
69
+ # Check name length (max 64 characters per spec)
70
+ if len(name) > 64:
71
+ return False, f"Name is too long ({len(name)} characters). Maximum is 64 characters."
72
+
73
+ # Extract and validate description
74
+ description = frontmatter.get('description', '')
75
+ if not isinstance(description, str):
76
+ return False, f"Description must be a string, got {type(description).__name__}"
77
+ description = description.strip()
78
+ if description:
79
+ # Check for angle brackets
80
+ if '<' in description or '>' in description:
81
+ return False, "Description cannot contain angle brackets (< or >)"
82
+ # Check description length (max 1024 characters per spec)
83
+ if len(description) > 1024:
84
+ return False, f"Description is too long ({len(description)} characters). Maximum is 1024 characters."
85
+
86
+ # Validate compatibility field if present (optional)
87
+ compatibility = frontmatter.get('compatibility', '')
88
+ if compatibility:
89
+ if not isinstance(compatibility, str):
90
+ return False, f"Compatibility must be a string, got {type(compatibility).__name__}"
91
+ if len(compatibility) > 500:
92
+ return False, f"Compatibility is too long ({len(compatibility)} characters). Maximum is 500 characters."
93
+
94
+ return True, "Skill is valid!"
95
+
96
+ if __name__ == "__main__":
97
+ if len(sys.argv) != 2:
98
+ print("Usage: python quick_validate.py <skill_directory>")
99
+ sys.exit(1)
100
+
101
+ valid, message = validate_skill(sys.argv[1])
102
+ print(message)
103
+ sys.exit(0 if valid else 1)
@@ -0,0 +1,344 @@
1
+ #!/usr/bin/env python3
2
+ """Run trigger evaluation for a skill description.
3
+
4
+ Tests whether a skill's description causes Claude to trigger (read the skill)
5
+ for a set of queries. Outputs results as JSON.
6
+ """
7
+
8
+ import argparse
9
+ import json
10
+ import os
11
+ import select
12
+ import subprocess
13
+ import sys
14
+ import time
15
+ import uuid
16
+ from concurrent.futures import ProcessPoolExecutor, as_completed
17
+ from pathlib import Path
18
+
19
+ from scripts.utils import parse_skill_md
20
+
21
+
22
+ def find_project_root() -> Path:
23
+ """Find the project root by walking up from cwd looking for .claude/.
24
+
25
+ Mimics how Claude Code discovers its project root, so the command file
26
+ we create ends up where claude -p will look for it.
27
+ """
28
+ current = Path.cwd()
29
+ for parent in [current, *current.parents]:
30
+ if (parent / ".claude").is_dir():
31
+ return parent
32
+ return current
33
+
34
+
35
+ def _build_match_patterns(clean_name: str, skill_name: str, skill_path: str | None) -> list[str]:
36
+ """Build the set of strings that indicate the skill was triggered.
37
+
38
+ Matches against:
39
+ - The temp command file name (for skills not yet installed)
40
+ - The real skill name (for already-installed plugin skills)
41
+ - The real SKILL.md path (for Read-based triggering of installed skills)
42
+ """
43
+ patterns = [clean_name, skill_name]
44
+ if skill_path:
45
+ # Match the SKILL.md path for Read tool calls to the real skill
46
+ real_skill_md = str(Path(skill_path) / "SKILL.md")
47
+ patterns.append(real_skill_md)
48
+ # Also match the skill directory path itself
49
+ patterns.append(str(Path(skill_path)))
50
+ return patterns
51
+
52
+
53
+ def _matches_skill(text: str, patterns: list[str]) -> bool:
54
+ """Check if any of the skill match patterns appear in text."""
55
+ return any(p in text for p in patterns)
56
+
57
+
58
+ def run_single_query(
59
+ query: str,
60
+ skill_name: str,
61
+ skill_description: str,
62
+ timeout: int,
63
+ project_root: str,
64
+ model: str | None = None,
65
+ skill_path: str | None = None,
66
+ ) -> bool:
67
+ """Run a single query and return whether the skill was triggered.
68
+
69
+ Creates a command file in .claude/commands/ so it appears in Claude's
70
+ available_skills list, then runs `claude -p` with the raw query.
71
+ Uses --include-partial-messages to detect triggering early from
72
+ stream events (content_block_start) rather than waiting for the
73
+ full assistant message, which only arrives after tool execution.
74
+
75
+ When the skill is already installed as a plugin, Claude may invoke
76
+ the real plugin skill instead of the temp command. The detection
77
+ logic matches against both the temp command name AND the real skill
78
+ name/path, so results are accurate either way.
79
+ """
80
+ unique_id = uuid.uuid4().hex[:8]
81
+ clean_name = f"{skill_name}-skill-{unique_id}"
82
+ project_commands_dir = Path(project_root) / ".claude" / "commands"
83
+ command_file = project_commands_dir / f"{clean_name}.md"
84
+
85
+ # Build match patterns for both temp command and real installed skill
86
+ patterns = _build_match_patterns(clean_name, skill_name, skill_path)
87
+
88
+ try:
89
+ project_commands_dir.mkdir(parents=True, exist_ok=True)
90
+ # Use YAML block scalar to avoid breaking on quotes in description
91
+ indented_desc = "\n ".join(skill_description.split("\n"))
92
+ command_content = (
93
+ f"---\n"
94
+ f"description: |\n"
95
+ f" {indented_desc}\n"
96
+ f"---\n\n"
97
+ f"# {skill_name}\n\n"
98
+ f"This skill handles: {skill_description}\n"
99
+ )
100
+ command_file.write_text(command_content)
101
+
102
+ cmd = [
103
+ "claude",
104
+ "-p", query,
105
+ "--output-format", "stream-json",
106
+ "--verbose",
107
+ "--include-partial-messages",
108
+ ]
109
+ if model:
110
+ cmd.extend(["--model", model])
111
+
112
+ # Remove CLAUDECODE env var to allow nesting claude -p inside a
113
+ # Claude Code session. The guard is for interactive terminal conflicts;
114
+ # programmatic subprocess usage is safe.
115
+ env = {k: v for k, v in os.environ.items() if k != "CLAUDECODE"}
116
+
117
+ process = subprocess.Popen(
118
+ cmd,
119
+ stdout=subprocess.PIPE,
120
+ stderr=subprocess.DEVNULL,
121
+ cwd=project_root,
122
+ env=env,
123
+ )
124
+
125
+ triggered = False
126
+ start_time = time.time()
127
+ buffer = ""
128
+ # Track state for stream event detection
129
+ pending_tool_name = None
130
+ accumulated_json = ""
131
+
132
+ try:
133
+ while time.time() - start_time < timeout:
134
+ if process.poll() is not None:
135
+ remaining = process.stdout.read()
136
+ if remaining:
137
+ buffer += remaining.decode("utf-8", errors="replace")
138
+ break
139
+
140
+ ready, _, _ = select.select([process.stdout], [], [], 1.0)
141
+ if not ready:
142
+ continue
143
+
144
+ chunk = os.read(process.stdout.fileno(), 8192)
145
+ if not chunk:
146
+ break
147
+ buffer += chunk.decode("utf-8", errors="replace")
148
+
149
+ while "\n" in buffer:
150
+ line, buffer = buffer.split("\n", 1)
151
+ line = line.strip()
152
+ if not line:
153
+ continue
154
+
155
+ try:
156
+ event = json.loads(line)
157
+ except json.JSONDecodeError:
158
+ continue
159
+
160
+ # Early detection via stream events
161
+ if event.get("type") == "stream_event":
162
+ se = event.get("event", {})
163
+ se_type = se.get("type", "")
164
+
165
+ if se_type == "content_block_start":
166
+ cb = se.get("content_block", {})
167
+ if cb.get("type") == "tool_use":
168
+ tool_name = cb.get("name", "")
169
+ if tool_name in ("Skill", "Read"):
170
+ pending_tool_name = tool_name
171
+ accumulated_json = ""
172
+ else:
173
+ return False
174
+
175
+ elif se_type == "content_block_delta" and pending_tool_name:
176
+ delta = se.get("delta", {})
177
+ if delta.get("type") == "input_json_delta":
178
+ accumulated_json += delta.get("partial_json", "")
179
+ if _matches_skill(accumulated_json, patterns):
180
+ return True
181
+
182
+ elif se_type in ("content_block_stop", "message_stop"):
183
+ if pending_tool_name:
184
+ return _matches_skill(accumulated_json, patterns)
185
+ if se_type == "message_stop":
186
+ return False
187
+
188
+ # Fallback: full assistant message
189
+ elif event.get("type") == "assistant":
190
+ message = event.get("message", {})
191
+ for content_item in message.get("content", []):
192
+ if content_item.get("type") != "tool_use":
193
+ continue
194
+ tool_name = content_item.get("name", "")
195
+ tool_input = content_item.get("input", {})
196
+ input_str = json.dumps(tool_input)
197
+ if tool_name in ("Skill", "Read") and _matches_skill(input_str, patterns):
198
+ triggered = True
199
+ return triggered
200
+
201
+ elif event.get("type") == "result":
202
+ return triggered
203
+ finally:
204
+ # Clean up process on any exit path (return, exception, timeout)
205
+ if process.poll() is None:
206
+ process.kill()
207
+ process.wait()
208
+
209
+ return triggered
210
+ finally:
211
+ if command_file.exists():
212
+ command_file.unlink()
213
+
214
+
215
+ def run_eval(
216
+ eval_set: list[dict],
217
+ skill_name: str,
218
+ description: str,
219
+ num_workers: int,
220
+ timeout: int,
221
+ project_root: Path,
222
+ runs_per_query: int = 1,
223
+ trigger_threshold: float = 0.5,
224
+ model: str | None = None,
225
+ skill_path: Path | None = None,
226
+ ) -> dict:
227
+ """Run the full eval set and return results."""
228
+ results = []
229
+
230
+ with ProcessPoolExecutor(max_workers=num_workers) as executor:
231
+ future_to_info = {}
232
+ for item in eval_set:
233
+ for run_idx in range(runs_per_query):
234
+ future = executor.submit(
235
+ run_single_query,
236
+ item["query"],
237
+ skill_name,
238
+ description,
239
+ timeout,
240
+ str(project_root),
241
+ model,
242
+ str(skill_path) if skill_path else None,
243
+ )
244
+ future_to_info[future] = (item, run_idx)
245
+
246
+ query_triggers: dict[str, list[bool]] = {}
247
+ query_items: dict[str, dict] = {}
248
+ for future in as_completed(future_to_info):
249
+ item, _ = future_to_info[future]
250
+ query = item["query"]
251
+ query_items[query] = item
252
+ if query not in query_triggers:
253
+ query_triggers[query] = []
254
+ try:
255
+ query_triggers[query].append(future.result())
256
+ except Exception as e:
257
+ print(f"Warning: query failed: {e}", file=sys.stderr)
258
+ query_triggers[query].append(False)
259
+
260
+ for query, triggers in query_triggers.items():
261
+ item = query_items[query]
262
+ trigger_rate = sum(triggers) / len(triggers)
263
+ should_trigger = item["should_trigger"]
264
+ if should_trigger:
265
+ did_pass = trigger_rate >= trigger_threshold
266
+ else:
267
+ did_pass = trigger_rate < trigger_threshold
268
+ results.append({
269
+ "query": query,
270
+ "should_trigger": should_trigger,
271
+ "trigger_rate": trigger_rate,
272
+ "triggers": sum(triggers),
273
+ "runs": len(triggers),
274
+ "pass": did_pass,
275
+ })
276
+
277
+ passed = sum(1 for r in results if r["pass"])
278
+ total = len(results)
279
+
280
+ return {
281
+ "skill_name": skill_name,
282
+ "description": description,
283
+ "results": results,
284
+ "summary": {
285
+ "total": total,
286
+ "passed": passed,
287
+ "failed": total - passed,
288
+ },
289
+ }
290
+
291
+
292
+ def main():
293
+ parser = argparse.ArgumentParser(description="Run trigger evaluation for a skill description")
294
+ parser.add_argument("--eval-set", required=True, help="Path to eval set JSON file")
295
+ parser.add_argument("--skill-path", required=True, help="Path to skill directory")
296
+ parser.add_argument("--description", default=None, help="Override description to test")
297
+ parser.add_argument("--num-workers", type=int, default=10, help="Number of parallel workers")
298
+ parser.add_argument("--timeout", type=int, default=30, help="Timeout per query in seconds")
299
+ parser.add_argument("--runs-per-query", type=int, default=3, help="Number of runs per query")
300
+ parser.add_argument("--trigger-threshold", type=float, default=0.5, help="Trigger rate threshold")
301
+ parser.add_argument("--model", default=None, help="Model to use for claude -p (default: user's configured model)")
302
+ parser.add_argument("--verbose", action="store_true", help="Print progress to stderr")
303
+ args = parser.parse_args()
304
+
305
+ eval_set = json.loads(Path(args.eval_set).read_text())
306
+ skill_path = Path(args.skill_path)
307
+
308
+ if not (skill_path / "SKILL.md").exists():
309
+ print(f"Error: No SKILL.md found at {skill_path}", file=sys.stderr)
310
+ sys.exit(1)
311
+
312
+ name, original_description, content = parse_skill_md(skill_path)
313
+ description = args.description or original_description
314
+ project_root = find_project_root()
315
+
316
+ if args.verbose:
317
+ print(f"Evaluating: {description}", file=sys.stderr)
318
+
319
+ output = run_eval(
320
+ eval_set=eval_set,
321
+ skill_name=name,
322
+ description=description,
323
+ num_workers=args.num_workers,
324
+ timeout=args.timeout,
325
+ project_root=project_root,
326
+ runs_per_query=args.runs_per_query,
327
+ trigger_threshold=args.trigger_threshold,
328
+ model=args.model,
329
+ skill_path=skill_path,
330
+ )
331
+
332
+ if args.verbose:
333
+ summary = output["summary"]
334
+ print(f"Results: {summary['passed']}/{summary['total']} passed", file=sys.stderr)
335
+ for r in output["results"]:
336
+ status = "PASS" if r["pass"] else "FAIL"
337
+ rate_str = f"{r['triggers']}/{r['runs']}"
338
+ print(f" [{status}] rate={rate_str} expected={r['should_trigger']}: {r['query'][:70]}", file=sys.stderr)
339
+
340
+ print(json.dumps(output, indent=2))
341
+
342
+
343
+ if __name__ == "__main__":
344
+ main()