Public Access
b491241a58
Universal: Sync Feature Branch Versions / Sync feature branches with dev (push) Has been skipped
# Conflicts: # .mokogitea/CLAUDE.md # .mokogitea/ISSUE_TEMPLATE/config.yml # .mokogitea/ISSUE_TEMPLATE/documentation.md # .mokogitea/ISSUE_TEMPLATE/feature_request.md # .mokogitea/ISSUE_TEMPLATE/security.md # .mokogitea/branch-protection.yml # .mokogitea/bulk-repo-sync.yml # .mokogitea/pr-branch-check.yml # .mokogitea/renovate.yml # .mokogitea/sync-wikis.yml # .mokogitea/workflows/auto-bump.yml # .mokogitea/workflows/auto-release.yml # .mokogitea/workflows/ci-platform.yml # .mokogitea/workflows/cleanup.yml # .mokogitea/workflows/gitleaks.yml # .mokogitea/workflows/issue-branch.yml # .mokogitea/workflows/notify.yml # .mokogitea/workflows/pre-release.yml # .mokogitea/workflows/repo-health.yml # .mokogitea/workflows/security-audit.yml # .script-registry.json # CHANGELOG.md # PLUGIN_SCRIPTS.md # README.md # analysis/index.md # automation/bulk_joomla_template.php # automation/bulk_sync.php # automation/enrich_manifest_xml.php # automation/enrich_mokostandards_xml.php # automation/index.md # automation/migrate_to_gitea.php # automation/push_files.php # automation/push_manifest_xml.php # automation/push_mokostandards_xml.php # automation/repo_cleanup.php # bin/moko # cli/archive_repo.php # cli/audit_query.php # cli/badge_update.php # cli/branch_rename.php # cli/bulk_workflow_push.php # cli/bulk_workflow_trigger.php # cli/changelog_promote.php # cli/changelog_prune.php # cli/client_dashboard.php # cli/client_health_check.php # cli/client_inventory.php # cli/client_provision.php # cli/completion.php # cli/create_project.php # cli/create_repo.php # cli/deploy_joomla.php # cli/dev_branch_reset.php # cli/grafana_dashboard.php # cli/joomla_build.php # cli/joomla_compat_check.php # cli/joomla_metadata_validate.php # cli/joomla_release.php # cli/license_manage.php # cli/manifest_element.php # cli/manifest_licensing.php # cli/manifest_read.php # cli/package_build.php # cli/platform_detect.php # cli/release.php # cli/release_body_update.php # cli/release_cascade.php # cli/release_create.php # cli/release_manage.php # cli/release_mirror.php # cli/release_notes.php # cli/release_package.php # cli/release_promote.php # cli/release_publish.php # cli/release_validate.php # cli/release_verify.php # cli/scaffold_client.php # cli/sync_rulesets.php # cli/theme_lint.php # cli/updates_xml_build.php # cli/updates_xml_sync.php # cli/version_auto_bump.php # cli/version_bump.php # cli/version_bump_remote.php # cli/version_check.php # cli/version_read.php # cli/version_reset_dev.php # cli/version_set_platform.php # cli/wiki_sync.php # cli/workflow_sync.php # composer.json # deploy/backup-before-deploy.php # deploy/deploy-dolibarr.php # deploy/deploy-joomla.php # deploy/deploy-sftp.php # deploy/health-check.php # deploy/rollback-joomla.php # deploy/sync-joomla.php # fix/fix_line_endings.php # fix/fix_permissions.php # fix/fix_tabs.php # fix/fix_trailing_spaces.php # fix/index.md # index.md # lib/CliBase.php # lib/Common.php # lib/Enterprise/AbstractProjectPlugin.php # lib/Enterprise/ApiClient.php # lib/Enterprise/AuditLogger.php # lib/Enterprise/CheckpointManager.php # lib/Enterprise/CliFramework.php # lib/Enterprise/Config.php # lib/Enterprise/ConfigValidator.php # lib/Enterprise/EnterpriseReadinessValidator.php # lib/Enterprise/ErrorRecovery.php # lib/Enterprise/FileFixUtility.php # lib/Enterprise/GitHubAdapter.php # lib/Enterprise/GitPlatformAdapter.php # lib/Enterprise/InputValidator.php # lib/Enterprise/ManifestParser.php # lib/Enterprise/ManifestReader.php # lib/Enterprise/MetricsCollector.php # lib/Enterprise/MokoGiteaAdapter.php # lib/Enterprise/PackageBuilder.php # lib/Enterprise/PlatformAdapterFactory.php # lib/Enterprise/PluginFactory.php # lib/Enterprise/PluginRegistry.php # lib/Enterprise/Plugins/ApiPlugin.php # lib/Enterprise/Plugins/DocumentationPlugin.php # lib/Enterprise/Plugins/DolibarrPlugin.php # lib/Enterprise/Plugins/GenericPlugin.php # lib/Enterprise/Plugins/JoomlaPlugin.php # lib/Enterprise/Plugins/McpServerPlugin.php # lib/Enterprise/Plugins/MobilePlugin.php # lib/Enterprise/Plugins/NodeJsPlugin.php # lib/Enterprise/Plugins/PythonPlugin.php # lib/Enterprise/Plugins/TerraformPlugin.php # lib/Enterprise/Plugins/WordPressPlugin.php # lib/Enterprise/ProjectConfigValidator.php # lib/Enterprise/ProjectMetricsCollector.php # lib/Enterprise/ProjectPluginInterface.php # lib/Enterprise/ProjectTypeDetector.php # lib/Enterprise/RecoveryError.php # lib/Enterprise/RecoveryManager.php # lib/Enterprise/RepositoryHealthChecker.php # lib/Enterprise/RepositorySynchronizer.php # lib/Enterprise/RetryHelper.php # lib/Enterprise/SecurityValidator.php # lib/Enterprise/SourceResolver.php # lib/Enterprise/SynchronizationException.php # lib/Enterprise/TransactionManager.php # lib/Enterprise/UnifiedValidation.php # lib/index.md # lib/plugins/Joomla/UpdateXmlGenerator.php # maintenance/index.md # maintenance/pin_action_shas.php # maintenance/repo_inventory.php # maintenance/rotate_secrets.php # maintenance/setup_labels.php # maintenance/sync_dolibarr_readmes.php # maintenance/update_repo_inventory.php # maintenance/update_sha_hashes.php # maintenance/update_version_from_readme.php # mcp/config.example.json # mcp/package.json # mcp/src/config.ts # mcp/src/index.ts # mcp/src/runner.ts # mcp/src/types.ts # phpcs.xml # plugin_health_check.php # plugin_list.php # plugin_metrics.php # plugin_readiness.php # plugin_validate.php # release/generate_dolibarr_version_txt.php # release/generate_joomla_update_xml.php # src/functions.php # templates/configs/README.md # templates/configs/index.md # templates/configs/manifest.xml.template # templates/configs/manifest.yml.template # templates/configs/mokostandards.xml.template # templates/configs/mokostandards.yml.template # templates/configs/phpcs.xml # templates/docs/README.md # templates/docs/extra/README.md # templates/docs/extra/index.md # templates/docs/index.md # templates/docs/required/GOVERNANCE.md # templates/docs/required/README.md # templates/docs/required/index.md # templates/docs/required/template-CONTRIBUTING.md # templates/docs/required/template-README.md # templates/docs/required/template-SECURITY.md # templates/index.md # templates/licenses/README.md # templates/licenses/index.md # templates/makefiles/README.md # templates/mokogitea/CLAUDE.dolibarr.md.template # templates/mokogitea/CLAUDE.joomla.md.template # templates/mokogitea/CLAUDE.md.template # templates/mokogitea/ISSUE_TEMPLATE/config.yml # templates/mokogitea/ISSUE_TEMPLATE/documentation.md # templates/mokogitea/ISSUE_TEMPLATE/dolibarr_module_id_request.md # templates/mokogitea/ISSUE_TEMPLATE/feature_request.md # templates/mokogitea/ISSUE_TEMPLATE/security.md # templates/mokogitea/README.md # templates/mokogitea/copilot-instructions.dolibarr.md.template # templates/mokogitea/copilot-instructions.joomla.md.template # templates/mokogitea/copilot-instructions.md.template # templates/mokogitea/dependabot.yml.template # templates/mokogitea/override.tf.template # templates/required/README.md # templates/schemas/README.md # templates/schemas/manifest-schema.xsd # templates/schemas/moko-platform-schema.xsd # templates/schemas/mokostandards-schema.xsd # templates/schemas/schemas/README.md # templates/schemas/template-repository-structure.xml # templates/scripts/README.md # templates/scripts/common/CliBase.template.php # templates/scripts/fix/index.md # templates/scripts/index.md # templates/scripts/release/index.md # templates/scripts/release/package_dolibarr.php # templates/scripts/release/package_joomla.php # templates/scripts/sftp-config/README.md # templates/scripts/validate/dolibarr_module.php # templates/scripts/validate/index.md # templates/scripts/validate/validate_manifest.php # templates/scripts/validate/validate_structure.php # templates/security/README.md # templates/security/index.php # templates/stubs/dolibarr.php # templates/stubs/joomla.php # templates/web/index.php # tests/Enterprise/GitPlatformAdapterTest.php # tests/Unit/VersionBumpTest.php # tests/Unit/VersionReadTest.php # tests/index.md # tests/test_circuit_breaker_handling.php # tests/test_enterprise_libraries.php # validate/SECURITY_SCANNING.md # validate/auto_detect_platform.php # validate/check_changelog.php # validate/check_client_theme.php # validate/check_composer_deps.php # validate/check_dolibarr_module.php # validate/check_enterprise_readiness.php # validate/check_file_integrity.php # validate/check_joomla_manifest.php # validate/check_language_structure.php # validate/check_license_headers.php # validate/check_no_secrets.php # validate/check_paths.php # validate/check_php_syntax.php # validate/check_repo_health.php # validate/check_structure.php # validate/check_tabs.php # validate/check_version_consistency.php # validate/check_wiki_health.php # validate/check_xml_wellformed.php # validate/index.md # validate/scan_drift.php # wrappers/auto_detect_platform.php # wrappers/bulk_sync.php # wrappers/check_changelog.php # wrappers/check_dolibarr_module.php # wrappers/check_enterprise_readiness.php # wrappers/check_joomla_manifest.php # wrappers/check_language_structure.php # wrappers/check_license_headers.php # wrappers/check_no_secrets.php # wrappers/check_paths.php # wrappers/check_php_syntax.php # wrappers/check_repo_health.php # wrappers/check_structure.php # wrappers/check_tabs.php # wrappers/check_version_consistency.php # wrappers/check_xml_wellformed.php # wrappers/deploy_sftp.php # wrappers/fix_line_endings.php # wrappers/fix_permissions.php # wrappers/fix_tabs.php # wrappers/fix_trailing_spaces.php # wrappers/gen_wrappers.php # wrappers/index.md # wrappers/pin_action_shas.php # wrappers/plugin_health_check.php # wrappers/plugin_list.php # wrappers/plugin_metrics.php # wrappers/plugin_readiness.php # wrappers/plugin_validate.php # wrappers/scan_drift.php # wrappers/setup_labels.php # wrappers/sync_dolibarr_readmes.php # wrappers/update_sha_hashes.php # wrappers/update_version_from_readme.php
466 lines
16 KiB
PHP
466 lines
16 KiB
PHP
#!/usr/bin/env php
|
|
<?php
|
|
|
|
/* Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
|
*
|
|
* This file is part of a Moko Consulting project.
|
|
*
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*
|
|
* FILE INFORMATION
|
|
<<<<<<< HEAD
|
|
* DEFGROUP: MokoCLI.CLI
|
|
* INGROUP: MokoCLI
|
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
|
=======
|
|
* DEFGROUP: mokoplatform.CLI
|
|
* INGROUP: mokoplatform
|
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokoplatform
|
|
>>>>>>> main
|
|
* PATH: /cli/create_project.php
|
|
* BRIEF: Create baseline GitHub Projects for repositories with standard fields and views
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
|
|
|
use MokoCli\CliFramework;
|
|
|
|
class CreateProjectCli extends CliFramework
|
|
{
|
|
/** @var string[] */
|
|
<<<<<<< HEAD
|
|
private array $ALWAYS_EXCLUDE = ['MokoCLI', '.github-private'];
|
|
=======
|
|
private array $ALWAYS_EXCLUDE = ['mokoplatform', '.github-private'];
|
|
>>>>>>> main
|
|
|
|
/** @var array<string, string> */
|
|
private array $PLATFORM_TO_TYPE = [
|
|
'crm-module' => 'dolibarr',
|
|
'crm-platform' => 'dolibarr',
|
|
'waas-component' => 'joomla',
|
|
'waas-library' => 'joomla',
|
|
'waas-plugin' => 'joomla',
|
|
'waas-package' => 'joomla',
|
|
'nodejs' => 'nodejs',
|
|
'terraform' => 'terraform',
|
|
'python' => 'python',
|
|
'wordpress' => 'wordpress',
|
|
'mobile' => 'mobile-app',
|
|
'api' => 'api',
|
|
'documentation' => 'documentation',
|
|
];
|
|
|
|
/** @var array<string, string> */
|
|
private array $TYPE_TO_TEMPLATE = [
|
|
'generic' => 'generic-project-definition.tf',
|
|
'dolibarr' => 'dolibarr-project-definition.tf',
|
|
'joomla' => 'joomla-project-definition.tf',
|
|
'nodejs' => 'nodejs-project-definition.tf',
|
|
'terraform' => 'terraform-project-definition.tf',
|
|
'python' => 'python-project-definition.tf',
|
|
'wordpress' => 'wordpress-project-definition.tf',
|
|
'mobile-app' => 'mobile-app-project-definition.tf',
|
|
'api' => 'api-project-definition.tf',
|
|
'documentation' => 'documentation-project-definition.tf',
|
|
];
|
|
|
|
protected function configure(): void
|
|
{
|
|
$this->setDescription('Create baseline GitHub Projects for repositories with standard fields and views');
|
|
$this->addArgument('--repo', 'Repository name', '');
|
|
$this->addArgument('--org', 'Organization (default: mokoconsulting-tech)', 'mokoconsulting-tech');
|
|
$this->addArgument('--type', 'Force project type', '');
|
|
$this->addArgument('--all', 'Process all repos without projects', false);
|
|
}
|
|
|
|
protected function run(): int
|
|
{
|
|
$repoName = $this->getArgument('--repo') ?: null;
|
|
$org = $this->getArgument('--org');
|
|
$typeOverride = $this->getArgument('--type') ?: null;
|
|
$allMode = $this->getArgument('--all');
|
|
|
|
if (!$repoName && !$allMode) {
|
|
$this->log('ERROR', "Usage: php create_project.php --repo <name> [--type <type>] [--dry-run]");
|
|
$this->log('ERROR', " php create_project.php --all [--org <org>] [--dry-run]");
|
|
$this->log('ERROR', "Types: generic, dolibarr, joomla, nodejs, terraform, python, wordpress, mobile-app, api, documentation");
|
|
return 2;
|
|
}
|
|
|
|
$config = \MokoCli\Config::load();
|
|
$platformName = $config->getString('platform', 'gitea');
|
|
try {
|
|
$adapter = \MokoCli\PlatformAdapterFactory::create($config);
|
|
$api = $adapter->getApiClient();
|
|
} catch (\Exception $e) {
|
|
$this->log('ERROR', "Platform initialization failed: " . $e->getMessage());
|
|
return 1;
|
|
}
|
|
$token = $platformName === 'gitea'
|
|
? $config->getString('gitea.token', '')
|
|
: $config->getString('github.token', '');
|
|
|
|
$repoRoot = dirname(__DIR__, 2);
|
|
$templatesDir = "{$repoRoot}/templates/projects";
|
|
|
|
$repos = [];
|
|
|
|
if ($allMode) {
|
|
echo "Fetching repositories from {$org}...\n";
|
|
$page = 1;
|
|
do {
|
|
$batch = $this->restGet("orgs/{$org}/repos?per_page=100&page={$page}&type=all", $token, $api);
|
|
foreach ($batch as $r) {
|
|
if (!$r['archived'] && !in_array($r['name'], $this->ALWAYS_EXCLUDE, true)) {
|
|
$repos[] = $r['name'];
|
|
}
|
|
}
|
|
$page++;
|
|
} while (count($batch) === 100);
|
|
|
|
sort($repos);
|
|
echo "Found " . count($repos) . " repositories\n\n";
|
|
} else {
|
|
$repos = [$repoName];
|
|
}
|
|
|
|
$ownerId = $this->getOrgNodeId($org, $token);
|
|
if (empty($ownerId)) {
|
|
$this->log('ERROR', "Could not resolve org node ID for {$org}");
|
|
return 1;
|
|
}
|
|
|
|
$created = 0;
|
|
$skipped = 0;
|
|
$failed = 0;
|
|
|
|
foreach ($repos as $repo) {
|
|
echo "Processing {$repo}...\n";
|
|
|
|
[$hasProject, $existingTitle] = $this->repoHasProject($org, $repo, $token);
|
|
if ($hasProject) {
|
|
echo " Already has project: {$existingTitle} -- skipping\n";
|
|
$skipped++;
|
|
continue;
|
|
}
|
|
|
|
$type = $typeOverride;
|
|
if (!$type) {
|
|
$platform = $this->detectRepoPlatform($org, $repo, $token, $api);
|
|
$type = $this->PLATFORM_TO_TYPE[$platform] ?? 'generic';
|
|
echo " Platform: {$platform} -> type: {$type}\n";
|
|
}
|
|
|
|
$templateFile = $this->TYPE_TO_TEMPLATE[$type] ?? $this->TYPE_TO_TEMPLATE['generic'];
|
|
$template = $this->parseTemplate("{$templatesDir}/{$templateFile}");
|
|
|
|
$repoId = $this->getRepoNodeId($org, $repo, $token);
|
|
if (empty($repoId)) {
|
|
$this->log('ERROR', " Could not resolve repo node ID for {$repo}");
|
|
$failed++;
|
|
continue;
|
|
}
|
|
|
|
$ok = $this->createProject($org, $repo, $ownerId, $repoId, $template, $token);
|
|
if ($ok) {
|
|
$created++;
|
|
} else {
|
|
$failed++;
|
|
}
|
|
|
|
echo "\n";
|
|
}
|
|
|
|
echo str_repeat('-', 50) . "\n";
|
|
echo "Done: {$created} created, {$skipped} skipped, {$failed} failed\n";
|
|
return $failed > 0 ? 1 : 0;
|
|
}
|
|
|
|
private function graphql(string $query, array $variables, string $token, string $platformName = 'gitea'): array
|
|
{
|
|
if ($platformName !== 'github') {
|
|
return [];
|
|
}
|
|
$payload = json_encode(['query' => $query, 'variables' => $variables]);
|
|
$ch = curl_init('https://api.github.com/graphql');
|
|
curl_setopt_array($ch, [
|
|
CURLOPT_RETURNTRANSFER => true,
|
|
CURLOPT_POST => true,
|
|
CURLOPT_POSTFIELDS => $payload,
|
|
CURLOPT_HTTPHEADER => [
|
|
'Authorization: bearer ' . $token,
|
|
'Content-Type: application/json',
|
|
<<<<<<< HEAD
|
|
'User-Agent: MokoCLI-CreateProject',
|
|
=======
|
|
'User-Agent: mokoplatform-CreateProject',
|
|
>>>>>>> main
|
|
],
|
|
]);
|
|
$body = (string) curl_exec($ch);
|
|
$status = (int) curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
|
curl_close($ch);
|
|
|
|
if ($status !== 200) {
|
|
$this->log('ERROR', "GraphQL request failed (HTTP {$status}): {$body}");
|
|
return [];
|
|
}
|
|
|
|
$data = json_decode($body, true) ?? [];
|
|
if (!empty($data['errors'])) {
|
|
foreach ($data['errors'] as $err) {
|
|
$this->log('ERROR', " GraphQL error: " . ($err['message'] ?? 'unknown'));
|
|
}
|
|
}
|
|
|
|
return $data['data'] ?? [];
|
|
}
|
|
|
|
private function restGet(string $path, string $token, ?\MokoCli\ApiClient $apiClient = null): array
|
|
{
|
|
if ($apiClient !== null) {
|
|
try {
|
|
return $apiClient->get("/{$path}");
|
|
} catch (\Exception $e) {
|
|
return [];
|
|
}
|
|
}
|
|
return [];
|
|
}
|
|
|
|
private function detectRepoPlatform(string $org, string $repo, string $token, ?\MokoCli\ApiClient $apiClient = null): string
|
|
{
|
|
foreach (['.github/.mokostandards', '.mokogitea/.mokostandards', '.mokostandards'] as $path) {
|
|
$data = $this->restGet("repos/{$org}/{$repo}/contents/{$path}", $token, $apiClient);
|
|
if (!empty($data['content'])) {
|
|
$content = base64_decode($data['content']);
|
|
if (preg_match('/^platform:\s*(.+)/m', $content, $m)) {
|
|
return trim($m[1], " \t\n\r\"'");
|
|
}
|
|
}
|
|
}
|
|
return '';
|
|
}
|
|
|
|
private function getRepoNodeId(string $org, string $repo, string $token): string
|
|
{
|
|
$data = $this->graphql(
|
|
'query($owner: String!, $name: String!) { repository(owner: $owner, name: $name) { id } }',
|
|
['owner' => $org, 'name' => $repo],
|
|
$token
|
|
);
|
|
return $data['repository']['id'] ?? '';
|
|
}
|
|
|
|
private function getOrgNodeId(string $org, string $token): string
|
|
{
|
|
$data = $this->graphql(
|
|
'query($login: String!) { organization(login: $login) { id } }',
|
|
['login' => $org],
|
|
$token
|
|
);
|
|
return $data['organization']['id'] ?? '';
|
|
}
|
|
|
|
/** @return array{bool, string} */
|
|
private function repoHasProject(string $org, string $repo, string $token): array
|
|
{
|
|
$data = $this->graphql(
|
|
'query($owner: String!, $name: String!) {
|
|
repository(owner: $owner, name: $name) {
|
|
projectsV2(first: 1) { nodes { id title } totalCount }
|
|
}
|
|
}',
|
|
['owner' => $org, 'name' => $repo],
|
|
$token
|
|
);
|
|
|
|
$count = $data['repository']['projectsV2']['totalCount'] ?? 0;
|
|
$title = $data['repository']['projectsV2']['nodes'][0]['title'] ?? '';
|
|
return [$count > 0, $title];
|
|
}
|
|
|
|
/** @return array{name: string, fields: array, views: array} */
|
|
private function parseTemplate(string $filePath): array
|
|
{
|
|
if (!file_exists($filePath)) {
|
|
return ['name' => 'Development Board', 'fields' => [], 'views' => []];
|
|
}
|
|
|
|
$content = file_get_contents($filePath);
|
|
$result = ['name' => 'Development Board', 'fields' => [], 'views' => []];
|
|
|
|
if (preg_match('/name\s*=\s*"([^"]+)"/', $content, $m)) {
|
|
$result['name'] = $m[1];
|
|
}
|
|
|
|
$fieldPattern = '/\{\s*name\s*=\s*"([^"]+)"\s*type\s*=\s*"([^"]+)"'
|
|
. '\s*description\s*=\s*"([^"]+)"'
|
|
. '(?:\s*options\s*=\s*\[([^\]]*)\])?\s*\}/s';
|
|
if (preg_match_all($fieldPattern, $content, $matches, PREG_SET_ORDER)) {
|
|
foreach ($matches as $match) {
|
|
$field = [
|
|
'name' => $match[1],
|
|
'type' => $match[2],
|
|
'description' => $match[3],
|
|
];
|
|
if (!empty($match[4])) {
|
|
$field['options'] = array_map(
|
|
fn($o) => trim($o, " \t\n\r\"'"),
|
|
explode(',', $match[4])
|
|
);
|
|
$field['options'] = array_filter($field['options']);
|
|
}
|
|
$result['fields'][] = $field;
|
|
}
|
|
}
|
|
|
|
return $result;
|
|
}
|
|
|
|
private function createProject(
|
|
string $org,
|
|
string $repo,
|
|
string $ownerId,
|
|
string $repoId,
|
|
array $template,
|
|
string $token
|
|
): bool {
|
|
$title = "{$repo} -- {$template['name']}";
|
|
|
|
if ($this->dryRun) {
|
|
echo " (dry-run) would create project: {$title}\n";
|
|
echo " (dry-run) fields: " . count($template['fields']) . "\n";
|
|
return true;
|
|
}
|
|
|
|
echo " Creating project: {$title}\n";
|
|
$data = $this->graphql(
|
|
'mutation($ownerId: ID!, $title: String!) {
|
|
createProjectV2(input: { ownerId: $ownerId, title: $title }) {
|
|
projectV2 { id number url }
|
|
}
|
|
}',
|
|
['ownerId' => $ownerId, 'title' => $title],
|
|
$token
|
|
);
|
|
|
|
$projectId = $data['createProjectV2']['projectV2']['id'] ?? '';
|
|
$projectUrl = $data['createProjectV2']['projectV2']['url'] ?? '';
|
|
|
|
if (empty($projectId)) {
|
|
$this->log('ERROR', " Failed to create project for {$repo}");
|
|
return false;
|
|
}
|
|
|
|
echo " Project created: {$projectUrl}\n";
|
|
|
|
$this->graphql(
|
|
'mutation($projectId: ID!, $repositoryId: ID!) {
|
|
linkProjectV2ToRepository(input: { projectId: $projectId, repositoryId: $repositoryId }) {
|
|
repository { id }
|
|
}
|
|
}',
|
|
['projectId' => $projectId, 'repositoryId' => $repoId],
|
|
$token
|
|
);
|
|
echo " Linked to {$org}/{$repo}\n";
|
|
|
|
$fieldCount = 0;
|
|
foreach ($template['fields'] as $field) {
|
|
$fieldType = match ($field['type']) {
|
|
'single_select' => 'SINGLE_SELECT',
|
|
'text' => 'TEXT',
|
|
'number' => 'NUMBER',
|
|
'date' => 'DATE',
|
|
'iteration' => 'ITERATION',
|
|
default => 'TEXT',
|
|
};
|
|
|
|
$vars = [
|
|
'projectId' => $projectId,
|
|
'name' => $field['name'],
|
|
'dataType' => $fieldType,
|
|
];
|
|
|
|
if ($fieldType === 'SINGLE_SELECT' && !empty($field['options'])) {
|
|
$optionInputs = array_map(
|
|
fn($o) => ['name' => $o, 'description' => '', 'color' => 'GRAY'],
|
|
$field['options']
|
|
);
|
|
$vars['singleSelectOptions'] = $optionInputs;
|
|
|
|
$this->graphql(
|
|
'mutation($projectId: ID!, $name: String!,'
|
|
. ' $dataType: ProjectV2CustomFieldType!,'
|
|
. ' $singleSelectOptions:'
|
|
. ' [ProjectV2SingleSelectFieldOptionInput!]) {
|
|
createProjectV2Field(input: {
|
|
projectId: $projectId,
|
|
dataType: $dataType,
|
|
name: $name,
|
|
singleSelectOptions: $singleSelectOptions
|
|
}) {
|
|
projectV2Field { ... on ProjectV2SingleSelectField { id name } }
|
|
}
|
|
}',
|
|
$vars,
|
|
$token
|
|
);
|
|
} else {
|
|
$this->graphql(
|
|
'mutation($projectId: ID!, $name: String!, $dataType: ProjectV2CustomFieldType!) {
|
|
createProjectV2Field(input: {
|
|
projectId: $projectId,
|
|
dataType: $dataType,
|
|
name: $name
|
|
}) {
|
|
projectV2Field { ... on ProjectV2Field { id name } }
|
|
}
|
|
}',
|
|
$vars,
|
|
$token
|
|
);
|
|
}
|
|
|
|
$fieldCount++;
|
|
}
|
|
|
|
echo " Created {$fieldCount} custom fields\n";
|
|
|
|
$this->graphql(
|
|
'mutation($projectId: ID!, $shortDescription: String!) {
|
|
updateProjectV2(input: {
|
|
projectId: $projectId,
|
|
shortDescription: $shortDescription,
|
|
<<<<<<< HEAD
|
|
readme: "Managed by MokoCLI. Run `php cli/create_project.php` to regenerate."
|
|
=======
|
|
readme: "Managed by mokoplatform. Run `php cli/create_project.php` to regenerate."
|
|
>>>>>>> main
|
|
}) {
|
|
projectV2 { id }
|
|
}
|
|
}',
|
|
[
|
|
'projectId' => $projectId,
|
|
<<<<<<< HEAD
|
|
'shortDescription' => "Standard project board for {$repo}. Auto-created by MokoCLI.",
|
|
=======
|
|
'shortDescription' => "Standard project board for {$repo}. Auto-created by mokoplatform.",
|
|
>>>>>>> main
|
|
],
|
|
$token
|
|
);
|
|
|
|
echo " Project setup complete\n";
|
|
return true;
|
|
}
|
|
}
|
|
|
|
$app = new CreateProjectCli();
|
|
exit($app->execute());
|