66e728b078
Platform: moko-platform CI / Gate 2: Unit Tests (8.1) (pull_request) Blocked by required conditions
Platform: moko-platform CI / Gate 2: Unit Tests (8.2) (pull_request) Blocked by required conditions
Platform: moko-platform CI / Gate 2: Unit Tests (8.1) (push) Blocked by required conditions
Platform: moko-platform CI / Gate 2: Unit Tests (8.3) (pull_request) Blocked by required conditions
Platform: moko-platform CI / Gate 2: Unit Tests (8.2) (push) Blocked by required conditions
Platform: moko-platform CI / Gate 3: Self-Health Check (pull_request) Blocked by required conditions
Platform: moko-platform CI / Gate 2: Unit Tests (8.3) (push) Blocked by required conditions
Platform: moko-platform CI / Gate 4: Governance (pull_request) Blocked by required conditions
Platform: moko-platform CI / Gate 3: Self-Health Check (push) Blocked by required conditions
Platform: moko-platform CI / Gate 5: Template Integrity (pull_request) Blocked by required conditions
Platform: moko-platform CI / Gate 4: Governance (push) Blocked by required conditions
Platform: moko-platform CI / CI Summary (pull_request) Blocked by required conditions
Platform: moko-platform CI / Gate 5: Template Integrity (push) Blocked by required conditions
Platform: moko-platform CI / CI Summary (push) Blocked by required conditions
Generic: Repo Health / Release configuration (push) Blocked by required conditions
Generic: Repo Health / Scripts governance (push) Blocked by required conditions
Generic: Repo Health / Repository health (push) Blocked by required conditions
Universal: PR Check / Build RC Package (pull_request) Blocked by required conditions
Generic: Repo Health / Release configuration (pull_request) Blocked by required conditions
Generic: Repo Health / Scripts governance (pull_request) Blocked by required conditions
Generic: Repo Health / Repository health (pull_request) Blocked by required conditions
Generic: Repo Health / Access control (push) Successful in 18s
Generic: Repo Health / Site Health (push) Has been skipped
Universal: PR Check / Branch Policy (pull_request) Successful in 3s
Universal: Auto Version Bump / Version Bump (push) Failing after 27s
Universal: Secret Scanning / Gitleaks Secret Scan (pull_request) Successful in 28s
Universal: PR Check / Validate PR (pull_request) Failing after 6s
Generic: Repo Health / Site Health (pull_request) Has been skipped
Generic: Repo Health / Access control (pull_request) Successful in 3s
Platform: moko-platform CI / Gate 1: Code Quality (pull_request) Failing after 1m7s
Platform: moko-platform CI / Gate 1: Code Quality (push) Failing after 1m7s
Auto-fixed 5006 tab-indent and line-ending errors via phpcbf, then manually broke 100 lines exceeding 150-char limit. All 74 files in cli/, automation/, maintenance/, deploy/ now pass PHPCS PSR-12 clean. Authored-by: Moko Consulting Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
176 lines
8.2 KiB
PHP
176 lines
8.2 KiB
PHP
#!/usr/bin/env php
|
|
<?php
|
|
|
|
/* Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
|
*
|
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
|
*
|
|
* FILE INFORMATION
|
|
* DEFGROUP: moko-platform.CLI
|
|
* INGROUP: moko-platform
|
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
|
* PATH: /cli/release_manage.php
|
|
* BRIEF: Create/update Gitea releases, upload assets, update release body
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
|
|
|
use MokoEnterprise\CliFramework;
|
|
|
|
class ReleaseManageCli extends CliFramework
|
|
{
|
|
protected function configure(): void
|
|
{
|
|
$this->setDescription('Create/update Gitea releases, upload assets, update release body');
|
|
$this->addArgument('--action', 'create | upload | update-body | delete', null);
|
|
$this->addArgument('--tag', 'Release tag name', null);
|
|
$this->addArgument('--name', 'Release name/title', null);
|
|
$this->addArgument('--body', 'Release body/description', null);
|
|
$this->addArgument('--body-file', 'Read body from file', null);
|
|
$this->addArgument('--target', 'Target branch/commitish', 'main');
|
|
$this->addArgument('--files', 'Comma-separated file paths to upload', null);
|
|
$this->addArgument('--token', 'Gitea API token', null);
|
|
$this->addArgument('--api-base', 'Gitea API base URL', null);
|
|
}
|
|
|
|
protected function run(): int
|
|
{
|
|
$action = $this->getArgument('--action');
|
|
$tag = $this->getArgument('--tag');
|
|
$name = $this->getArgument('--name');
|
|
$body = $this->getArgument('--body');
|
|
$bodyFile = $this->getArgument('--body-file');
|
|
$target = $this->getArgument('--target');
|
|
$filesArg = $this->getArgument('--files');
|
|
$token = $this->getArgument('--token');
|
|
$apiBase = $this->getArgument('--api-base');
|
|
$files = $filesArg !== null ? array_filter(explode(',', $filesArg)) : [];
|
|
if ($token === null) {
|
|
$token = getenv('MOKOGITEA_TOKEN') ?: getenv('GITEA_TOKEN') ?: null;
|
|
}
|
|
if ($bodyFile !== null && file_exists($bodyFile)) {
|
|
$body = file_get_contents($bodyFile);
|
|
}
|
|
if ($action === null || $tag === null || $token === null || $apiBase === null) {
|
|
$this->log('ERROR', "Usage: release_manage.php --action [create|upload|update-body|delete] --tag TAG --token TOKEN --api-base URL");
|
|
return 1;
|
|
}
|
|
switch ($action) {
|
|
case 'create':
|
|
$existing = $this->getReleaseByTag($apiBase, $tag, $token);
|
|
if ($existing !== null) {
|
|
$existingId = $existing['id'];
|
|
$this->releaseGiteaApi("{$apiBase}/releases/{$existingId}", 'DELETE', $token);
|
|
$this->releaseGiteaApi("{$apiBase}/tags/{$tag}", 'DELETE', $token);
|
|
echo "Deleted previous release: {$tag} (id: {$existingId})\n";
|
|
}
|
|
$payload = json_encode(['tag_name' => $tag, 'name' => $name ?? $tag, 'body' => $body ?? '', 'target_commitish' => $target]);
|
|
$result = $this->releaseGiteaApi("{$apiBase}/releases", 'POST', $token, $payload);
|
|
if ($result['code'] >= 200 && $result['code'] < 300) {
|
|
$releaseId = $result['data']['id'] ?? 'unknown';
|
|
echo "Release created: {$name} (tag: {$tag}, id: {$releaseId})\n";
|
|
} else {
|
|
$this->log('ERROR', "Failed to create release: HTTP {$result['code']}");
|
|
return 1;
|
|
}
|
|
break;
|
|
case 'upload':
|
|
if (empty($files)) {
|
|
$this->log('ERROR', "No files specified. Use --files /path/to/file1,/path/to/file2");
|
|
return 1;
|
|
}
|
|
$release = $this->getReleaseByTag($apiBase, $tag, $token);
|
|
if ($release === null) {
|
|
$this->log('ERROR', "No release found for tag: {$tag}");
|
|
return 1;
|
|
}
|
|
$releaseId = $release['id'];
|
|
$assetsResult = $this->releaseGiteaApi("{$apiBase}/releases/{$releaseId}/assets", 'GET', $token);
|
|
$existingAssets = $assetsResult['data'] ?? [];
|
|
foreach ($files as $filePath) {
|
|
$filePath = trim($filePath);
|
|
if (!file_exists($filePath)) {
|
|
$this->log('ERROR', "File not found: {$filePath}");
|
|
continue;
|
|
}
|
|
$fileName = basename($filePath);
|
|
foreach ($existingAssets as $asset) {
|
|
if (($asset['name'] ?? '') === $fileName) {
|
|
$this->releaseGiteaApi("{$apiBase}/releases/{$releaseId}/assets/{$asset['id']}", 'DELETE', $token);
|
|
echo "Deleted existing asset: {$fileName}\n";
|
|
break;
|
|
}
|
|
}
|
|
$uploadUrl = "{$apiBase}/releases/{$releaseId}/assets?name=" . urlencode($fileName);
|
|
$result = $this->releaseGiteaApi($uploadUrl, 'POST', $token, null, $filePath);
|
|
if ($result['code'] >= 200 && $result['code'] < 300) {
|
|
echo "Uploaded: {$fileName}\n";
|
|
} else {
|
|
$this->log('ERROR', "Failed to upload {$fileName}: HTTP {$result['code']}");
|
|
}
|
|
}
|
|
break;
|
|
case 'update-body':
|
|
$release = $this->getReleaseByTag($apiBase, $tag, $token);
|
|
if ($release === null) {
|
|
$this->log('ERROR', "No release found for tag: {$tag}");
|
|
return 1;
|
|
}
|
|
$payload = json_encode(['body' => $body ?? '']);
|
|
$result = $this->releaseGiteaApi("{$apiBase}/releases/{$release['id']}", 'PATCH', $token, $payload);
|
|
if ($result['code'] >= 200 && $result['code'] < 300) {
|
|
echo "Release body updated for tag: {$tag}\n";
|
|
} else {
|
|
$this->log('ERROR', "Failed to update body: HTTP {$result['code']}");
|
|
return 1;
|
|
}
|
|
break;
|
|
case 'delete':
|
|
$existing = $this->getReleaseByTag($apiBase, $tag, $token);
|
|
if ($existing !== null) {
|
|
$this->releaseGiteaApi("{$apiBase}/releases/{$existing['id']}", 'DELETE', $token);
|
|
$this->releaseGiteaApi("{$apiBase}/tags/{$tag}", 'DELETE', $token);
|
|
echo "Deleted: {$tag} (id: {$existing['id']})\n";
|
|
} else {
|
|
echo "No release found for tag: {$tag}\n";
|
|
}
|
|
break;
|
|
default:
|
|
$this->log('ERROR', "Unknown action: {$action}");
|
|
return 1;
|
|
}
|
|
return 0;
|
|
}
|
|
|
|
private function releaseGiteaApi(string $url, string $method, string $token, ?string $jsonBody = null, ?string $filePath = null): array
|
|
{
|
|
$ch = curl_init($url);
|
|
$headers = ["Authorization: token {$token}"];
|
|
$opts = [CURLOPT_RETURNTRANSFER => true, CURLOPT_TIMEOUT => 60, CURLOPT_CUSTOMREQUEST => $method];
|
|
if ($jsonBody !== null) {
|
|
$headers[] = 'Content-Type: application/json';
|
|
$opts[CURLOPT_POSTFIELDS] = $jsonBody;
|
|
} elseif ($filePath !== null) {
|
|
$headers[] = 'Content-Type: application/octet-stream';
|
|
$opts[CURLOPT_POSTFIELDS] = file_get_contents($filePath);
|
|
}
|
|
$opts[CURLOPT_HTTPHEADER] = $headers;
|
|
curl_setopt_array($ch, $opts);
|
|
$response = curl_exec($ch);
|
|
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
|
curl_close($ch);
|
|
return ['code' => $httpCode, 'data' => json_decode($response ?: '{}', true) ?: []];
|
|
}
|
|
|
|
private function getReleaseByTag(string $apiBase, string $tag, string $token): ?array
|
|
{
|
|
$result = $this->releaseGiteaApi("{$apiBase}/releases/tags/{$tag}", 'GET', $token);
|
|
return ($result['code'] === 200 && isset($result['data']['id'])) ? $result['data'] : null;
|
|
}
|
|
}
|
|
|
|
$app = new ReleaseManageCli();
|
|
exit($app->execute());
|