@kata-sh/cli 0.1.0 → 0.1.1
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.
- package/LICENSE +21 -0
- package/README.md +156 -0
- package/dist/app-paths.d.ts +4 -0
- package/dist/app-paths.js +6 -0
- package/dist/cli.d.ts +1 -0
- package/dist/cli.js +56 -0
- package/dist/loader.d.ts +2 -0
- package/dist/loader.js +95 -0
- package/dist/resource-loader.d.ts +18 -0
- package/dist/resource-loader.js +50 -0
- package/dist/wizard.d.ts +15 -0
- package/dist/wizard.js +159 -0
- package/package.json +50 -21
- package/pkg/dist/modes/interactive/theme/dark.json +85 -0
- package/pkg/dist/modes/interactive/theme/light.json +84 -0
- package/pkg/dist/modes/interactive/theme/theme-schema.json +335 -0
- package/pkg/dist/modes/interactive/theme/theme.d.ts +78 -0
- package/pkg/dist/modes/interactive/theme/theme.d.ts.map +1 -0
- package/pkg/dist/modes/interactive/theme/theme.js +949 -0
- package/pkg/dist/modes/interactive/theme/theme.js.map +1 -0
- package/pkg/package.json +8 -0
- package/scripts/postinstall.js +45 -0
- package/src/resources/AGENTS.md +108 -0
- package/src/resources/KATA-WORKFLOW.md +661 -0
- package/src/resources/agents/researcher.md +29 -0
- package/src/resources/agents/scout.md +56 -0
- package/src/resources/agents/worker.md +31 -0
- package/src/resources/extensions/ask-user-questions.ts +200 -0
- package/src/resources/extensions/bg-shell/index.ts +2758 -0
- package/src/resources/extensions/browser-tools/BROWSER-TOOLS-V2-PROPOSAL.md +1277 -0
- package/src/resources/extensions/browser-tools/core.js +1057 -0
- package/src/resources/extensions/browser-tools/index.ts +4916 -0
- package/src/resources/extensions/browser-tools/package.json +20 -0
- package/src/resources/extensions/context7/index.ts +428 -0
- package/src/resources/extensions/context7/package.json +11 -0
- package/src/resources/extensions/get-secrets-from-user.ts +352 -0
- package/src/resources/extensions/github/formatters.ts +207 -0
- package/src/resources/extensions/github/gh-api.ts +537 -0
- package/src/resources/extensions/github/index.ts +778 -0
- package/src/resources/extensions/kata/activity-log.ts +88 -0
- package/src/resources/extensions/kata/auto.ts +2786 -0
- package/src/resources/extensions/kata/commands.ts +355 -0
- package/src/resources/extensions/kata/crash-recovery.ts +85 -0
- package/src/resources/extensions/kata/dashboard-overlay.ts +516 -0
- package/src/resources/extensions/kata/docs/preferences-reference.md +103 -0
- package/src/resources/extensions/kata/doctor.ts +683 -0
- package/src/resources/extensions/kata/files.ts +730 -0
- package/src/resources/extensions/kata/gitignore.ts +165 -0
- package/src/resources/extensions/kata/guided-flow.ts +976 -0
- package/src/resources/extensions/kata/index.ts +556 -0
- package/src/resources/extensions/kata/metrics.ts +397 -0
- package/src/resources/extensions/kata/observability-validator.ts +408 -0
- package/src/resources/extensions/kata/package.json +11 -0
- package/src/resources/extensions/kata/paths.ts +346 -0
- package/src/resources/extensions/kata/preferences.ts +695 -0
- package/src/resources/extensions/kata/prompt-loader.ts +50 -0
- package/src/resources/extensions/kata/prompts/complete-milestone.md +25 -0
- package/src/resources/extensions/kata/prompts/complete-slice.md +27 -0
- package/src/resources/extensions/kata/prompts/discuss.md +151 -0
- package/src/resources/extensions/kata/prompts/doctor-heal.md +29 -0
- package/src/resources/extensions/kata/prompts/execute-task.md +64 -0
- package/src/resources/extensions/kata/prompts/guided-complete-slice.md +1 -0
- package/src/resources/extensions/kata/prompts/guided-discuss-milestone.md +3 -0
- package/src/resources/extensions/kata/prompts/guided-discuss-slice.md +59 -0
- package/src/resources/extensions/kata/prompts/guided-execute-task.md +1 -0
- package/src/resources/extensions/kata/prompts/guided-plan-milestone.md +23 -0
- package/src/resources/extensions/kata/prompts/guided-plan-slice.md +1 -0
- package/src/resources/extensions/kata/prompts/guided-research-slice.md +11 -0
- package/src/resources/extensions/kata/prompts/guided-resume-task.md +1 -0
- package/src/resources/extensions/kata/prompts/plan-milestone.md +47 -0
- package/src/resources/extensions/kata/prompts/plan-slice.md +63 -0
- package/src/resources/extensions/kata/prompts/queue.md +85 -0
- package/src/resources/extensions/kata/prompts/reassess-roadmap.md +48 -0
- package/src/resources/extensions/kata/prompts/replan-slice.md +39 -0
- package/src/resources/extensions/kata/prompts/research-milestone.md +37 -0
- package/src/resources/extensions/kata/prompts/research-slice.md +28 -0
- package/src/resources/extensions/kata/prompts/run-uat.md +109 -0
- package/src/resources/extensions/kata/prompts/system.md +341 -0
- package/src/resources/extensions/kata/session-forensics.ts +550 -0
- package/src/resources/extensions/kata/skill-discovery.ts +137 -0
- package/src/resources/extensions/kata/state.ts +509 -0
- package/src/resources/extensions/kata/templates/context.md +76 -0
- package/src/resources/extensions/kata/templates/decisions.md +8 -0
- package/src/resources/extensions/kata/templates/milestone-summary.md +73 -0
- package/src/resources/extensions/kata/templates/plan.md +133 -0
- package/src/resources/extensions/kata/templates/preferences.md +15 -0
- package/src/resources/extensions/kata/templates/project.md +31 -0
- package/src/resources/extensions/kata/templates/reassessment.md +28 -0
- package/src/resources/extensions/kata/templates/requirements.md +81 -0
- package/src/resources/extensions/kata/templates/research.md +46 -0
- package/src/resources/extensions/kata/templates/roadmap.md +118 -0
- package/src/resources/extensions/kata/templates/slice-context.md +58 -0
- package/src/resources/extensions/kata/templates/slice-summary.md +99 -0
- package/src/resources/extensions/kata/templates/state.md +19 -0
- package/src/resources/extensions/kata/templates/task-plan.md +52 -0
- package/src/resources/extensions/kata/templates/task-summary.md +57 -0
- package/src/resources/extensions/kata/templates/uat.md +54 -0
- package/src/resources/extensions/kata/tests/activity-log-prune.test.ts +327 -0
- package/src/resources/extensions/kata/tests/auto-preflight.test.ts +97 -0
- package/src/resources/extensions/kata/tests/auto-supervisor.test.mjs +53 -0
- package/src/resources/extensions/kata/tests/complete-milestone.test.ts +317 -0
- package/src/resources/extensions/kata/tests/cost-projection.test.ts +160 -0
- package/src/resources/extensions/kata/tests/derive-state-deps.test.ts +477 -0
- package/src/resources/extensions/kata/tests/derive-state.test.ts +1013 -0
- package/src/resources/extensions/kata/tests/doctor.test.ts +718 -0
- package/src/resources/extensions/kata/tests/idle-recovery.test.ts +490 -0
- package/src/resources/extensions/kata/tests/metrics-io.test.ts +254 -0
- package/src/resources/extensions/kata/tests/metrics.test.ts +217 -0
- package/src/resources/extensions/kata/tests/must-have-parser.test.ts +309 -0
- package/src/resources/extensions/kata/tests/parsers.test.ts +1257 -0
- package/src/resources/extensions/kata/tests/plan-milestone.test.ts +185 -0
- package/src/resources/extensions/kata/tests/plan-quality-validator.test.ts +386 -0
- package/src/resources/extensions/kata/tests/reassess-prompt.test.ts +208 -0
- package/src/resources/extensions/kata/tests/replan-slice.test.ts +686 -0
- package/src/resources/extensions/kata/tests/requirements.test.ts +151 -0
- package/src/resources/extensions/kata/tests/resolve-ts-hooks.mjs +17 -0
- package/src/resources/extensions/kata/tests/resolve-ts.mjs +11 -0
- package/src/resources/extensions/kata/tests/run-uat.test.ts +383 -0
- package/src/resources/extensions/kata/tests/unit-runtime.test.ts +388 -0
- package/src/resources/extensions/kata/tests/workspace-index.test.ts +118 -0
- package/src/resources/extensions/kata/tests/worktree.test.ts +222 -0
- package/src/resources/extensions/kata/types.ts +159 -0
- package/src/resources/extensions/kata/unit-runtime.ts +163 -0
- package/src/resources/extensions/kata/workspace-index.ts +203 -0
- package/src/resources/extensions/kata/worktree.ts +182 -0
- package/src/resources/extensions/mac-tools/index.ts +852 -0
- package/src/resources/extensions/mac-tools/swift-cli/Package.swift +22 -0
- package/src/resources/extensions/mac-tools/swift-cli/Sources/main.swift +1318 -0
- package/src/resources/extensions/search-the-web/cache.ts +78 -0
- package/src/resources/extensions/search-the-web/format.ts +258 -0
- package/src/resources/extensions/search-the-web/http.ts +238 -0
- package/src/resources/extensions/search-the-web/index.ts +68 -0
- package/src/resources/extensions/search-the-web/tool-fetch-page.ts +519 -0
- package/src/resources/extensions/search-the-web/tool-llm-context.ts +404 -0
- package/src/resources/extensions/search-the-web/tool-search.ts +503 -0
- package/src/resources/extensions/search-the-web/url-utils.ts +91 -0
- package/src/resources/extensions/shared/confirm-ui.ts +126 -0
- package/src/resources/extensions/shared/interview-ui.ts +822 -0
- package/src/resources/extensions/shared/next-action-ui.ts +235 -0
- package/src/resources/extensions/shared/progress-widget.ts +282 -0
- package/src/resources/extensions/shared/thinking-widget.ts +107 -0
- package/src/resources/extensions/shared/ui.ts +400 -0
- package/src/resources/extensions/shared/wizard-ui.ts +551 -0
- package/src/resources/extensions/slash-commands/audit.ts +92 -0
- package/src/resources/extensions/slash-commands/create-extension.ts +375 -0
- package/src/resources/extensions/slash-commands/create-slash-command.ts +280 -0
- package/src/resources/extensions/slash-commands/index.ts +12 -0
- package/src/resources/extensions/slash-commands/kata-run.ts +34 -0
- package/src/resources/extensions/subagent/agents.ts +126 -0
- package/src/resources/extensions/subagent/index.ts +1293 -0
- package/src/resources/skills/debug-like-expert/SKILL.md +231 -0
- package/src/resources/skills/debug-like-expert/references/debugging-mindset.md +253 -0
- package/src/resources/skills/debug-like-expert/references/hypothesis-testing.md +373 -0
- package/src/resources/skills/debug-like-expert/references/investigation-techniques.md +337 -0
- package/src/resources/skills/debug-like-expert/references/verification-patterns.md +425 -0
- package/src/resources/skills/debug-like-expert/references/when-to-research.md +361 -0
- package/src/resources/skills/frontend-design/SKILL.md +45 -0
- package/src/resources/skills/swiftui/SKILL.md +208 -0
- package/src/resources/skills/swiftui/references/animations.md +921 -0
- package/src/resources/skills/swiftui/references/architecture.md +1561 -0
- package/src/resources/skills/swiftui/references/layout-system.md +1186 -0
- package/src/resources/skills/swiftui/references/navigation.md +1492 -0
- package/src/resources/skills/swiftui/references/networking-async.md +214 -0
- package/src/resources/skills/swiftui/references/performance.md +1706 -0
- package/src/resources/skills/swiftui/references/platform-integration.md +204 -0
- package/src/resources/skills/swiftui/references/state-management.md +1443 -0
- package/src/resources/skills/swiftui/references/swiftdata.md +297 -0
- package/src/resources/skills/swiftui/references/testing-debugging.md +247 -0
- package/src/resources/skills/swiftui/references/uikit-appkit-interop.md +218 -0
- package/src/resources/skills/swiftui/workflows/add-feature.md +191 -0
- package/src/resources/skills/swiftui/workflows/build-new-app.md +311 -0
- package/src/resources/skills/swiftui/workflows/debug-swiftui.md +192 -0
- package/src/resources/skills/swiftui/workflows/optimize-performance.md +197 -0
- package/src/resources/skills/swiftui/workflows/ship-app.md +203 -0
- package/src/resources/skills/swiftui/workflows/write-tests.md +235 -0
- package/dist/commands/task.d.ts +0 -9
- package/dist/commands/task.d.ts.map +0 -1
- package/dist/commands/task.js +0 -129
- package/dist/commands/task.js.map +0 -1
- package/dist/commands/task.test.d.ts +0 -2
- package/dist/commands/task.test.d.ts.map +0 -1
- package/dist/commands/task.test.js +0 -169
- package/dist/commands/task.test.js.map +0 -1
- package/dist/e2e/task-e2e.test.d.ts +0 -2
- package/dist/e2e/task-e2e.test.d.ts.map +0 -1
- package/dist/e2e/task-e2e.test.js +0 -173
- package/dist/e2e/task-e2e.test.js.map +0 -1
- package/dist/index.d.ts +0 -3
- package/dist/index.d.ts.map +0 -1
- package/dist/index.js +0 -93
- package/dist/index.js.map +0 -1
- package/dist/slug.d.ts +0 -2
- package/dist/slug.d.ts.map +0 -1
- package/dist/slug.js +0 -12
- package/dist/slug.js.map +0 -1
- package/dist/slug.test.d.ts +0 -2
- package/dist/slug.test.d.ts.map +0 -1
- package/dist/slug.test.js +0 -32
- package/dist/slug.test.js.map +0 -1
|
@@ -0,0 +1,208 @@
|
|
|
1
|
+
import { readFileSync } from "node:fs";
|
|
2
|
+
import { join, dirname } from "node:path";
|
|
3
|
+
import { fileURLToPath } from "node:url";
|
|
4
|
+
|
|
5
|
+
// loadPrompt reads from ~/.kata/agent/extensions/kata/prompts/ (main checkout).
|
|
6
|
+
// In a worktree the file may not exist there yet, so we resolve prompts
|
|
7
|
+
// relative to this test file's location (the worktree copy).
|
|
8
|
+
const __dirname = dirname(fileURLToPath(import.meta.url));
|
|
9
|
+
const worktreePromptsDir = join(__dirname, "..", "prompts");
|
|
10
|
+
|
|
11
|
+
let passed = 0;
|
|
12
|
+
let failed = 0;
|
|
13
|
+
|
|
14
|
+
function assert(condition: boolean, message: string): void {
|
|
15
|
+
if (condition) {
|
|
16
|
+
passed++;
|
|
17
|
+
} else {
|
|
18
|
+
failed++;
|
|
19
|
+
console.error(` FAIL: ${message}`);
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
|
|
23
|
+
function assertEq<T>(actual: T, expected: T, message: string): void {
|
|
24
|
+
if (JSON.stringify(actual) === JSON.stringify(expected)) {
|
|
25
|
+
passed++;
|
|
26
|
+
} else {
|
|
27
|
+
failed++;
|
|
28
|
+
console.error(
|
|
29
|
+
` FAIL: ${message} — expected ${JSON.stringify(expected)}, got ${JSON.stringify(actual)}`,
|
|
30
|
+
);
|
|
31
|
+
}
|
|
32
|
+
}
|
|
33
|
+
|
|
34
|
+
/**
|
|
35
|
+
* Load a prompt template from the worktree prompts directory
|
|
36
|
+
* and apply variable substitution (mirrors loadPrompt logic).
|
|
37
|
+
*/
|
|
38
|
+
function loadPromptFromWorktree(
|
|
39
|
+
name: string,
|
|
40
|
+
vars: Record<string, string> = {},
|
|
41
|
+
): string {
|
|
42
|
+
const path = join(worktreePromptsDir, `${name}.md`);
|
|
43
|
+
let content = readFileSync(path, "utf-8");
|
|
44
|
+
for (const [key, value] of Object.entries(vars)) {
|
|
45
|
+
content = content.replaceAll(`{{${key}}}`, value);
|
|
46
|
+
}
|
|
47
|
+
return content.trim();
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// ═══════════════════════════════════════════════════════════════════════════
|
|
51
|
+
// Tests
|
|
52
|
+
// ═══════════════════════════════════════════════════════════════════════════
|
|
53
|
+
|
|
54
|
+
async function main(): Promise<void> {
|
|
55
|
+
// ─── reassess-roadmap prompt loads and substitutes ─────────────────────
|
|
56
|
+
console.log("\n=== reassess-roadmap prompt loads and substitutes ===");
|
|
57
|
+
{
|
|
58
|
+
const testVars = {
|
|
59
|
+
milestoneId: "M099",
|
|
60
|
+
completedSliceId: "S03",
|
|
61
|
+
assessmentAbsPath: ".kata/milestones/M099/slices/S03/S03-ASSESSMENT.md",
|
|
62
|
+
roadmapPath: ".kata/milestones/M099/M099-ROADMAP.md",
|
|
63
|
+
inlinedContext: "--- test inlined context block ---",
|
|
64
|
+
};
|
|
65
|
+
|
|
66
|
+
let result: string;
|
|
67
|
+
let threw = false;
|
|
68
|
+
try {
|
|
69
|
+
result = loadPromptFromWorktree("reassess-roadmap", testVars);
|
|
70
|
+
} catch (err) {
|
|
71
|
+
threw = true;
|
|
72
|
+
result = "";
|
|
73
|
+
console.error(` ERROR: loadPrompt threw: ${err}`);
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
assert(!threw, "loadPrompt does not throw for reassess-roadmap");
|
|
77
|
+
assert(
|
|
78
|
+
typeof result === "string" && result.length > 0,
|
|
79
|
+
"loadPrompt returns a non-empty string",
|
|
80
|
+
);
|
|
81
|
+
|
|
82
|
+
// Verify all test variables were substituted into the output
|
|
83
|
+
assert(result.includes("M099"), "prompt contains milestoneId 'M099'");
|
|
84
|
+
assert(result.includes("S03"), "prompt contains completedSliceId 'S03'");
|
|
85
|
+
assert(
|
|
86
|
+
result.includes(".kata/milestones/M099/slices/S03/S03-ASSESSMENT.md"),
|
|
87
|
+
"prompt contains assessmentAbsPath",
|
|
88
|
+
);
|
|
89
|
+
assert(
|
|
90
|
+
result.includes(".kata/milestones/M099/M099-ROADMAP.md"),
|
|
91
|
+
"prompt contains roadmapPath",
|
|
92
|
+
);
|
|
93
|
+
assert(
|
|
94
|
+
result.includes("--- test inlined context block ---"),
|
|
95
|
+
"prompt contains inlinedContext",
|
|
96
|
+
);
|
|
97
|
+
|
|
98
|
+
// Verify no un-substituted variables remain
|
|
99
|
+
assert(
|
|
100
|
+
!result.includes("{{milestoneId}}"),
|
|
101
|
+
"no un-substituted {{milestoneId}}",
|
|
102
|
+
);
|
|
103
|
+
assert(
|
|
104
|
+
!result.includes("{{completedSliceId}}"),
|
|
105
|
+
"no un-substituted {{completedSliceId}}",
|
|
106
|
+
);
|
|
107
|
+
assert(
|
|
108
|
+
!result.includes("{{assessmentAbsPath}}"),
|
|
109
|
+
"no un-substituted {{assessmentAbsPath}}",
|
|
110
|
+
);
|
|
111
|
+
assert(
|
|
112
|
+
!result.includes("{{roadmapPath}}"),
|
|
113
|
+
"no un-substituted {{roadmapPath}}",
|
|
114
|
+
);
|
|
115
|
+
assert(
|
|
116
|
+
!result.includes("{{inlinedContext}}"),
|
|
117
|
+
"no un-substituted {{inlinedContext}}",
|
|
118
|
+
);
|
|
119
|
+
}
|
|
120
|
+
|
|
121
|
+
// ─── reassess-roadmap contains coverage-check instruction ─────────────
|
|
122
|
+
console.log("\n=== reassess-roadmap contains coverage-check instruction ===");
|
|
123
|
+
{
|
|
124
|
+
const prompt = loadPromptFromWorktree("reassess-roadmap", {
|
|
125
|
+
milestoneId: "M001",
|
|
126
|
+
completedSliceId: "S01",
|
|
127
|
+
assessmentAbsPath: ".kata/milestones/M001/slices/S01/S01-ASSESSMENT.md",
|
|
128
|
+
roadmapPath: ".kata/milestones/M001/M001-ROADMAP.md",
|
|
129
|
+
inlinedContext: "context",
|
|
130
|
+
});
|
|
131
|
+
|
|
132
|
+
// Normalize to lowercase for case-insensitive matching
|
|
133
|
+
const lower = prompt.toLowerCase();
|
|
134
|
+
|
|
135
|
+
// The prompt must mention "each success criterion" or "every success criterion"
|
|
136
|
+
assert(
|
|
137
|
+
lower.includes("each success criterion") ||
|
|
138
|
+
lower.includes("every success criterion"),
|
|
139
|
+
"prompt contains 'each success criterion' or 'every success criterion'",
|
|
140
|
+
);
|
|
141
|
+
|
|
142
|
+
// The prompt must mention "owning slice" or "remaining slice"
|
|
143
|
+
assert(
|
|
144
|
+
lower.includes("owning slice") || lower.includes("remaining slice"),
|
|
145
|
+
"prompt contains 'owning slice' or 'remaining slice'",
|
|
146
|
+
);
|
|
147
|
+
|
|
148
|
+
// The prompt must mention "no remaining owner" or "no owner" or "no slice"
|
|
149
|
+
assert(
|
|
150
|
+
lower.includes("no remaining owner") ||
|
|
151
|
+
lower.includes("no owner") ||
|
|
152
|
+
lower.includes("no slice"),
|
|
153
|
+
"prompt contains 'no remaining owner', 'no owner', or 'no slice'",
|
|
154
|
+
);
|
|
155
|
+
|
|
156
|
+
// The prompt must mention "blocking issue" or "blocking"
|
|
157
|
+
assert(
|
|
158
|
+
lower.includes("blocking issue") || lower.includes("blocking"),
|
|
159
|
+
"prompt contains 'blocking issue' or 'blocking'",
|
|
160
|
+
);
|
|
161
|
+
}
|
|
162
|
+
|
|
163
|
+
// ─── coverage-check requires at-least-one semantics ───────────────────
|
|
164
|
+
console.log("\n=== coverage-check requires at-least-one semantics ===");
|
|
165
|
+
{
|
|
166
|
+
const prompt = loadPromptFromWorktree("reassess-roadmap", {
|
|
167
|
+
milestoneId: "M001",
|
|
168
|
+
completedSliceId: "S01",
|
|
169
|
+
assessmentAbsPath: ".kata/milestones/M001/slices/S01/S01-ASSESSMENT.md",
|
|
170
|
+
roadmapPath: ".kata/milestones/M001/M001-ROADMAP.md",
|
|
171
|
+
inlinedContext: "context",
|
|
172
|
+
});
|
|
173
|
+
|
|
174
|
+
const lower = prompt.toLowerCase();
|
|
175
|
+
|
|
176
|
+
// The instruction must use "at least one" or equivalent inclusive language
|
|
177
|
+
assert(
|
|
178
|
+
lower.includes("at least one") ||
|
|
179
|
+
lower.includes("at-least-one") ||
|
|
180
|
+
lower.includes("one or more"),
|
|
181
|
+
"prompt uses 'at least one' or equivalent inclusive language for slice ownership",
|
|
182
|
+
);
|
|
183
|
+
|
|
184
|
+
// The instruction must NOT require "exactly one" — that would be too rigid
|
|
185
|
+
assert(
|
|
186
|
+
!lower.includes("exactly one owner") &&
|
|
187
|
+
!lower.includes("exactly one slice"),
|
|
188
|
+
"prompt does NOT use 'exactly one' for slice ownership (would be too rigid)",
|
|
189
|
+
);
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
// ═════════════════════════════════════════════════════════════════════════
|
|
193
|
+
// Results
|
|
194
|
+
// ═════════════════════════════════════════════════════════════════════════
|
|
195
|
+
|
|
196
|
+
console.log(`\n${"=".repeat(40)}`);
|
|
197
|
+
console.log(`Results: ${passed} passed, ${failed} failed`);
|
|
198
|
+
if (failed > 0) {
|
|
199
|
+
process.exit(1);
|
|
200
|
+
} else {
|
|
201
|
+
console.log("All tests passed ✓");
|
|
202
|
+
}
|
|
203
|
+
}
|
|
204
|
+
|
|
205
|
+
main().catch((error) => {
|
|
206
|
+
console.error(error);
|
|
207
|
+
process.exit(1);
|
|
208
|
+
});
|