@sk8metal/michi-cli 0.0.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.
Files changed (99) hide show
  1. package/LICENSE +22 -0
  2. package/README.md +465 -0
  3. package/dist/scripts/config/config-schema.d.ts +707 -0
  4. package/dist/scripts/config/config-schema.d.ts.map +1 -0
  5. package/dist/scripts/config/config-schema.js +100 -0
  6. package/dist/scripts/config/config-schema.js.map +1 -0
  7. package/dist/scripts/config-interactive.d.ts +10 -0
  8. package/dist/scripts/config-interactive.d.ts.map +1 -0
  9. package/dist/scripts/config-interactive.js +357 -0
  10. package/dist/scripts/config-interactive.js.map +1 -0
  11. package/dist/scripts/confluence-sync.d.ts +54 -0
  12. package/dist/scripts/confluence-sync.d.ts.map +1 -0
  13. package/dist/scripts/confluence-sync.js +407 -0
  14. package/dist/scripts/confluence-sync.js.map +1 -0
  15. package/dist/scripts/create-project.d.ts +12 -0
  16. package/dist/scripts/create-project.d.ts.map +1 -0
  17. package/dist/scripts/create-project.js +244 -0
  18. package/dist/scripts/create-project.js.map +1 -0
  19. package/dist/scripts/jira-sync.d.ts +39 -0
  20. package/dist/scripts/jira-sync.d.ts.map +1 -0
  21. package/dist/scripts/jira-sync.js +549 -0
  22. package/dist/scripts/jira-sync.js.map +1 -0
  23. package/dist/scripts/list-projects.d.ts +7 -0
  24. package/dist/scripts/list-projects.d.ts.map +1 -0
  25. package/dist/scripts/list-projects.js +61 -0
  26. package/dist/scripts/list-projects.js.map +1 -0
  27. package/dist/scripts/markdown-to-confluence.d.ts +19 -0
  28. package/dist/scripts/markdown-to-confluence.d.ts.map +1 -0
  29. package/dist/scripts/markdown-to-confluence.js +121 -0
  30. package/dist/scripts/markdown-to-confluence.js.map +1 -0
  31. package/dist/scripts/multi-project-estimate.d.ts +27 -0
  32. package/dist/scripts/multi-project-estimate.d.ts.map +1 -0
  33. package/dist/scripts/multi-project-estimate.js +202 -0
  34. package/dist/scripts/multi-project-estimate.js.map +1 -0
  35. package/dist/scripts/phase-runner.d.ts +19 -0
  36. package/dist/scripts/phase-runner.d.ts.map +1 -0
  37. package/dist/scripts/phase-runner.js +260 -0
  38. package/dist/scripts/phase-runner.js.map +1 -0
  39. package/dist/scripts/pr-automation.d.ts +12 -0
  40. package/dist/scripts/pr-automation.d.ts.map +1 -0
  41. package/dist/scripts/pr-automation.js +45 -0
  42. package/dist/scripts/pr-automation.js.map +1 -0
  43. package/dist/scripts/pre-flight-check.d.ts +15 -0
  44. package/dist/scripts/pre-flight-check.d.ts.map +1 -0
  45. package/dist/scripts/pre-flight-check.js +244 -0
  46. package/dist/scripts/pre-flight-check.js.map +1 -0
  47. package/dist/scripts/resource-dashboard.d.ts +7 -0
  48. package/dist/scripts/resource-dashboard.d.ts.map +1 -0
  49. package/dist/scripts/resource-dashboard.js +100 -0
  50. package/dist/scripts/resource-dashboard.js.map +1 -0
  51. package/dist/scripts/setup-existing-project.d.ts +13 -0
  52. package/dist/scripts/setup-existing-project.d.ts.map +1 -0
  53. package/dist/scripts/setup-existing-project.js +334 -0
  54. package/dist/scripts/setup-existing-project.js.map +1 -0
  55. package/dist/scripts/utils/config-loader.d.ts +20 -0
  56. package/dist/scripts/utils/config-loader.d.ts.map +1 -0
  57. package/dist/scripts/utils/config-loader.js +291 -0
  58. package/dist/scripts/utils/config-loader.js.map +1 -0
  59. package/dist/scripts/utils/config-validator.d.ts +34 -0
  60. package/dist/scripts/utils/config-validator.d.ts.map +1 -0
  61. package/dist/scripts/utils/config-validator.js +286 -0
  62. package/dist/scripts/utils/config-validator.js.map +1 -0
  63. package/dist/scripts/utils/confluence-hierarchy.d.ts +47 -0
  64. package/dist/scripts/utils/confluence-hierarchy.d.ts.map +1 -0
  65. package/dist/scripts/utils/confluence-hierarchy.js +516 -0
  66. package/dist/scripts/utils/confluence-hierarchy.js.map +1 -0
  67. package/dist/scripts/utils/feature-name-validator.d.ts +25 -0
  68. package/dist/scripts/utils/feature-name-validator.d.ts.map +1 -0
  69. package/dist/scripts/utils/feature-name-validator.js +113 -0
  70. package/dist/scripts/utils/feature-name-validator.js.map +1 -0
  71. package/dist/scripts/utils/project-meta.d.ts +23 -0
  72. package/dist/scripts/utils/project-meta.d.ts.map +1 -0
  73. package/dist/scripts/utils/project-meta.js +50 -0
  74. package/dist/scripts/utils/project-meta.js.map +1 -0
  75. package/dist/scripts/validate-phase.d.ts +17 -0
  76. package/dist/scripts/validate-phase.d.ts.map +1 -0
  77. package/dist/scripts/validate-phase.js +234 -0
  78. package/dist/scripts/validate-phase.js.map +1 -0
  79. package/dist/scripts/workflow-orchestrator.d.ts +35 -0
  80. package/dist/scripts/workflow-orchestrator.d.ts.map +1 -0
  81. package/dist/scripts/workflow-orchestrator.js +130 -0
  82. package/dist/scripts/workflow-orchestrator.js.map +1 -0
  83. package/dist/src/cli.d.ts +11 -0
  84. package/dist/src/cli.d.ts.map +1 -0
  85. package/dist/src/cli.js +229 -0
  86. package/dist/src/cli.js.map +1 -0
  87. package/docs/config-reference.md +666 -0
  88. package/docs/customization-guide.md +432 -0
  89. package/docs/multi-project.md +233 -0
  90. package/docs/new-project-setup.md +576 -0
  91. package/docs/phase-automation.md +361 -0
  92. package/docs/quick-reference.md +291 -0
  93. package/docs/setup.md +404 -0
  94. package/docs/tasks-template.md +550 -0
  95. package/docs/testing.md +202 -0
  96. package/docs/workflow.md +342 -0
  97. package/env.example +44 -0
  98. package/mcp.json.example +14 -0
  99. package/package.json +97 -0
@@ -0,0 +1,19 @@
1
+ /**
2
+ * Markdown → Confluence Storage Format 変換
3
+ */
4
+ /**
5
+ * Markdown を Confluence Storage Format (HTML) に変換
6
+ */
7
+ export declare function convertMarkdownToConfluence(markdown: string): string;
8
+ /**
9
+ * Confluenceページテンプレートを生成
10
+ */
11
+ export interface ConfluencePageOptions {
12
+ title: string;
13
+ githubUrl: string;
14
+ content: string;
15
+ approvers?: string[];
16
+ projectName?: string;
17
+ }
18
+ export declare function createConfluencePage(options: ConfluencePageOptions): string;
19
+ //# sourceMappingURL=markdown-to-confluence.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"markdown-to-confluence.d.ts","sourceRoot":"","sources":["../../scripts/markdown-to-confluence.ts"],"names":[],"mappings":"AAAA;;GAEG;AAUH;;GAEG;AACH,wBAAgB,2BAA2B,CAAC,QAAQ,EAAE,MAAM,GAAG,MAAM,CAUpE;AA+ED;;GAEG;AACH,MAAM,WAAW,qBAAqB;IACpC,KAAK,EAAE,MAAM,CAAC;IACd,SAAS,EAAE,MAAM,CAAC;IAClB,OAAO,EAAE,MAAM,CAAC;IAChB,SAAS,CAAC,EAAE,MAAM,EAAE,CAAC;IACrB,WAAW,CAAC,EAAE,MAAM,CAAC;CACtB;AAED,wBAAgB,oBAAoB,CAAC,OAAO,EAAE,qBAAqB,GAAG,MAAM,CA0B3E"}
@@ -0,0 +1,121 @@
1
+ /**
2
+ * Markdown → Confluence Storage Format 変換
3
+ */
4
+ import MarkdownIt from 'markdown-it';
5
+ const md = new MarkdownIt({
6
+ html: true,
7
+ breaks: true,
8
+ linkify: true
9
+ });
10
+ /**
11
+ * Markdown を Confluence Storage Format (HTML) に変換
12
+ */
13
+ export function convertMarkdownToConfluence(markdown) {
14
+ // MarkdownIt でHTMLに変換
15
+ let html = md.render(markdown);
16
+ // Confluence固有の変換
17
+ html = convertCodeBlocks(html);
18
+ html = convertTables(html);
19
+ html = convertInfoBoxes(html);
20
+ return html;
21
+ }
22
+ /**
23
+ * コードブロックをConfluenceマクロに変換
24
+ */
25
+ function convertCodeBlocks(html) {
26
+ // <pre><code class="language-xxx">...</code></pre>
27
+ // → <ac:structured-macro ac:name="code"><ac:parameter ac:name="language">xxx</ac:parameter>...</ac:structured-macro>
28
+ return html.replace(/<pre><code class="language-(\w+)">([\s\S]*?)<\/code><\/pre>/g, (match, lang, code) => {
29
+ const decodedCode = decodeHtmlEntities(code);
30
+ return `<ac:structured-macro ac:name="code">
31
+ <ac:parameter ac:name="language">${lang}</ac:parameter>
32
+ <ac:plain-text-body><![CDATA[${decodedCode}]]></ac:plain-text-body>
33
+ </ac:structured-macro>`;
34
+ });
35
+ }
36
+ /**
37
+ * テーブルをConfluence形式に変換(そのままHTMLでOK)
38
+ */
39
+ function convertTables(html) {
40
+ // HTMLテーブルはConfluenceでもサポートされているのでそのまま
41
+ return html;
42
+ }
43
+ /**
44
+ * 特殊なブロック(> で始まる引用など)をConfluence infoマクロに変換
45
+ */
46
+ function convertInfoBoxes(html) {
47
+ // <blockquote>...</blockquote> → <ac:structured-macro ac:name="info">
48
+ let transformed = html.replace(/<blockquote>\s*<p><strong>(.*?)<\/strong>:\s*([\s\S]*?)<\/p>\s*<\/blockquote>/g, (match, title, content) => {
49
+ return `<ac:structured-macro ac:name="info">
50
+ <ac:parameter ac:name="title">${title}</ac:parameter>
51
+ <ac:rich-text-body>
52
+ <p>${content}</p>
53
+ </ac:rich-text-body>
54
+ </ac:structured-macro>`;
55
+ });
56
+ // 通常のblockquoteもinfoマクロに
57
+ transformed = transformed.replace(/<blockquote>([\s\S]*?)<\/blockquote>/g, (match, content) => {
58
+ return `<ac:structured-macro ac:name="info">
59
+ <ac:rich-text-body>
60
+ ${content}
61
+ </ac:rich-text-body>
62
+ </ac:structured-macro>`;
63
+ });
64
+ return transformed;
65
+ }
66
+ /**
67
+ * HTMLエンティティをデコード
68
+ */
69
+ function decodeHtmlEntities(text) {
70
+ const entities = {
71
+ '&lt;': '<',
72
+ '&gt;': '>',
73
+ '&amp;': '&',
74
+ '&quot;': '"',
75
+ '&#39;': "'",
76
+ '&nbsp;': ' '
77
+ };
78
+ return text.replace(/&[a-z]+;|&#\d+;/g, (entity) => {
79
+ return entities[entity] || entity;
80
+ });
81
+ }
82
+ export function createConfluencePage(options) {
83
+ const { title, githubUrl, content, approvers = ['企画', '部長'], projectName } = options;
84
+ const approversList = approvers.map(a => a.startsWith('@') ? a : `@${a}`).join(',');
85
+ return `
86
+ <ac:structured-macro ac:name="info">
87
+ <ac:parameter ac:name="title">GitHub連携</ac:parameter>
88
+ <ac:rich-text-body>
89
+ <p>📄 最新版は <a href="${githubUrl}">GitHub</a> で管理</p>
90
+ <p>編集はGitHubで行い、自動同期されます</p>
91
+ ${projectName ? `<p><strong>プロジェクト</strong>: ${projectName}</p>` : ''}
92
+ </ac:rich-text-body>
93
+ </ac:structured-macro>
94
+
95
+ <hr/>
96
+
97
+ ${content}
98
+
99
+ <hr/>
100
+
101
+ <ac:structured-macro ac:name="page-properties">
102
+ <ac:parameter ac:name="approval">${approversList}</ac:parameter>
103
+ <ac:parameter ac:name="status">レビュー待ち</ac:parameter>
104
+ </ac:structured-macro>
105
+ `.trim();
106
+ }
107
+ // CLI実行用
108
+ if (import.meta.url === `file://${process.argv[1]}`) {
109
+ const { readFileSync } = await import('fs');
110
+ const { resolve } = await import('path');
111
+ const args = process.argv.slice(2);
112
+ if (args.length === 0) {
113
+ console.error('Usage: tsx markdown-to-confluence.ts <markdown-file>');
114
+ process.exit(1);
115
+ }
116
+ const markdownFile = resolve(args[0]);
117
+ const markdown = readFileSync(markdownFile, 'utf-8');
118
+ const confluenceHtml = convertMarkdownToConfluence(markdown);
119
+ console.log(confluenceHtml);
120
+ }
121
+ //# sourceMappingURL=markdown-to-confluence.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"markdown-to-confluence.js","sourceRoot":"","sources":["../../scripts/markdown-to-confluence.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,27 @@
1
+ /**
2
+ * マルチプロジェクト見積もり集計
3
+ */
4
+ export interface EstimateData {
5
+ feature: string;
6
+ tasks: TaskEstimate[];
7
+ totalDays: number;
8
+ totalPoints: number;
9
+ risks: RiskEstimate[];
10
+ optimistic: number;
11
+ standard: number;
12
+ pessimistic: number;
13
+ }
14
+ interface TaskEstimate {
15
+ name: string;
16
+ days: number;
17
+ assignee: string;
18
+ notes?: string;
19
+ }
20
+ interface RiskEstimate {
21
+ risk: string;
22
+ impact: number;
23
+ mitigation: string;
24
+ }
25
+ declare function aggregateEstimates(): Promise<void>;
26
+ export { aggregateEstimates };
27
+ //# sourceMappingURL=multi-project-estimate.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"multi-project-estimate.d.ts","sourceRoot":"","sources":["../../scripts/multi-project-estimate.ts"],"names":[],"mappings":"AAAA;;GAEG;AAaH,MAAM,WAAW,YAAY;IAC3B,OAAO,EAAE,MAAM,CAAC;IAChB,KAAK,EAAE,YAAY,EAAE,CAAC;IACtB,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;IACpB,KAAK,EAAE,YAAY,EAAE,CAAC;IACtB,UAAU,EAAE,MAAM,CAAC;IACnB,QAAQ,EAAE,MAAM,CAAC;IACjB,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,UAAU,YAAY;IACpB,IAAI,EAAE,MAAM,CAAC;IACb,IAAI,EAAE,MAAM,CAAC;IACb,QAAQ,EAAE,MAAM,CAAC;IACjB,KAAK,CAAC,EAAE,MAAM,CAAC;CAChB;AAED,UAAU,YAAY;IACpB,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,MAAM,CAAC;IACf,UAAU,EAAE,MAAM,CAAC;CACpB;AAoID,iBAAe,kBAAkB,IAAI,OAAO,CAAC,IAAI,CAAC,CAqEjD;AAeD,OAAO,EAAE,kBAAkB,EAAE,CAAC"}
@@ -0,0 +1,202 @@
1
+ /**
2
+ * マルチプロジェクト見積もり集計
3
+ */
4
+ import { Octokit } from '@octokit/rest';
5
+ import { config } from 'dotenv';
6
+ import ExcelJS from 'exceljs';
7
+ import { resolve, join, dirname } from 'path';
8
+ import { writeFileSync, mkdirSync, unlinkSync, readFileSync } from 'fs';
9
+ import { tmpdir } from 'os';
10
+ import { mkdir } from 'fs/promises';
11
+ config();
12
+ /**
13
+ * design.mdから見積もりを抽出(estimate-generator.tsから統合)
14
+ */
15
+ function parseEstimateFromDesign(designPath) {
16
+ const content = readFileSync(designPath, 'utf-8');
17
+ const tasks = [];
18
+ let totalDays = 0;
19
+ // 見積もりテーブルを正規表現で抽出
20
+ const tableRegex = /\|\s*([^|]+)\s*\|\s*(\d+(?:\.\d+)?)\s*\|\s*([^|]+)\s*\|/g;
21
+ let match;
22
+ while ((match = tableRegex.exec(content)) !== null) {
23
+ const [, name, daysStr, assignee] = match;
24
+ const days = parseFloat(daysStr);
25
+ if (!isNaN(days) && name.trim() !== 'タスク' && name.trim() !== '**合計**') {
26
+ tasks.push({
27
+ name: name.trim(),
28
+ days,
29
+ assignee: assignee.trim()
30
+ });
31
+ totalDays += days;
32
+ }
33
+ }
34
+ const risks = [
35
+ { risk: '技術的課題', impact: 5, mitigation: 'プロトタイプ検証' },
36
+ { risk: '要件変更', impact: 3, mitigation: 'バッファ確保' }
37
+ ];
38
+ const riskTotal = risks.reduce((sum, r) => sum + r.impact, 0);
39
+ return {
40
+ feature: 'Unknown',
41
+ tasks,
42
+ totalDays,
43
+ totalPoints: Math.ceil(totalDays / 0.5),
44
+ risks,
45
+ optimistic: totalDays,
46
+ standard: totalDays + riskTotal,
47
+ pessimistic: Math.ceil(totalDays * 1.5)
48
+ };
49
+ }
50
+ /**
51
+ * Excel出力(excel-sync.tsから統合)
52
+ */
53
+ async function exportToExcel(estimates, outputPath) {
54
+ const workbook = new ExcelJS.Workbook();
55
+ const worksheet = workbook.addWorksheet('見積もりサマリー');
56
+ worksheet.columns = [
57
+ { header: 'プロジェクト/機能', key: 'feature', width: 30 },
58
+ { header: '楽観的(人日)', key: 'optimistic', width: 15 },
59
+ { header: '標準的(人日)', key: 'standard', width: 15 },
60
+ { header: '悲観的(人日)', key: 'pessimistic', width: 15 },
61
+ { header: 'ストーリーポイント', key: 'points', width: 18 },
62
+ { header: 'タスク数', key: 'taskCount', width: 12 }
63
+ ];
64
+ for (const estimate of estimates) {
65
+ worksheet.addRow({
66
+ feature: estimate.feature,
67
+ optimistic: estimate.optimistic,
68
+ standard: estimate.standard,
69
+ pessimistic: estimate.pessimistic,
70
+ points: estimate.totalPoints,
71
+ taskCount: estimate.tasks.length
72
+ });
73
+ }
74
+ worksheet.getRow(1).font = { bold: true };
75
+ worksheet.getRow(1).fill = {
76
+ type: 'pattern',
77
+ pattern: 'solid',
78
+ fgColor: { argb: 'FFE0E0E0' }
79
+ };
80
+ const totalRow = worksheet.addRow({
81
+ feature: '合計',
82
+ optimistic: estimates.reduce((sum, e) => sum + e.optimistic, 0),
83
+ standard: estimates.reduce((sum, e) => sum + e.standard, 0),
84
+ pessimistic: estimates.reduce((sum, e) => sum + e.pessimistic, 0),
85
+ points: estimates.reduce((sum, e) => sum + e.totalPoints, 0),
86
+ taskCount: estimates.reduce((sum, e) => sum + e.tasks.length, 0)
87
+ });
88
+ totalRow.font = { bold: true };
89
+ await mkdir(dirname(outputPath), { recursive: true });
90
+ await workbook.xlsx.writeFile(outputPath);
91
+ console.log(`✅ Excel file saved: ${outputPath}`);
92
+ }
93
+ /**
94
+ * content文字列から見積もりを抽出(一時ファイル経由)
95
+ */
96
+ function parseEstimateFromContent(content, featureName) {
97
+ let tempFile = null;
98
+ try {
99
+ // 一時ファイルに書き出してからパース
100
+ const tempDir = join(tmpdir(), 'michi-estimate');
101
+ mkdirSync(tempDir, { recursive: true });
102
+ tempFile = join(tempDir, `design-${Date.now()}.md`);
103
+ writeFileSync(tempFile, content);
104
+ const estimate = parseEstimateFromDesign(tempFile);
105
+ estimate.feature = featureName;
106
+ return estimate;
107
+ }
108
+ catch (error) {
109
+ console.warn(` ⚠️ Failed to parse estimate:`, error instanceof Error ? error.message : error);
110
+ return null;
111
+ }
112
+ finally {
113
+ // 一時ファイルをクリーンアップ
114
+ if (tempFile) {
115
+ try {
116
+ unlinkSync(tempFile);
117
+ }
118
+ catch {
119
+ // 削除失敗は無視
120
+ }
121
+ }
122
+ }
123
+ }
124
+ async function aggregateEstimates() {
125
+ const token = process.env.GITHUB_TOKEN;
126
+ const org = process.env.GITHUB_ORG;
127
+ if (!token || !org) {
128
+ throw new Error('Missing GitHub credentials');
129
+ }
130
+ const octokit = new Octokit({ auth: token });
131
+ const estimates = [];
132
+ console.log('Aggregating estimates from all projects...');
133
+ const { data: repos } = await octokit.repos.listForOrg({ org });
134
+ for (const repo of repos) {
135
+ try {
136
+ // .kiro/specs/ ディレクトリを取得
137
+ const { data: specs } = await octokit.repos.getContent({
138
+ owner: org,
139
+ repo: repo.name,
140
+ path: '.kiro/specs'
141
+ });
142
+ if (Array.isArray(specs)) {
143
+ for (const spec of specs) {
144
+ if (spec.type === 'dir') {
145
+ // design.md を取得
146
+ try {
147
+ const { data: designFile } = await octokit.repos.getContent({
148
+ owner: org,
149
+ repo: repo.name,
150
+ path: `.kiro/specs/${spec.name}/design.md`
151
+ });
152
+ if ('content' in designFile) {
153
+ const content = Buffer.from(designFile.content, 'base64').toString('utf-8');
154
+ // コンテンツから見積もりを抽出
155
+ try {
156
+ const estimateData = parseEstimateFromContent(content, `${repo.name}/${spec.name}`);
157
+ if (estimateData) {
158
+ estimates.push(estimateData);
159
+ console.log(` ✅ Parsed: ${repo.name}/${spec.name} (${estimateData.totalDays}日)`);
160
+ }
161
+ }
162
+ catch (error) {
163
+ console.warn(` ⚠️ Failed to parse ${repo.name}/${spec.name}:`, error instanceof Error ? error.message : error);
164
+ }
165
+ }
166
+ }
167
+ catch {
168
+ continue;
169
+ }
170
+ }
171
+ }
172
+ }
173
+ }
174
+ catch {
175
+ continue;
176
+ }
177
+ }
178
+ // Excel出力
179
+ if (estimates.length > 0) {
180
+ const outputDir = resolve('./estimates');
181
+ mkdirSync(outputDir, { recursive: true });
182
+ const outputPath = join(outputDir, 'multi-project-estimates.xlsx');
183
+ await exportToExcel(estimates, outputPath);
184
+ }
185
+ else {
186
+ console.log('No estimates found');
187
+ }
188
+ }
189
+ // CLI実行
190
+ if (import.meta.url === `file://${process.argv[1]}`) {
191
+ aggregateEstimates()
192
+ .then(() => {
193
+ console.log('✅ Aggregation completed');
194
+ process.exit(0);
195
+ })
196
+ .catch((error) => {
197
+ console.error('❌ Failed:', error.message);
198
+ process.exit(1);
199
+ });
200
+ }
201
+ export { aggregateEstimates };
202
+ //# sourceMappingURL=multi-project-estimate.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"multi-project-estimate.js","sourceRoot":"","sources":["../../scripts/multi-project-estimate.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,19 @@
1
+ /**
2
+ * フェーズランナー
3
+ * 各フェーズを実行し、Confluence/JIRA作成を確実に実行
4
+ */
5
+ type Phase = 'requirements' | 'design' | 'tasks';
6
+ interface PhaseRunResult {
7
+ phase: Phase;
8
+ success: boolean;
9
+ confluenceCreated: boolean;
10
+ jiraCreated: boolean;
11
+ validationPassed: boolean;
12
+ errors: string[];
13
+ }
14
+ /**
15
+ * フェーズを実行
16
+ */
17
+ export declare function runPhase(feature: string, phase: Phase): Promise<PhaseRunResult>;
18
+ export {};
19
+ //# sourceMappingURL=phase-runner.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"phase-runner.d.ts","sourceRoot":"","sources":["../../scripts/phase-runner.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAUH,KAAK,KAAK,GAAG,cAAc,GAAG,QAAQ,GAAG,OAAO,CAAC;AAEjD,UAAU,cAAc;IACtB,KAAK,EAAE,KAAK,CAAC;IACb,OAAO,EAAE,OAAO,CAAC;IACjB,iBAAiB,EAAE,OAAO,CAAC;IAC3B,WAAW,EAAE,OAAO,CAAC;IACrB,gBAAgB,EAAE,OAAO,CAAC;IAC1B,MAAM,EAAE,MAAM,EAAE,CAAC;CAClB;AAoOD;;GAEG;AACH,wBAAsB,QAAQ,CAAC,OAAO,EAAE,MAAM,EAAE,KAAK,EAAE,KAAK,GAAG,OAAO,CAAC,cAAc,CAAC,CAcrF"}