@sk8metal/michi-cli 0.0.1 → 0.0.3
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/CHANGELOG.md +30 -0
- package/README.md +60 -24
- package/dist/scripts/__tests__/validate-phase.test.d.ts +5 -0
- package/dist/scripts/__tests__/validate-phase.test.d.ts.map +1 -0
- package/dist/scripts/__tests__/validate-phase.test.js +162 -0
- package/dist/scripts/__tests__/validate-phase.test.js.map +1 -0
- package/dist/scripts/utils/__tests__/config-validator.test.d.ts +5 -0
- package/dist/scripts/utils/__tests__/config-validator.test.d.ts.map +1 -0
- package/dist/scripts/utils/__tests__/config-validator.test.js +247 -0
- package/dist/scripts/utils/__tests__/config-validator.test.js.map +1 -0
- package/dist/scripts/utils/__tests__/feature-name-validator.test.d.ts +5 -0
- package/dist/scripts/utils/__tests__/feature-name-validator.test.d.ts.map +1 -0
- package/dist/scripts/utils/__tests__/feature-name-validator.test.js +106 -0
- package/dist/scripts/utils/__tests__/feature-name-validator.test.js.map +1 -0
- package/dist/scripts/utils/config-loader.js +1 -1
- package/dist/scripts/utils/config-loader.js.map +1 -1
- package/dist/scripts/utils/confluence-hierarchy.d.ts.map +1 -1
- package/dist/scripts/utils/confluence-hierarchy.js +2 -1
- package/dist/scripts/utils/confluence-hierarchy.js.map +1 -1
- package/dist/src/__tests__/cli.test.d.ts +5 -0
- package/dist/src/__tests__/cli.test.d.ts.map +1 -0
- package/dist/src/__tests__/cli.test.js +58 -0
- package/dist/src/__tests__/cli.test.js.map +1 -0
- package/dist/src/cli.js +0 -0
- package/dist/vitest.config.d.ts +3 -0
- package/dist/vitest.config.d.ts.map +1 -0
- package/dist/vitest.config.js +29 -0
- package/dist/vitest.config.js.map +1 -0
- package/docs/setup.md +1 -1
- package/package.json +8 -4
- package/scripts/__tests__/README.md +101 -0
- package/scripts/__tests__/validate-phase.test.ts +185 -0
- package/scripts/config/config-schema.ts +130 -0
- package/scripts/config/default-config.json +57 -0
- package/scripts/config-interactive.ts +494 -0
- package/scripts/confluence-sync.ts +503 -0
- package/scripts/create-project.ts +293 -0
- package/scripts/jira-sync.ts +644 -0
- package/scripts/list-projects.ts +85 -0
- package/scripts/markdown-to-confluence.ts +161 -0
- package/scripts/multi-project-estimate.ts +255 -0
- package/scripts/phase-runner.ts +303 -0
- package/scripts/pr-automation.ts +67 -0
- package/scripts/pre-flight-check.ts +285 -0
- package/scripts/resource-dashboard.ts +124 -0
- package/scripts/setup-env.sh +52 -0
- package/scripts/setup-existing-project.ts +381 -0
- package/scripts/setup-existing.sh +145 -0
- package/scripts/utils/__tests__/config-validator.test.ts +302 -0
- package/scripts/utils/__tests__/feature-name-validator.test.ts +129 -0
- package/scripts/utils/config-loader.ts +326 -0
- package/scripts/utils/config-validator.ts +347 -0
- package/scripts/utils/confluence-hierarchy.ts +855 -0
- package/scripts/utils/feature-name-validator.ts +135 -0
- package/scripts/utils/project-meta.ts +69 -0
- package/scripts/validate-phase.ts +279 -0
- package/scripts/workflow-orchestrator.ts +178 -0
|
@@ -0,0 +1,124 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* リソースダッシュボード生成
|
|
3
|
+
* プロジェクト横断のリソース管理ダッシュボードをConfluenceに作成
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { Octokit } from '@octokit/rest';
|
|
7
|
+
import { config } from 'dotenv';
|
|
8
|
+
import { ConfluenceClient } from './confluence-sync.js';
|
|
9
|
+
|
|
10
|
+
config();
|
|
11
|
+
|
|
12
|
+
interface ProjectResource {
|
|
13
|
+
projectName: string;
|
|
14
|
+
projectId: string;
|
|
15
|
+
status: string;
|
|
16
|
+
team: string[];
|
|
17
|
+
progress: number;
|
|
18
|
+
}
|
|
19
|
+
|
|
20
|
+
/**
|
|
21
|
+
* HTMLエスケープ
|
|
22
|
+
*/
|
|
23
|
+
function escapeHtml(text: string): string {
|
|
24
|
+
return text
|
|
25
|
+
.replace(/&/g, '&')
|
|
26
|
+
.replace(/</g, '<')
|
|
27
|
+
.replace(/>/g, '>')
|
|
28
|
+
.replace(/"/g, '"')
|
|
29
|
+
.replace(/'/g, ''');
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
async function createResourceDashboard(): Promise<void> {
|
|
33
|
+
const token = process.env.GITHUB_TOKEN;
|
|
34
|
+
const org = process.env.GITHUB_ORG;
|
|
35
|
+
|
|
36
|
+
if (!token || !org) {
|
|
37
|
+
throw new Error('Missing GitHub credentials');
|
|
38
|
+
}
|
|
39
|
+
|
|
40
|
+
const octokit = new Octokit({ auth: token });
|
|
41
|
+
const projects: ProjectResource[] = [];
|
|
42
|
+
|
|
43
|
+
console.log('Gathering project resources...');
|
|
44
|
+
|
|
45
|
+
const { data: repos } = await octokit.repos.listForOrg({ org });
|
|
46
|
+
|
|
47
|
+
for (const repo of repos) {
|
|
48
|
+
try {
|
|
49
|
+
const { data } = await octokit.repos.getContent({
|
|
50
|
+
owner: org,
|
|
51
|
+
repo: repo.name,
|
|
52
|
+
path: '.kiro/project.json'
|
|
53
|
+
});
|
|
54
|
+
|
|
55
|
+
if ('content' in data) {
|
|
56
|
+
const content = Buffer.from(data.content, 'base64').toString('utf-8');
|
|
57
|
+
const meta = JSON.parse(content);
|
|
58
|
+
|
|
59
|
+
projects.push({
|
|
60
|
+
projectName: meta.projectName,
|
|
61
|
+
projectId: meta.projectId,
|
|
62
|
+
status: meta.status,
|
|
63
|
+
team: meta.team,
|
|
64
|
+
progress: 0 // TODO: JIRAから進捗を取得
|
|
65
|
+
});
|
|
66
|
+
}
|
|
67
|
+
} catch (error) {
|
|
68
|
+
console.warn(`⚠️ Skipping ${repo.name}:`, error instanceof Error ? error.message : 'Unknown error');
|
|
69
|
+
continue;
|
|
70
|
+
}
|
|
71
|
+
}
|
|
72
|
+
|
|
73
|
+
// Confluenceページコンテンツ生成
|
|
74
|
+
const dashboardContent = `
|
|
75
|
+
<h2>プロジェクトリソースダッシュボード</h2>
|
|
76
|
+
|
|
77
|
+
<p>更新日時: ${new Date().toLocaleString('ja-JP')}</p>
|
|
78
|
+
|
|
79
|
+
<table>
|
|
80
|
+
<tr>
|
|
81
|
+
<th>プロジェクト</th>
|
|
82
|
+
<th>ID</th>
|
|
83
|
+
<th>ステータス</th>
|
|
84
|
+
<th>チーム</th>
|
|
85
|
+
<th>進捗</th>
|
|
86
|
+
</tr>
|
|
87
|
+
${projects.map(p => `
|
|
88
|
+
<tr>
|
|
89
|
+
<td>${escapeHtml(p.projectName)}</td>
|
|
90
|
+
<td>${escapeHtml(p.projectId)}</td>
|
|
91
|
+
<td>${escapeHtml(p.status)}</td>
|
|
92
|
+
<td>${escapeHtml(p.team.join(', '))}</td>
|
|
93
|
+
<td>${p.progress}%</td>
|
|
94
|
+
</tr>
|
|
95
|
+
`).join('')}
|
|
96
|
+
</table>
|
|
97
|
+
|
|
98
|
+
<p><strong>合計</strong>: ${projects.length} プロジェクト</p>
|
|
99
|
+
`.trim();
|
|
100
|
+
|
|
101
|
+
console.log('Dashboard content generated');
|
|
102
|
+
console.log(dashboardContent);
|
|
103
|
+
|
|
104
|
+
// TODO: Confluenceに保存
|
|
105
|
+
// const confluenceConfig = getConfluenceConfig();
|
|
106
|
+
// const client = new ConfluenceClient(confluenceConfig);
|
|
107
|
+
// await client.createPage('PRD', 'プロジェクトリソースダッシュボード', dashboardContent);
|
|
108
|
+
}
|
|
109
|
+
|
|
110
|
+
// CLI実行
|
|
111
|
+
if (import.meta.url === `file://${process.argv[1]}`) {
|
|
112
|
+
createResourceDashboard()
|
|
113
|
+
.then(() => {
|
|
114
|
+
console.log('✅ Dashboard created');
|
|
115
|
+
process.exit(0);
|
|
116
|
+
})
|
|
117
|
+
.catch((error) => {
|
|
118
|
+
console.error('❌ Failed:', error.message);
|
|
119
|
+
process.exit(1);
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
|
|
123
|
+
export { createResourceDashboard };
|
|
124
|
+
|
|
@@ -0,0 +1,52 @@
|
|
|
1
|
+
#!/bin/bash
|
|
2
|
+
|
|
3
|
+
# 環境変数設定スクリプト
|
|
4
|
+
# このスクリプトは.envファイルのテンプレートを作成します
|
|
5
|
+
|
|
6
|
+
# 既存の.envファイルがある場合は確認
|
|
7
|
+
if [ -f .env ]; then
|
|
8
|
+
echo "⚠️ .env ファイルが既に存在します"
|
|
9
|
+
read -p "上書きしますか? (y/N): " -n 1 -r
|
|
10
|
+
echo
|
|
11
|
+
if [[ ! $REPLY =~ ^[Yy]$ ]]; then
|
|
12
|
+
echo "キャンセルしました"
|
|
13
|
+
exit 0
|
|
14
|
+
fi
|
|
15
|
+
fi
|
|
16
|
+
|
|
17
|
+
cat > .env << 'EOF'
|
|
18
|
+
# Atlassian設定(MCP + REST API共通)
|
|
19
|
+
ATLASSIAN_URL=https://your-domain.atlassian.net
|
|
20
|
+
ATLASSIAN_EMAIL=your-email@company.com
|
|
21
|
+
ATLASSIAN_API_TOKEN=your-token-here
|
|
22
|
+
|
|
23
|
+
# GitHub設定
|
|
24
|
+
GITHUB_ORG=your-org
|
|
25
|
+
GITHUB_TOKEN=ghp_xxx
|
|
26
|
+
GITHUB_REPO=sk8metalme/michi
|
|
27
|
+
|
|
28
|
+
# Confluence共有スペース
|
|
29
|
+
CONFLUENCE_PRD_SPACE=PRD
|
|
30
|
+
CONFLUENCE_QA_SPACE=QA
|
|
31
|
+
CONFLUENCE_RELEASE_SPACE=RELEASE
|
|
32
|
+
|
|
33
|
+
# JIRAプロジェクトキー(カンマ区切り)
|
|
34
|
+
JIRA_PROJECT_KEYS=MICHI,PRJA,PRJB
|
|
35
|
+
|
|
36
|
+
# Slack通知用(オプション)
|
|
37
|
+
SLACK_WEBHOOK_URL=https://hooks.slack.com/xxx
|
|
38
|
+
EOF
|
|
39
|
+
|
|
40
|
+
# セキュアなパーミッション設定(所有者のみ読み取り/書き込み可能)
|
|
41
|
+
chmod 600 .env
|
|
42
|
+
|
|
43
|
+
echo "✅ .env ファイルが作成されました"
|
|
44
|
+
echo "✅ パーミッション 600 を設定しました(所有者のみ読み取り/書き込み可能)"
|
|
45
|
+
echo ""
|
|
46
|
+
echo "⚠️ 重要: .env ファイルをGitにコミットしないでください"
|
|
47
|
+
echo " .gitignore で除外されていることを確認してください"
|
|
48
|
+
echo ""
|
|
49
|
+
echo "📝 次のステップ:"
|
|
50
|
+
echo " 1. 実際の認証情報で .env の値を更新してください"
|
|
51
|
+
echo " 2. セットアップ手順に従ってください"
|
|
52
|
+
|
|
@@ -0,0 +1,381 @@
|
|
|
1
|
+
#!/usr/bin/env tsx
|
|
2
|
+
/**
|
|
3
|
+
* 既存プロジェクトにMichiワークフローを追加するスクリプト
|
|
4
|
+
*
|
|
5
|
+
* 使い方:
|
|
6
|
+
* cd /path/to/existing-repo
|
|
7
|
+
* npx tsx /path/to/michi/scripts/setup-existing-project.ts \
|
|
8
|
+
* --michi-path /path/to/michi \
|
|
9
|
+
* --project-name "既存プロジェクト" \
|
|
10
|
+
* --jira-key "EXIST"
|
|
11
|
+
*/
|
|
12
|
+
|
|
13
|
+
import { cpSync, existsSync, mkdirSync, writeFileSync, readFileSync } from 'fs';
|
|
14
|
+
import { resolve, join, basename } from 'path';
|
|
15
|
+
import { execSync } from 'child_process';
|
|
16
|
+
|
|
17
|
+
interface SetupConfig {
|
|
18
|
+
michiPath: string; // Michiリポジトリのパス
|
|
19
|
+
projectName: string; // プロジェクト表示名
|
|
20
|
+
jiraKey: string; // JIRAプロジェクトキー
|
|
21
|
+
labels?: string[]; // Confluenceラベル(オプション)
|
|
22
|
+
}
|
|
23
|
+
|
|
24
|
+
function parseArgs(): SetupConfig {
|
|
25
|
+
const args = process.argv.slice(2);
|
|
26
|
+
const config: Partial<SetupConfig> = {};
|
|
27
|
+
|
|
28
|
+
for (let i = 0; i < args.length; i += 2) {
|
|
29
|
+
const key = args[i].replace(/^--/, '');
|
|
30
|
+
const value = args[i + 1];
|
|
31
|
+
|
|
32
|
+
switch (key) {
|
|
33
|
+
case 'michi-path':
|
|
34
|
+
config.michiPath = value;
|
|
35
|
+
break;
|
|
36
|
+
case 'project-name':
|
|
37
|
+
config.projectName = value;
|
|
38
|
+
break;
|
|
39
|
+
case 'jira-key':
|
|
40
|
+
config.jiraKey = value;
|
|
41
|
+
break;
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
|
|
45
|
+
// デフォルト値
|
|
46
|
+
if (!config.michiPath) {
|
|
47
|
+
config.michiPath = resolve(__dirname, '..');
|
|
48
|
+
}
|
|
49
|
+
|
|
50
|
+
// 必須フィールドチェック
|
|
51
|
+
if (!config.projectName || !config.jiraKey) {
|
|
52
|
+
console.error('Missing required parameters');
|
|
53
|
+
console.error('Usage: tsx setup-existing-project.ts --project-name <name> --jira-key <key> [--michi-path <path>]');
|
|
54
|
+
process.exit(1);
|
|
55
|
+
}
|
|
56
|
+
|
|
57
|
+
return config as SetupConfig;
|
|
58
|
+
}
|
|
59
|
+
|
|
60
|
+
async function setupExistingProject(config: SetupConfig): Promise<void> {
|
|
61
|
+
const currentDir = process.cwd();
|
|
62
|
+
const projectId = basename(currentDir);
|
|
63
|
+
|
|
64
|
+
console.log('🚀 既存プロジェクトにMichiワークフローを追加');
|
|
65
|
+
console.log(` プロジェクト: ${config.projectName}`);
|
|
66
|
+
console.log(` ディレクトリ: ${currentDir}`);
|
|
67
|
+
console.log(` Michiパス: ${config.michiPath}`);
|
|
68
|
+
console.log('');
|
|
69
|
+
|
|
70
|
+
// Step 1: cc-sdd導入確認
|
|
71
|
+
console.log('📦 Step 1: Checking cc-sdd installation...');
|
|
72
|
+
if (!existsSync('.cursor/commands/kiro')) {
|
|
73
|
+
console.log(' Installing cc-sdd...');
|
|
74
|
+
execSync('npx cc-sdd@latest --cursor --lang ja --yes', { stdio: 'inherit' });
|
|
75
|
+
console.log(' ✅ cc-sdd installed');
|
|
76
|
+
} else {
|
|
77
|
+
console.log(' ✅ cc-sdd already installed');
|
|
78
|
+
}
|
|
79
|
+
|
|
80
|
+
// Step 2: .kiro ディレクトリ作成
|
|
81
|
+
console.log('\n📁 Step 2: Creating .kiro directory structure...');
|
|
82
|
+
mkdirSync('.kiro/settings/templates', { recursive: true });
|
|
83
|
+
mkdirSync('.kiro/steering', { recursive: true });
|
|
84
|
+
mkdirSync('.kiro/specs', { recursive: true });
|
|
85
|
+
console.log(' ✅ Directory structure created');
|
|
86
|
+
|
|
87
|
+
// Step 3: プロジェクトメタデータ作成
|
|
88
|
+
console.log('\n📝 Step 3: Creating project metadata...');
|
|
89
|
+
|
|
90
|
+
// GitHub URLを取得(既存リポジトリから)
|
|
91
|
+
let repoUrl = '';
|
|
92
|
+
try {
|
|
93
|
+
repoUrl = execSync('git config --get remote.origin.url', { encoding: 'utf-8' }).trim();
|
|
94
|
+
// SSH形式をHTTPS形式に変換
|
|
95
|
+
if (repoUrl.startsWith('git@github.com:')) {
|
|
96
|
+
repoUrl = repoUrl.replace('git@github.com:', 'https://github.com/').replace('.git', '');
|
|
97
|
+
}
|
|
98
|
+
} catch {
|
|
99
|
+
repoUrl = `https://github.com/org/${projectId}`;
|
|
100
|
+
}
|
|
101
|
+
|
|
102
|
+
const labels = config.labels || (() => {
|
|
103
|
+
// プロジェクトIDからプロジェクトラベル生成
|
|
104
|
+
const projectLabel = projectId.toLowerCase().replace(/[^a-z0-9-]/g, '');
|
|
105
|
+
const labelSet = new Set([`project:${projectLabel}`]);
|
|
106
|
+
|
|
107
|
+
// ハイフンが存在する場合のみサービスラベルを生成
|
|
108
|
+
if (projectId.includes('-')) {
|
|
109
|
+
const parts = projectId.split('-');
|
|
110
|
+
const servicePart = parts[parts.length - 1];
|
|
111
|
+
const serviceLabel = servicePart.toLowerCase().replace(/[^a-z0-9-]/g, '');
|
|
112
|
+
|
|
113
|
+
// サービスラベルがプロジェクトラベルと異なる場合のみ追加
|
|
114
|
+
if (serviceLabel !== projectLabel) {
|
|
115
|
+
labelSet.add(`service:${serviceLabel}`);
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
|
|
119
|
+
return Array.from(labelSet);
|
|
120
|
+
})();
|
|
121
|
+
|
|
122
|
+
const projectJson = {
|
|
123
|
+
projectId,
|
|
124
|
+
projectName: config.projectName,
|
|
125
|
+
jiraProjectKey: config.jiraKey,
|
|
126
|
+
confluenceLabels: labels,
|
|
127
|
+
status: 'active',
|
|
128
|
+
team: [],
|
|
129
|
+
stakeholders: ['@企画', '@部長'],
|
|
130
|
+
repository: repoUrl,
|
|
131
|
+
description: `${config.projectName}の開発`
|
|
132
|
+
};
|
|
133
|
+
|
|
134
|
+
writeFileSync('.kiro/project.json', JSON.stringify(projectJson, null, 2));
|
|
135
|
+
console.log(' ✅ project.json created');
|
|
136
|
+
|
|
137
|
+
// Step 4: Michiから共通ルールをコピー
|
|
138
|
+
console.log('\n📋 Step 4: Copying common rules from Michi...');
|
|
139
|
+
|
|
140
|
+
const rulesToCopy = [
|
|
141
|
+
'multi-project.mdc',
|
|
142
|
+
'github-ssot.mdc',
|
|
143
|
+
'atlassian-mcp.mdc'
|
|
144
|
+
];
|
|
145
|
+
|
|
146
|
+
for (const rule of rulesToCopy) {
|
|
147
|
+
const src = join(config.michiPath, '.cursor/rules', rule);
|
|
148
|
+
const dest = join(currentDir, '.cursor/rules', rule);
|
|
149
|
+
if (existsSync(src)) {
|
|
150
|
+
cpSync(src, dest);
|
|
151
|
+
console.log(` ✅ ${rule}`);
|
|
152
|
+
} else {
|
|
153
|
+
console.log(` ⚠️ ${rule} not found in Michi`);
|
|
154
|
+
}
|
|
155
|
+
}
|
|
156
|
+
|
|
157
|
+
// Step 5: カスタムコマンドをコピー
|
|
158
|
+
console.log('\n🔧 Step 5: Copying custom commands...');
|
|
159
|
+
|
|
160
|
+
const commandsToCopy = [
|
|
161
|
+
'confluence-sync.md',
|
|
162
|
+
'project-switch.md'
|
|
163
|
+
];
|
|
164
|
+
|
|
165
|
+
for (const cmd of commandsToCopy) {
|
|
166
|
+
const src = join(config.michiPath, '.cursor/commands/kiro', cmd);
|
|
167
|
+
const dest = join(currentDir, '.cursor/commands/kiro', cmd);
|
|
168
|
+
if (existsSync(src)) {
|
|
169
|
+
cpSync(src, dest);
|
|
170
|
+
console.log(` ✅ ${cmd}`);
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
|
|
174
|
+
// Step 6: Steeringテンプレートをコピー
|
|
175
|
+
console.log('\n📚 Step 6: Copying steering templates...');
|
|
176
|
+
|
|
177
|
+
const steeringDir = join(config.michiPath, '.kiro/steering');
|
|
178
|
+
if (existsSync(steeringDir)) {
|
|
179
|
+
cpSync(steeringDir, '.kiro/steering', { recursive: true });
|
|
180
|
+
console.log(' ✅ product.md, tech.md, structure.md');
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
// Step 7: テンプレートをコピー
|
|
184
|
+
console.log('\n📄 Step 7: Copying spec templates...');
|
|
185
|
+
|
|
186
|
+
const templatesDir = join(config.michiPath, '.kiro/settings/templates');
|
|
187
|
+
if (existsSync(templatesDir)) {
|
|
188
|
+
cpSync(templatesDir, '.kiro/settings/templates', { recursive: true });
|
|
189
|
+
console.log(' ✅ requirements.md, design.md, tasks.md');
|
|
190
|
+
}
|
|
191
|
+
|
|
192
|
+
// Step 8: CLIツールのセットアップ案内
|
|
193
|
+
console.log('\n⚙️ Step 8: Setting up Michi CLI...');
|
|
194
|
+
console.log(' ✅ Michi CLI setup complete!');
|
|
195
|
+
console.log('');
|
|
196
|
+
console.log(' 📋 使用方法:');
|
|
197
|
+
console.log(' npx @michi/cli jira:sync <feature>');
|
|
198
|
+
console.log(' npx @michi/cli confluence:sync <feature> requirements');
|
|
199
|
+
console.log(' npx @michi/cli phase:run <feature> tasks');
|
|
200
|
+
console.log('');
|
|
201
|
+
console.log(' または、グローバルインストール:');
|
|
202
|
+
console.log(' npm install -g @michi/cli');
|
|
203
|
+
console.log(' michi jira:sync <feature>');
|
|
204
|
+
|
|
205
|
+
// Step 9: package.json と tsconfig.json をコピー
|
|
206
|
+
console.log('\n📦 Step 9: Setting up package.json and TypeScript...');
|
|
207
|
+
|
|
208
|
+
// 既存の package.json があるかチェック
|
|
209
|
+
const hasPackageJson = existsSync('package.json');
|
|
210
|
+
|
|
211
|
+
if (!hasPackageJson) {
|
|
212
|
+
// package.json がない場合はコピー
|
|
213
|
+
const src = join(config.michiPath, 'package.json');
|
|
214
|
+
if (existsSync(src)) {
|
|
215
|
+
cpSync(src, 'package.json');
|
|
216
|
+
console.log(' ✅ package.json created');
|
|
217
|
+
}
|
|
218
|
+
} else {
|
|
219
|
+
// 既存の package.json にスクリプトを追加
|
|
220
|
+
console.log(' ℹ️ Existing package.json found');
|
|
221
|
+
console.log(' 📝 手動で以下のスクリプトを追加してください:');
|
|
222
|
+
console.log('');
|
|
223
|
+
console.log(' "scripts": {');
|
|
224
|
+
console.log(' "jira:sync": "npx @michi/cli jira:sync",');
|
|
225
|
+
console.log(' "confluence:sync": "npx @michi/cli confluence:sync",');
|
|
226
|
+
console.log(' "phase:run": "npx @michi/cli phase:run",');
|
|
227
|
+
console.log(' "validate:phase": "npx @michi/cli validate:phase",');
|
|
228
|
+
console.log(' "preflight": "npx @michi/cli preflight",');
|
|
229
|
+
console.log(' "project:list": "npx @michi/cli project:list",');
|
|
230
|
+
console.log(' "project:dashboard": "npx @michi/cli project:dashboard",');
|
|
231
|
+
console.log(' "workflow:run": "npx @michi/cli workflow:run"');
|
|
232
|
+
console.log(' }');
|
|
233
|
+
console.log('');
|
|
234
|
+
}
|
|
235
|
+
|
|
236
|
+
// tsconfig.json をコピー
|
|
237
|
+
if (!existsSync('tsconfig.json')) {
|
|
238
|
+
const src = join(config.michiPath, 'tsconfig.json');
|
|
239
|
+
if (existsSync(src)) {
|
|
240
|
+
cpSync(src, 'tsconfig.json');
|
|
241
|
+
console.log(' ✅ tsconfig.json created');
|
|
242
|
+
}
|
|
243
|
+
} else {
|
|
244
|
+
console.log(' ℹ️ Existing tsconfig.json found (kept)');
|
|
245
|
+
}
|
|
246
|
+
|
|
247
|
+
// Step 10: .env テンプレート作成
|
|
248
|
+
console.log('\n🔐 Step 10: Creating .env template...');
|
|
249
|
+
|
|
250
|
+
const envTemplate = `# Atlassian設定(MCP + REST API共通)
|
|
251
|
+
ATLASSIAN_URL=https://your-domain.atlassian.net
|
|
252
|
+
ATLASSIAN_EMAIL=your-email@company.com
|
|
253
|
+
ATLASSIAN_API_TOKEN=your-token-here
|
|
254
|
+
|
|
255
|
+
# GitHub設定
|
|
256
|
+
GITHUB_ORG=your-org
|
|
257
|
+
GITHUB_TOKEN=ghp_xxx
|
|
258
|
+
GITHUB_REPO=${repoUrl.replace('https://github.com/', '')}
|
|
259
|
+
|
|
260
|
+
# Confluence共有スペース
|
|
261
|
+
CONFLUENCE_PRD_SPACE=PRD
|
|
262
|
+
CONFLUENCE_QA_SPACE=QA
|
|
263
|
+
CONFLUENCE_RELEASE_SPACE=RELEASE
|
|
264
|
+
|
|
265
|
+
# JIRAプロジェクトキー
|
|
266
|
+
JIRA_PROJECT_KEYS=${config.jiraKey}
|
|
267
|
+
`;
|
|
268
|
+
|
|
269
|
+
if (!existsSync('.env')) {
|
|
270
|
+
writeFileSync('.env', envTemplate);
|
|
271
|
+
console.log(' ✅ .env template created');
|
|
272
|
+
} else {
|
|
273
|
+
console.log(' ℹ️ .env already exists (kept)');
|
|
274
|
+
}
|
|
275
|
+
|
|
276
|
+
// Step 11: README.md を更新(オプション)
|
|
277
|
+
console.log('\n📖 Step 11: Updating documentation...');
|
|
278
|
+
|
|
279
|
+
const readmePath = 'README.md';
|
|
280
|
+
if (existsSync(readmePath)) {
|
|
281
|
+
const currentReadme = readFileSync(readmePath, 'utf-8');
|
|
282
|
+
|
|
283
|
+
// Michiワークフロー情報を追加
|
|
284
|
+
const workflowSection = `
|
|
285
|
+
|
|
286
|
+
## AI開発ワークフロー
|
|
287
|
+
|
|
288
|
+
このプロジェクトは Michi AI開発フロー自動化システムを使用しています。
|
|
289
|
+
|
|
290
|
+
### 開発フロー
|
|
291
|
+
|
|
292
|
+
\`\`\`
|
|
293
|
+
/kiro:spec-init <機能説明>
|
|
294
|
+
→ /kiro:spec-requirements <feature>
|
|
295
|
+
→ /kiro:spec-design <feature>
|
|
296
|
+
→ /kiro:spec-tasks <feature>
|
|
297
|
+
→ /kiro:spec-impl <feature> <tasks>
|
|
298
|
+
\`\`\`
|
|
299
|
+
|
|
300
|
+
### Confluence/JIRA連携
|
|
301
|
+
|
|
302
|
+
\`\`\`bash
|
|
303
|
+
npm run confluence:sync <feature> # Confluence同期
|
|
304
|
+
npm run jira:sync <feature> # JIRA連携
|
|
305
|
+
npm run github:create-pr <branch> # PR作成
|
|
306
|
+
\`\`\`
|
|
307
|
+
|
|
308
|
+
詳細: [Michi Documentation](https://github.com/sk8metalme/michi)
|
|
309
|
+
`;
|
|
310
|
+
|
|
311
|
+
if (!currentReadme.includes('AI開発ワークフロー')) {
|
|
312
|
+
writeFileSync(readmePath, currentReadme + workflowSection);
|
|
313
|
+
console.log(' ✅ README.md updated');
|
|
314
|
+
} else {
|
|
315
|
+
console.log(' ℹ️ README.md already has workflow section');
|
|
316
|
+
}
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
// Step 12: .gitignore 更新
|
|
320
|
+
console.log('\n🚫 Step 12: Updating .gitignore...');
|
|
321
|
+
|
|
322
|
+
const gitignoreEntries = [
|
|
323
|
+
'# AI Development Workflow',
|
|
324
|
+
'node_modules/',
|
|
325
|
+
'.env',
|
|
326
|
+
'.env.local',
|
|
327
|
+
'dist/',
|
|
328
|
+
'*.log'
|
|
329
|
+
];
|
|
330
|
+
|
|
331
|
+
let gitignore = '';
|
|
332
|
+
if (existsSync('.gitignore')) {
|
|
333
|
+
gitignore = readFileSync('.gitignore', 'utf-8');
|
|
334
|
+
}
|
|
335
|
+
|
|
336
|
+
let updated = false;
|
|
337
|
+
for (const entry of gitignoreEntries) {
|
|
338
|
+
if (!gitignore.includes(entry)) {
|
|
339
|
+
gitignore += `\n${entry}`;
|
|
340
|
+
updated = true;
|
|
341
|
+
}
|
|
342
|
+
}
|
|
343
|
+
|
|
344
|
+
if (updated) {
|
|
345
|
+
writeFileSync('.gitignore', gitignore);
|
|
346
|
+
console.log(' ✅ .gitignore updated');
|
|
347
|
+
} else {
|
|
348
|
+
console.log(' ℹ️ .gitignore already up to date');
|
|
349
|
+
}
|
|
350
|
+
|
|
351
|
+
// 完了メッセージ
|
|
352
|
+
console.log('\n');
|
|
353
|
+
console.log('🎉 セットアップ完了!');
|
|
354
|
+
console.log('');
|
|
355
|
+
console.log('次のステップ:');
|
|
356
|
+
console.log(' 1. .env ファイルを編集して認証情報を設定');
|
|
357
|
+
console.log(' 2. package.json が既存の場合、スクリプトを手動追加');
|
|
358
|
+
console.log(' 3. npm install で依存関係をインストール');
|
|
359
|
+
console.log(' 4. jj commit でセットアップをコミット');
|
|
360
|
+
console.log(' 5. Cursor で開く: cursor .');
|
|
361
|
+
console.log(' 6. /kiro:spec-init <機能説明> で開発開始');
|
|
362
|
+
console.log('');
|
|
363
|
+
console.log('作成されたファイル:');
|
|
364
|
+
console.log(' - .kiro/project.json');
|
|
365
|
+
console.log(' - .cursor/rules/ (3ファイル)');
|
|
366
|
+
console.log(' - .cursor/commands/kiro/ (2ファイル)');
|
|
367
|
+
console.log(' - .kiro/steering/ (3ファイル)');
|
|
368
|
+
console.log(' - .kiro/settings/templates/ (3ファイル)');
|
|
369
|
+
console.log(' - scripts/ (7+ ファイル)');
|
|
370
|
+
console.log(' - package.json (新規の場合)');
|
|
371
|
+
console.log(' - tsconfig.json (新規の場合)');
|
|
372
|
+
console.log(' - .env (テンプレート)');
|
|
373
|
+
}
|
|
374
|
+
|
|
375
|
+
// 実行
|
|
376
|
+
const config = parseArgs();
|
|
377
|
+
setupExistingProject(config).catch(error => {
|
|
378
|
+
console.error('❌ Error:', error.message);
|
|
379
|
+
process.exit(1);
|
|
380
|
+
});
|
|
381
|
+
|