@sk8metal/michi-cli 0.3.0 → 0.5.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 (237) hide show
  1. package/CHANGELOG.md +83 -0
  2. package/dist/scripts/__tests__/spec-impl-workflow.test.js +4 -2
  3. package/dist/scripts/__tests__/spec-impl-workflow.test.js.map +1 -1
  4. package/dist/scripts/config/config-schema.d.ts +52 -0
  5. package/dist/scripts/config/config-schema.d.ts.map +1 -1
  6. package/dist/scripts/config/config-schema.js +25 -0
  7. package/dist/scripts/config/config-schema.js.map +1 -1
  8. package/dist/scripts/config-global.d.ts +10 -0
  9. package/dist/scripts/config-global.d.ts.map +1 -0
  10. package/dist/scripts/config-global.js +111 -0
  11. package/dist/scripts/config-global.js.map +1 -0
  12. package/dist/scripts/confluence-sync.d.ts +22 -4
  13. package/dist/scripts/confluence-sync.d.ts.map +1 -1
  14. package/dist/scripts/confluence-sync.js +22 -12
  15. package/dist/scripts/confluence-sync.js.map +1 -1
  16. package/dist/scripts/jira-sync.d.ts.map +1 -1
  17. package/dist/scripts/jira-sync.js +201 -167
  18. package/dist/scripts/jira-sync.js.map +1 -1
  19. package/dist/scripts/list-projects.js.map +1 -1
  20. package/dist/scripts/multi-project-estimate.js.map +1 -1
  21. package/dist/scripts/phase-runner.d.ts +1 -1
  22. package/dist/scripts/phase-runner.d.ts.map +1 -1
  23. package/dist/scripts/phase-runner.js +295 -522
  24. package/dist/scripts/phase-runner.js.map +1 -1
  25. package/dist/scripts/pr-automation.d.ts.map +1 -1
  26. package/dist/scripts/pr-automation.js +11 -3
  27. package/dist/scripts/pr-automation.js.map +1 -1
  28. package/dist/scripts/pre-flight-check.d.ts.map +1 -1
  29. package/dist/scripts/pre-flight-check.js +10 -6
  30. package/dist/scripts/pre-flight-check.js.map +1 -1
  31. package/dist/scripts/resource-dashboard.js.map +1 -1
  32. package/dist/scripts/spec-impl-workflow.d.ts.map +1 -1
  33. package/dist/scripts/spec-impl-workflow.js +23 -7
  34. package/dist/scripts/spec-impl-workflow.js.map +1 -1
  35. package/dist/scripts/template/renderer.d.ts +1 -1
  36. package/dist/scripts/template/renderer.d.ts.map +1 -1
  37. package/dist/scripts/test-interactive.d.ts.map +1 -1
  38. package/dist/scripts/test-interactive.js +0 -15
  39. package/dist/scripts/test-interactive.js.map +1 -1
  40. package/dist/scripts/test-new-features.js +6 -3
  41. package/dist/scripts/test-new-features.js.map +1 -1
  42. package/dist/scripts/test-spec-generator.d.ts.map +1 -1
  43. package/dist/scripts/test-spec-generator.js +1 -2
  44. package/dist/scripts/test-spec-generator.js.map +1 -1
  45. package/dist/scripts/utils/__tests__/config-loader.test.js +114 -1
  46. package/dist/scripts/utils/__tests__/config-loader.test.js.map +1 -1
  47. package/dist/scripts/utils/__tests__/config-validator.test.js +2 -0
  48. package/dist/scripts/utils/__tests__/config-validator.test.js.map +1 -1
  49. package/dist/scripts/utils/__tests__/env-config.test.js +0 -2
  50. package/dist/scripts/utils/__tests__/env-config.test.js.map +1 -1
  51. package/dist/scripts/utils/__tests__/project-meta.test.d.ts +6 -0
  52. package/dist/scripts/utils/__tests__/project-meta.test.d.ts.map +1 -0
  53. package/dist/scripts/utils/__tests__/project-meta.test.js +154 -0
  54. package/dist/scripts/utils/__tests__/project-meta.test.js.map +1 -0
  55. package/dist/scripts/utils/__tests__/security-validator.test.d.ts +6 -0
  56. package/dist/scripts/utils/__tests__/security-validator.test.d.ts.map +1 -0
  57. package/dist/scripts/utils/__tests__/security-validator.test.js +219 -0
  58. package/dist/scripts/utils/__tests__/security-validator.test.js.map +1 -0
  59. package/dist/scripts/utils/config-loader.d.ts +14 -3
  60. package/dist/scripts/utils/config-loader.d.ts.map +1 -1
  61. package/dist/scripts/utils/config-loader.js +284 -46
  62. package/dist/scripts/utils/config-loader.js.map +1 -1
  63. package/dist/scripts/utils/config-sections.d.ts +54 -0
  64. package/dist/scripts/utils/config-sections.d.ts.map +1 -0
  65. package/dist/scripts/utils/config-sections.js +178 -0
  66. package/dist/scripts/utils/config-sections.js.map +1 -0
  67. package/dist/scripts/utils/config-validator.d.ts +4 -0
  68. package/dist/scripts/utils/config-validator.d.ts.map +1 -1
  69. package/dist/scripts/utils/config-validator.js +57 -1
  70. package/dist/scripts/utils/config-validator.js.map +1 -1
  71. package/dist/scripts/utils/confluence-approval.d.ts.map +1 -1
  72. package/dist/scripts/utils/confluence-approval.js +5 -3
  73. package/dist/scripts/utils/confluence-approval.js.map +1 -1
  74. package/dist/scripts/utils/confluence-hierarchy.d.ts.map +1 -1
  75. package/dist/scripts/utils/confluence-hierarchy.js.map +1 -1
  76. package/dist/scripts/utils/env-config.d.ts +1 -1
  77. package/dist/scripts/utils/env-config.d.ts.map +1 -1
  78. package/dist/scripts/utils/env-config.js +2 -14
  79. package/dist/scripts/utils/env-config.js.map +1 -1
  80. package/dist/scripts/utils/interactive-helpers.d.ts +32 -0
  81. package/dist/scripts/utils/interactive-helpers.d.ts.map +1 -0
  82. package/dist/scripts/utils/interactive-helpers.js +92 -0
  83. package/dist/scripts/utils/interactive-helpers.js.map +1 -0
  84. package/dist/scripts/utils/jira-issue-type-fetcher.d.ts.map +1 -1
  85. package/dist/scripts/utils/jira-issue-type-fetcher.js +27 -18
  86. package/dist/scripts/utils/jira-issue-type-fetcher.js.map +1 -1
  87. package/dist/scripts/utils/project-meta.d.ts +9 -0
  88. package/dist/scripts/utils/project-meta.d.ts.map +1 -1
  89. package/dist/scripts/utils/project-meta.js +22 -0
  90. package/dist/scripts/utils/project-meta.js.map +1 -1
  91. package/dist/scripts/utils/release-notes-generator.d.ts.map +1 -1
  92. package/dist/scripts/utils/release-notes-generator.js +2 -1
  93. package/dist/scripts/utils/release-notes-generator.js.map +1 -1
  94. package/dist/scripts/utils/security-validator.d.ts +55 -0
  95. package/dist/scripts/utils/security-validator.d.ts.map +1 -0
  96. package/dist/scripts/utils/security-validator.js +232 -0
  97. package/dist/scripts/utils/security-validator.js.map +1 -0
  98. package/dist/scripts/utils/spec-updater.d.ts +19 -0
  99. package/dist/scripts/utils/spec-updater.d.ts.map +1 -1
  100. package/dist/scripts/utils/spec-updater.js.map +1 -1
  101. package/dist/scripts/utils/tasks-converter.d.ts.map +1 -1
  102. package/dist/scripts/utils/tasks-converter.js +2 -2
  103. package/dist/scripts/utils/tasks-converter.js.map +1 -1
  104. package/dist/scripts/utils/tasks-format-validator.d.ts.map +1 -1
  105. package/dist/scripts/utils/tasks-format-validator.js +0 -12
  106. package/dist/scripts/utils/tasks-format-validator.js.map +1 -1
  107. package/dist/scripts/utils/test-runner.d.ts.map +1 -1
  108. package/dist/scripts/utils/test-runner.js +3 -2
  109. package/dist/scripts/utils/test-runner.js.map +1 -1
  110. package/dist/scripts/validate-phase.d.ts +1 -1
  111. package/dist/scripts/validate-phase.d.ts.map +1 -1
  112. package/dist/scripts/validate-phase.js +12 -62
  113. package/dist/scripts/validate-phase.js.map +1 -1
  114. package/dist/scripts/workflow-orchestrator.d.ts.map +1 -1
  115. package/dist/scripts/workflow-orchestrator.js +11 -16
  116. package/dist/scripts/workflow-orchestrator.js.map +1 -1
  117. package/dist/src/__tests__/integration/setup/init.test.d.ts +5 -0
  118. package/dist/src/__tests__/integration/setup/init.test.d.ts.map +1 -0
  119. package/dist/src/__tests__/integration/setup/init.test.js +352 -0
  120. package/dist/src/__tests__/integration/setup/init.test.js.map +1 -0
  121. package/dist/src/cli.d.ts.map +1 -1
  122. package/dist/src/cli.js +67 -21
  123. package/dist/src/cli.js.map +1 -1
  124. package/dist/src/commands/__tests__/init.test.d.ts +5 -0
  125. package/dist/src/commands/__tests__/init.test.d.ts.map +1 -0
  126. package/dist/src/commands/__tests__/init.test.js +255 -0
  127. package/dist/src/commands/__tests__/init.test.js.map +1 -0
  128. package/dist/src/commands/__tests__/migrate.test.d.ts +5 -0
  129. package/dist/src/commands/__tests__/migrate.test.d.ts.map +1 -0
  130. package/dist/src/commands/__tests__/migrate.test.js +216 -0
  131. package/dist/src/commands/__tests__/migrate.test.js.map +1 -0
  132. package/dist/src/commands/config-validate.d.ts +9 -0
  133. package/dist/src/commands/config-validate.d.ts.map +1 -0
  134. package/dist/src/commands/config-validate.js +90 -0
  135. package/dist/src/commands/config-validate.js.map +1 -0
  136. package/dist/src/commands/init.d.ts +29 -0
  137. package/dist/src/commands/init.d.ts.map +1 -0
  138. package/dist/src/commands/init.js +513 -0
  139. package/dist/src/commands/init.js.map +1 -0
  140. package/dist/src/commands/migrate.d.ts +25 -0
  141. package/dist/src/commands/migrate.d.ts.map +1 -0
  142. package/dist/src/commands/migrate.js +341 -0
  143. package/dist/src/commands/migrate.js.map +1 -0
  144. package/dist/src/commands/setup-existing.d.ts.map +1 -1
  145. package/dist/src/commands/setup-existing.js +0 -1
  146. package/dist/src/commands/setup-existing.js.map +1 -1
  147. package/dist/vitest.config.d.ts.map +1 -1
  148. package/dist/vitest.config.js +32 -8
  149. package/dist/vitest.config.js.map +1 -1
  150. package/docs/michi-development/design/config-unification.md +4789 -0
  151. package/docs/user-guide/getting-started/github-token-setup.md +2 -1
  152. package/docs/user-guide/getting-started/new-repository-setup.md +1 -1
  153. package/docs/user-guide/getting-started/quick-start.md +1 -1
  154. package/docs/user-guide/getting-started/setup.md +35 -14
  155. package/docs/user-guide/guides/customization.md +64 -11
  156. package/docs/user-guide/guides/workflow.md +35 -21
  157. package/docs/user-guide/hands-on/claude-agent-setup.md +2 -2
  158. package/docs/user-guide/hands-on/claude-setup.md +2 -2
  159. package/docs/user-guide/hands-on/cursor-setup.md +2 -2
  160. package/docs/user-guide/hands-on/workflow-walkthrough.md +4 -1
  161. package/docs/user-guide/reference/config.md +30 -5
  162. package/docs/user-guide/reference/quick-reference.md +68 -74
  163. package/docs/user-guide/testing/test-planning-flow.md +4 -0
  164. package/env.example +1 -1
  165. package/package.json +3 -5
  166. package/scripts/__tests__/spec-impl-workflow.test.ts +5 -2
  167. package/scripts/config/config-schema.ts +40 -0
  168. package/scripts/config-global.ts +160 -0
  169. package/scripts/confluence-sync.ts +91 -27
  170. package/scripts/jira-sync.ts +284 -218
  171. package/scripts/list-projects.ts +2 -2
  172. package/scripts/multi-project-estimate.ts +3 -3
  173. package/scripts/phase-runner.ts +391 -594
  174. package/scripts/pr-automation.ts +15 -5
  175. package/scripts/pre-flight-check.ts +20 -9
  176. package/scripts/pre-publish-check.sh +3 -34
  177. package/scripts/resource-dashboard.ts +4 -4
  178. package/scripts/spec-impl-workflow.ts +23 -7
  179. package/scripts/template/renderer.ts +1 -1
  180. package/scripts/test-interactive.ts +0 -19
  181. package/scripts/test-new-features.ts +10 -7
  182. package/scripts/test-npm-package.sh +3 -34
  183. package/scripts/test-spec-generator.ts +3 -7
  184. package/scripts/utils/__tests__/config-loader.test.ts +149 -0
  185. package/scripts/utils/__tests__/config-validator.test.ts +2 -0
  186. package/scripts/utils/__tests__/env-config.test.ts +0 -2
  187. package/scripts/utils/__tests__/project-meta.test.ts +192 -0
  188. package/scripts/utils/__tests__/security-validator.test.ts +272 -0
  189. package/scripts/utils/config-loader.ts +328 -68
  190. package/scripts/utils/config-sections.ts +316 -0
  191. package/scripts/utils/config-validator.ts +66 -1
  192. package/scripts/utils/confluence-approval.ts +8 -6
  193. package/scripts/utils/confluence-hierarchy.ts +27 -27
  194. package/scripts/utils/env-config.ts +2 -14
  195. package/scripts/utils/interactive-helpers.ts +135 -0
  196. package/scripts/utils/jira-issue-type-fetcher.ts +29 -21
  197. package/scripts/utils/project-meta.ts +27 -0
  198. package/scripts/utils/release-notes-generator.ts +3 -2
  199. package/scripts/utils/security-validator.ts +286 -0
  200. package/scripts/utils/spec-updater.ts +37 -15
  201. package/scripts/utils/tasks-converter.ts +4 -6
  202. package/scripts/utils/tasks-format-validator.ts +0 -13
  203. package/scripts/utils/test-runner.ts +4 -3
  204. package/scripts/validate-phase.ts +21 -80
  205. package/scripts/workflow-orchestrator.ts +16 -25
  206. package/templates/claude/commands/kiro/kiro-spec-impl.md +5 -1
  207. package/templates/claude/commands/kiro/kiro-spec-tasks.md +3 -1
  208. package/templates/claude/commands/michi/confluence-sync.md +8 -2
  209. package/templates/claude/commands/michi/design-review.md +4 -0
  210. package/templates/claude/commands/michi/e2e-plan.md +4 -0
  211. package/templates/claude/commands/michi/license-check.md +4 -0
  212. package/templates/claude/commands/michi/pr-resolve.md +4 -0
  213. package/templates/claude/commands/michi/project-switch.md +8 -2
  214. package/templates/claude/commands/michi/spec-design.md +78 -0
  215. package/templates/claude/commands/michi/spec-impl.md +716 -0
  216. package/templates/claude/commands/michi/test-planning.md +174 -0
  217. package/templates/claude/commands/michi/validate-design.md +58 -0
  218. package/templates/claude/commands/michi/version-audit.md +4 -0
  219. package/templates/claude-agent/commands/kiro/kiro-spec-impl.md +1 -1
  220. package/templates/cursor/commands/kiro/kiro-spec-impl.md +1 -1
  221. package/templates/michi/cc-sdd-overrides/README.md +8 -0
  222. package/templates/michi/cc-sdd-overrides/settings/rules/design-review-michi.md +53 -0
  223. package/dist/scripts/config-interactive.d.ts +0 -10
  224. package/dist/scripts/config-interactive.d.ts.map +0 -1
  225. package/dist/scripts/config-interactive.js +0 -372
  226. package/dist/scripts/config-interactive.js.map +0 -1
  227. package/dist/scripts/setup-existing-project.d.ts +0 -15
  228. package/dist/scripts/setup-existing-project.d.ts.map +0 -1
  229. package/dist/scripts/setup-existing-project.js +0 -455
  230. package/dist/scripts/setup-existing-project.js.map +0 -1
  231. package/dist/scripts/setup-interactive.d.ts +0 -10
  232. package/dist/scripts/setup-interactive.d.ts.map +0 -1
  233. package/dist/scripts/setup-interactive.js +0 -413
  234. package/dist/scripts/setup-interactive.js.map +0 -1
  235. package/scripts/config-interactive.ts +0 -550
  236. package/scripts/setup-existing-project.ts +0 -585
  237. package/scripts/setup-interactive.ts +0 -565
@@ -0,0 +1,192 @@
1
+ /**
2
+ * project-meta.test.ts
3
+ * プロジェクトメタデータユーティリティのテスト
4
+ */
5
+
6
+ import { describe, it, expect, beforeEach, afterEach } from 'vitest';
7
+ import { mkdirSync, writeFileSync, rmSync, existsSync } from 'fs';
8
+ import { join } from 'path';
9
+ import { loadProjectMeta, getRepositoryInfo } from '../project-meta.js';
10
+
11
+ describe('project-meta', () => {
12
+ const testRoot = join(process.cwd(), '.test-tmp-project-meta');
13
+
14
+ beforeEach(() => {
15
+ if (existsSync(testRoot)) {
16
+ rmSync(testRoot, { recursive: true, force: true });
17
+ }
18
+ mkdirSync(testRoot, { recursive: true });
19
+ });
20
+
21
+ afterEach(() => {
22
+ if (existsSync(testRoot)) {
23
+ rmSync(testRoot, { recursive: true, force: true });
24
+ }
25
+ });
26
+
27
+ describe('loadProjectMeta', () => {
28
+ it('正しいproject.jsonを読み込める', () => {
29
+ const kiroDir = join(testRoot, '.kiro');
30
+ mkdirSync(kiroDir, { recursive: true });
31
+
32
+ const projectMeta = {
33
+ projectId: 'test-project',
34
+ projectName: 'Test Project',
35
+ jiraProjectKey: 'TEST',
36
+ confluenceLabels: ['label1', 'label2'],
37
+ status: 'active' as const,
38
+ team: ['member1'],
39
+ stakeholders: ['stakeholder1'],
40
+ repository: 'https://github.com/owner/repo.git',
41
+ };
42
+
43
+ writeFileSync(join(kiroDir, 'project.json'), JSON.stringify(projectMeta, null, 2));
44
+
45
+ const result = loadProjectMeta(testRoot);
46
+
47
+ expect(result).toEqual(projectMeta);
48
+ });
49
+
50
+ it('project.jsonが存在しない場合はエラー', () => {
51
+ expect(() => loadProjectMeta(testRoot)).toThrow('Project metadata not found');
52
+ });
53
+
54
+ it('.kiroディレクトリが存在しない場合はエラー', () => {
55
+ expect(() => loadProjectMeta(testRoot)).toThrow('Project metadata not found');
56
+ });
57
+
58
+ it('不正なJSON形式の場合はエラー', () => {
59
+ const kiroDir = join(testRoot, '.kiro');
60
+ mkdirSync(kiroDir, { recursive: true });
61
+ writeFileSync(join(kiroDir, 'project.json'), 'invalid json');
62
+
63
+ expect(() => loadProjectMeta(testRoot)).toThrow('Invalid JSON');
64
+ });
65
+
66
+ it('必須フィールド projectName が欠けている場合はエラー', () => {
67
+ const kiroDir = join(testRoot, '.kiro');
68
+ mkdirSync(kiroDir, { recursive: true });
69
+
70
+ const invalidMeta = {
71
+ projectId: 'test-project',
72
+ // projectName missing
73
+ jiraProjectKey: 'TEST',
74
+ confluenceLabels: ['label1'],
75
+ status: 'active',
76
+ team: [],
77
+ stakeholders: [],
78
+ repository: 'https://github.com/owner/repo.git',
79
+ };
80
+
81
+ writeFileSync(join(kiroDir, 'project.json'), JSON.stringify(invalidMeta));
82
+
83
+ expect(() => loadProjectMeta(testRoot)).toThrow('Required field missing');
84
+ });
85
+
86
+ it('必須フィールド confluenceLabels が欠けている場合はエラー', () => {
87
+ const kiroDir = join(testRoot, '.kiro');
88
+ mkdirSync(kiroDir, { recursive: true });
89
+
90
+ const invalidMeta = {
91
+ projectId: 'test-project',
92
+ projectName: 'Test Project',
93
+ jiraProjectKey: 'TEST',
94
+ // confluenceLabels missing
95
+ status: 'active',
96
+ team: [],
97
+ stakeholders: [],
98
+ repository: 'https://github.com/owner/repo.git',
99
+ };
100
+
101
+ writeFileSync(join(kiroDir, 'project.json'), JSON.stringify(invalidMeta));
102
+
103
+ expect(() => loadProjectMeta(testRoot)).toThrow('Required field missing');
104
+ });
105
+ });
106
+
107
+ describe('getRepositoryInfo', () => {
108
+ const createValidProjectMeta = (repository: string) => ({
109
+ projectId: 'test-project',
110
+ projectName: 'Test Project',
111
+ jiraProjectKey: 'TEST',
112
+ confluenceLabels: ['label1'],
113
+ status: 'active' as const,
114
+ team: [],
115
+ stakeholders: [],
116
+ repository,
117
+ });
118
+
119
+ beforeEach(() => {
120
+ const kiroDir = join(testRoot, '.kiro');
121
+ mkdirSync(kiroDir, { recursive: true });
122
+ });
123
+
124
+ it('HTTPS URL (.git付き) から owner/repo を抽出できる', () => {
125
+ const projectMeta = createValidProjectMeta('https://github.com/owner/repo.git');
126
+ writeFileSync(join(testRoot, '.kiro/project.json'), JSON.stringify(projectMeta));
127
+
128
+ const result = getRepositoryInfo(testRoot);
129
+ expect(result).toBe('owner/repo');
130
+ });
131
+
132
+ it('HTTPS URL (.gitなし) から owner/repo を抽出できる', () => {
133
+ const projectMeta = createValidProjectMeta('https://github.com/owner/repo');
134
+ writeFileSync(join(testRoot, '.kiro/project.json'), JSON.stringify(projectMeta));
135
+
136
+ const result = getRepositoryInfo(testRoot);
137
+ expect(result).toBe('owner/repo');
138
+ });
139
+
140
+ it('SSH URL (.git付き) から owner/repo を抽出できる', () => {
141
+ const projectMeta = createValidProjectMeta('git@github.com:owner/repo.git');
142
+ writeFileSync(join(testRoot, '.kiro/project.json'), JSON.stringify(projectMeta));
143
+
144
+ const result = getRepositoryInfo(testRoot);
145
+ expect(result).toBe('owner/repo');
146
+ });
147
+
148
+ it('SSH URL (.gitなし) から owner/repo を抽出できる', () => {
149
+ const projectMeta = createValidProjectMeta('git@github.com:owner/repo');
150
+ writeFileSync(join(testRoot, '.kiro/project.json'), JSON.stringify(projectMeta));
151
+
152
+ const result = getRepositoryInfo(testRoot);
153
+ expect(result).toBe('owner/repo');
154
+ });
155
+
156
+ it('repository フィールドが存在しない場合はエラー', () => {
157
+ const projectMeta = {
158
+ projectId: 'test-project',
159
+ projectName: 'Test Project',
160
+ jiraProjectKey: 'TEST',
161
+ confluenceLabels: ['label1'],
162
+ status: 'active' as const,
163
+ team: [],
164
+ stakeholders: [],
165
+ repository: '',
166
+ };
167
+
168
+ writeFileSync(join(testRoot, '.kiro/project.json'), JSON.stringify(projectMeta));
169
+
170
+ expect(() => getRepositoryInfo(testRoot)).toThrow('Repository information not found');
171
+ });
172
+
173
+ it('不正な repository 形式の場合はエラー', () => {
174
+ const projectMeta = createValidProjectMeta('https://example.com/not-github/repo.git');
175
+ writeFileSync(join(testRoot, '.kiro/project.json'), JSON.stringify(projectMeta));
176
+
177
+ expect(() => getRepositoryInfo(testRoot)).toThrow('Invalid GitHub repository format');
178
+ });
179
+
180
+ it('owner/repo にハイフンが含まれる場合も正しく抽出できる', () => {
181
+ const projectMeta = createValidProjectMeta('https://github.com/my-org/my-repo.git');
182
+ writeFileSync(join(testRoot, '.kiro/project.json'), JSON.stringify(projectMeta));
183
+
184
+ const result = getRepositoryInfo(testRoot);
185
+ expect(result).toBe('my-org/my-repo');
186
+ });
187
+
188
+ it('project.jsonが存在しない場合はエラー', () => {
189
+ expect(() => getRepositoryInfo(testRoot)).toThrow('Project metadata not found');
190
+ });
191
+ });
192
+ });
@@ -0,0 +1,272 @@
1
+ /**
2
+ * security-validator.test.ts
3
+ * セキュリティバリデーターのテスト
4
+ */
5
+
6
+ import { describe, it, expect } from 'vitest';
7
+ import {
8
+ validateAtlassianToken,
9
+ validateGitHubToken,
10
+ validateEmail,
11
+ validateUrl,
12
+ validateAtlassianUrl,
13
+ validateGitHubRepositoryUrl,
14
+ validateEnvironmentConfig,
15
+ } from '../security-validator.js';
16
+
17
+ describe('security-validator', () => {
18
+ describe('validateAtlassianToken', () => {
19
+ it('空のトークンはエラー', () => {
20
+ const result = validateAtlassianToken('');
21
+ expect(result.isValid).toBe(false);
22
+ expect(result.errors).toContain('Atlassian API token is empty');
23
+ });
24
+
25
+ it('スペースを含むトークンはエラー', () => {
26
+ const result = validateAtlassianToken('token with spaces');
27
+ expect(result.isValid).toBe(false);
28
+ expect(result.errors).toContain('Atlassian API token contains spaces');
29
+ });
30
+
31
+ it('短すぎるトークンは警告', () => {
32
+ const result = validateAtlassianToken('short');
33
+ expect(result.isValid).toBe(true);
34
+ expect(result.warnings).toContain(
35
+ 'Atlassian API token seems too short (expected >20 characters)',
36
+ );
37
+ });
38
+
39
+ it('プレースホルダー値はエラー', () => {
40
+ const result = validateAtlassianToken('your-token-here');
41
+ expect(result.isValid).toBe(false);
42
+ expect(result.errors).toContain('Atlassian API token is a placeholder value');
43
+ });
44
+
45
+ it('正しいトークンは成功', () => {
46
+ const result = validateAtlassianToken('ATATT3xFfGF0abcdefghijklmnopqrstuvwxyz1234567890');
47
+ expect(result.isValid).toBe(true);
48
+ expect(result.errors).toHaveLength(0);
49
+ expect(result.warnings).toHaveLength(0);
50
+ });
51
+ });
52
+
53
+ describe('validateGitHubToken', () => {
54
+ it('空のトークンはエラー', () => {
55
+ const result = validateGitHubToken('');
56
+ expect(result.isValid).toBe(false);
57
+ expect(result.errors).toContain('GitHub token is empty');
58
+ });
59
+
60
+ it('スペースを含むトークンはエラー', () => {
61
+ const result = validateGitHubToken('token with spaces');
62
+ expect(result.isValid).toBe(false);
63
+ expect(result.errors).toContain('GitHub token contains spaces');
64
+ });
65
+
66
+ it('プレフィックスなしのトークンは警告', () => {
67
+ const result = validateGitHubToken('abcdefghijklmnopqrstuvwxyz1234567890');
68
+ expect(result.isValid).toBe(true);
69
+ expect(result.warnings).toContain(
70
+ 'GitHub token does not start with expected prefix (ghp_ or github_pat_)',
71
+ );
72
+ });
73
+
74
+ it('プレースホルダー値はエラー', () => {
75
+ const result = validateGitHubToken('ghp_xxx');
76
+ expect(result.isValid).toBe(false);
77
+ expect(result.errors).toContain('GitHub token is a placeholder value');
78
+ });
79
+
80
+ it('正しいトークン(ghp_)は成功', () => {
81
+ const result = validateGitHubToken('ghp_abcdefghijklmnopqrstuvwxyz1234567890');
82
+ expect(result.isValid).toBe(true);
83
+ expect(result.errors).toHaveLength(0);
84
+ });
85
+
86
+ it('正しいトークン(github_pat_)は成功', () => {
87
+ const result = validateGitHubToken('github_pat_abcdefghijklmnopqrstuvwxyz1234567890');
88
+ expect(result.isValid).toBe(true);
89
+ expect(result.errors).toHaveLength(0);
90
+ });
91
+ });
92
+
93
+ describe('validateEmail', () => {
94
+ it('空のメールアドレスはエラー', () => {
95
+ const result = validateEmail('');
96
+ expect(result.isValid).toBe(false);
97
+ expect(result.errors).toContain('Email is empty');
98
+ });
99
+
100
+ it('不正な形式のメールアドレスはエラー', () => {
101
+ const result = validateEmail('invalid-email');
102
+ expect(result.isValid).toBe(false);
103
+ expect(result.errors).toContain('Email format is invalid');
104
+ });
105
+
106
+ it('プレースホルダー値はエラー', () => {
107
+ const result = validateEmail('user@example.com');
108
+ expect(result.isValid).toBe(false);
109
+ expect(result.errors).toContain('Email is a placeholder value');
110
+ });
111
+
112
+ it('正しいメールアドレスは成功', () => {
113
+ const result = validateEmail('real.user@company.com');
114
+ expect(result.isValid).toBe(true);
115
+ expect(result.errors).toHaveLength(0);
116
+ });
117
+ });
118
+
119
+ describe('validateUrl', () => {
120
+ it('空のURLはエラー', () => {
121
+ const result = validateUrl('');
122
+ expect(result.isValid).toBe(false);
123
+ expect(result.errors).toContain('URL is empty');
124
+ });
125
+
126
+ it('不正な形式のURLはエラー', () => {
127
+ const result = validateUrl('not-a-url');
128
+ expect(result.isValid).toBe(false);
129
+ expect(result.errors.length).toBeGreaterThan(0);
130
+ });
131
+
132
+ it('httpプロトコルは警告', () => {
133
+ const result = validateUrl('http://example.com');
134
+ expect(result.isValid).toBe(true);
135
+ expect(result.warnings).toContain(
136
+ 'URL uses insecure http:// protocol (consider using https://)',
137
+ );
138
+ });
139
+
140
+ it('プレースホルダー値はエラー', () => {
141
+ const result = validateUrl('https://example.com');
142
+ expect(result.isValid).toBe(false);
143
+ expect(result.errors).toContain('URL is a placeholder value');
144
+ });
145
+
146
+ it('正しいURLは成功', () => {
147
+ const result = validateUrl('https://real-domain.com');
148
+ expect(result.isValid).toBe(true);
149
+ expect(result.errors).toHaveLength(0);
150
+ });
151
+
152
+ it('expectedDomainが一致しない場合はエラー', () => {
153
+ const result = validateUrl('https://wrong.com', 'example.com');
154
+ expect(result.isValid).toBe(false);
155
+ expect(result.errors).toContain(
156
+ 'URL hostname wrong.com does not contain expected domain: example.com',
157
+ );
158
+ });
159
+
160
+ it('expectedDomainが一致する場合は成功', () => {
161
+ const result = validateUrl('https://sub.example.com', 'example.com');
162
+ expect(result.isValid).toBe(true);
163
+ });
164
+ });
165
+
166
+ describe('validateAtlassianUrl', () => {
167
+ it('atlassian.netドメインでない場合はエラー', () => {
168
+ const result = validateAtlassianUrl('https://wrong-domain.com');
169
+ expect(result.isValid).toBe(false);
170
+ expect(result.errors.some((e) => e.includes('atlassian.net'))).toBe(true);
171
+ });
172
+
173
+ it('正しいAtlassian URLは成功', () => {
174
+ const result = validateAtlassianUrl('https://mycompany.atlassian.net');
175
+ expect(result.isValid).toBe(true);
176
+ expect(result.errors).toHaveLength(0);
177
+ });
178
+ });
179
+
180
+ describe('validateGitHubRepositoryUrl', () => {
181
+ it('空のURLはエラー', () => {
182
+ const result = validateGitHubRepositoryUrl('');
183
+ expect(result.isValid).toBe(false);
184
+ expect(result.errors).toContain('Repository URL is empty');
185
+ });
186
+
187
+ it('不正な形式はエラー', () => {
188
+ const result = validateGitHubRepositoryUrl('https://example.com/repo');
189
+ expect(result.isValid).toBe(false);
190
+ expect(result.errors.some((e) => e.includes('format'))).toBe(true);
191
+ });
192
+
193
+ it('プレースホルダー値はエラー', () => {
194
+ const result = validateGitHubRepositoryUrl('https://github.com/your-org/your-repo.git');
195
+ expect(result.isValid).toBe(false);
196
+ expect(result.errors).toContain('Repository URL contains placeholder values');
197
+ });
198
+
199
+ it('正しいHTTPS URLは成功', () => {
200
+ const result = validateGitHubRepositoryUrl('https://github.com/owner/repo.git');
201
+ expect(result.isValid).toBe(true);
202
+ expect(result.errors).toHaveLength(0);
203
+ });
204
+
205
+ it('正しいHTTPS URL(.gitなし)も成功', () => {
206
+ const result = validateGitHubRepositoryUrl('https://github.com/owner/repo');
207
+ expect(result.isValid).toBe(true);
208
+ expect(result.errors).toHaveLength(0);
209
+ });
210
+
211
+ it('正しいSSH URLは成功', () => {
212
+ const result = validateGitHubRepositoryUrl('git@github.com:owner/repo.git');
213
+ expect(result.isValid).toBe(true);
214
+ expect(result.errors).toHaveLength(0);
215
+ });
216
+
217
+ it('正しいSSH URL(.gitなし)も成功', () => {
218
+ const result = validateGitHubRepositoryUrl('git@github.com:owner/repo');
219
+ expect(result.isValid).toBe(true);
220
+ expect(result.errors).toHaveLength(0);
221
+ });
222
+ });
223
+
224
+ describe('validateEnvironmentConfig', () => {
225
+ it('すべて正しい設定は成功', () => {
226
+ const result = validateEnvironmentConfig({
227
+ atlassianUrl: 'https://mycompany.atlassian.net',
228
+ atlassianEmail: 'user@company.com',
229
+ atlassianApiToken: 'ATATT3xFfGF0abcdefghijklmnopqrstuvwxyz1234567890',
230
+ githubOrg: 'myorg',
231
+ githubToken: 'ghp_abcdefghijklmnopqrstuvwxyz1234567890',
232
+ repositoryUrl: 'https://github.com/myorg/myrepo.git',
233
+ });
234
+
235
+ expect(result.isValid).toBe(true);
236
+ expect(result.errors).toHaveLength(0);
237
+ });
238
+
239
+ it('複数のエラーをすべて報告', () => {
240
+ const result = validateEnvironmentConfig({
241
+ atlassianUrl: 'https://example.com', // プレースホルダー
242
+ atlassianEmail: 'invalid-email', // 不正な形式
243
+ atlassianApiToken: 'test-token', // プレースホルダー
244
+ githubOrg: 'your-org', // プレースホルダー
245
+ githubToken: 'ghp_xxx', // プレースホルダー
246
+ repositoryUrl: '', // 空
247
+ });
248
+
249
+ expect(result.isValid).toBe(false);
250
+ expect(result.errors.length).toBeGreaterThan(5);
251
+ expect(result.errors.some((e) => e.startsWith('ATLASSIAN_URL:'))).toBe(true);
252
+ expect(result.errors.some((e) => e.startsWith('ATLASSIAN_EMAIL:'))).toBe(true);
253
+ expect(result.errors.some((e) => e.startsWith('ATLASSIAN_API_TOKEN:'))).toBe(true);
254
+ expect(result.errors.some((e) => e.startsWith('GITHUB_ORG:'))).toBe(true);
255
+ expect(result.errors.some((e) => e.startsWith('GITHUB_TOKEN:'))).toBe(true);
256
+ expect(result.errors.some((e) => e.startsWith('repository:'))).toBe(true);
257
+ });
258
+
259
+ it('警告も正しく集約される', () => {
260
+ const result = validateEnvironmentConfig({
261
+ atlassianUrl: 'http://mycompany.atlassian.net', // http警告
262
+ atlassianApiToken: 'shorttoken', // 短い警告
263
+ githubToken: 'abcdefghijklmnopqrstuvwxyz1234567890', // プレフィックスなし警告
264
+ });
265
+
266
+ expect(result.warnings.length).toBeGreaterThan(0);
267
+ expect(result.warnings.some((w) => w.includes('ATLASSIAN_URL:'))).toBe(true);
268
+ expect(result.warnings.some((w) => w.includes('ATLASSIAN_API_TOKEN:'))).toBe(true);
269
+ expect(result.warnings.some((w) => w.includes('GITHUB_TOKEN:'))).toBe(true);
270
+ });
271
+ });
272
+ });