07ea171af9
Platform: moko-platform CI / Gate 2: Unit Tests (8.1) (push) Blocked by required conditions
Platform: moko-platform CI / Gate 2: Unit Tests (8.2) (push) Blocked by required conditions
Platform: moko-platform CI / Gate 2: Unit Tests (8.3) (push) Blocked by required conditions
Platform: moko-platform CI / Gate 3: Self-Health Check (push) Blocked by required conditions
Platform: moko-platform CI / Gate 4: Governance (push) 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
Generic: Repo Health / Site Health (push) Has been skipped
Generic: Repo Health / Access control (push) Successful in 1s
Platform: moko-platform CI / Gate 1: Code Quality (push) Failing after 43s
New CLI tools: - manifest_element.php — extract element/type/prefix from any platform manifest - release_create.php — create/overwrite Gitea releases with proper naming - release_package.php — build ZIP+tar.gz, SHA-256, upload assets - release_promote.php — promote releases between channels (dev→RC→stable) - version_reset_dev.php — reset platform version on dev branch after release Updated CLI tools: - version_bump.php — now writes to manifests, Dolibarr mod, composer.json (not just README) - release_cascade.php — added --version for version-aware deletion of stale releases - release_validate.php — auto-detect platform, --github-output, source dir check Workflow changes (auto-release.yml): - Draft PR to main → auto-promote highest pre-release to RC - Merged PR to main → promote RC to stable (skip rebuild when RC exists) - Removed paths filter for Go/Node/generic repo compatibility - Fixed cascade --api-base parameter bug Workflow changes (pre-release.yml): - Auto-trigger development pre-release on feature branch merge to dev - Removed paths filter Infrastructure: - RepositorySynchronizer: fixed template repo names, .mokogitea/workflows path, universal workflow cascade (Template-Generic → other templates) - bulk_sync.php: syncs universal workflows to templates before repo sync - PHPDoc added to 4 classes missing class-level docs - Version bump 09.00.00 → 09.01.00 Closes #152 #153 #154 #155 #156 #157 #158 #159 #161 #162 Authored-by: Moko Consulting Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
329 lines
11 KiB
PHP
329 lines
11 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_create.php
|
|
* BRIEF: Create or overwrite a Gitea release with proper naming
|
|
*
|
|
* Usage:
|
|
* php release_create.php --version 09.01.00 --tag stable --token TOKEN --api-base URL
|
|
* php release_create.php --version 09.01.00 --tag development --token TOKEN --api-base URL --prerelease
|
|
* php release_create.php --version 09.01.00 --tag stable --token TOKEN --api-base URL --path . --repo MyRepo
|
|
*
|
|
* Replaces the inline bash in auto-release.yml Step 7b.
|
|
* Detects extension metadata from manifest, builds a proper release name,
|
|
* generates release notes, and creates (or overwrites) a Gitea release.
|
|
*/
|
|
|
|
declare(strict_types=1);
|
|
|
|
// ── Argument parsing ────────────────────────────────────────────────────────
|
|
|
|
$path = '.';
|
|
$version = null;
|
|
$tag = null;
|
|
$token = null;
|
|
$apiBase = null;
|
|
$branch = 'main';
|
|
$repoName = '';
|
|
$prerelease = false;
|
|
|
|
foreach ($argv as $i => $arg) {
|
|
if ($arg === '--path' && isset($argv[$i + 1])) {
|
|
$path = $argv[$i + 1];
|
|
}
|
|
if ($arg === '--version' && isset($argv[$i + 1])) {
|
|
$version = $argv[$i + 1];
|
|
}
|
|
if ($arg === '--tag' && isset($argv[$i + 1])) {
|
|
$tag = $argv[$i + 1];
|
|
}
|
|
if ($arg === '--token' && isset($argv[$i + 1])) {
|
|
$token = $argv[$i + 1];
|
|
}
|
|
if ($arg === '--api-base' && isset($argv[$i + 1])) {
|
|
$apiBase = $argv[$i + 1];
|
|
}
|
|
if ($arg === '--branch' && isset($argv[$i + 1])) {
|
|
$branch = $argv[$i + 1];
|
|
}
|
|
if ($arg === '--repo' && isset($argv[$i + 1])) {
|
|
$repoName = $argv[$i + 1];
|
|
}
|
|
if ($arg === '--prerelease') {
|
|
$prerelease = true;
|
|
}
|
|
}
|
|
|
|
// Allow token from environment
|
|
if ($token === null) {
|
|
$envToken = getenv('GA_TOKEN');
|
|
if ($envToken === false || $envToken === '') {
|
|
$envToken = getenv('GITEA_TOKEN');
|
|
}
|
|
if ($envToken !== false && $envToken !== '') {
|
|
$token = $envToken;
|
|
}
|
|
}
|
|
|
|
if ($version === null || $tag === null || $token === null || $apiBase === null) {
|
|
fwrite(STDERR, "Usage: release_create.php --version VER --tag TAG --token TOKEN --api-base URL [options]\n");
|
|
fwrite(STDERR, " --path . Repo root for manifest detection (default: .)\n");
|
|
fwrite(STDERR, " --branch main Target commitish (default: main)\n");
|
|
fwrite(STDERR, " --repo REPO Repo name for fallback element detection\n");
|
|
fwrite(STDERR, " --prerelease Mark release as prerelease\n");
|
|
fwrite(STDERR, " Token can also be set via GA_TOKEN or GITEA_TOKEN env var\n");
|
|
exit(1);
|
|
}
|
|
|
|
// ── Helper: Gitea API request ───────────────────────────────────────────────
|
|
|
|
/**
|
|
* Send a request to the Gitea API.
|
|
*
|
|
* @param string $url Full API URL
|
|
* @param string $token Authorization token
|
|
* @param string $method HTTP method (GET, POST, DELETE, etc.)
|
|
* @param string|null $body JSON request body
|
|
*
|
|
* @return array<string, mixed>|null Decoded response or null on failure
|
|
*/
|
|
function giteaApi(string $url, string $token, string $method = 'GET', ?string $body = null): ?array
|
|
{
|
|
$ch = curl_init($url);
|
|
if ($ch === false) {
|
|
return null;
|
|
}
|
|
curl_setopt_array($ch, [
|
|
CURLOPT_RETURNTRANSFER => true,
|
|
CURLOPT_HTTPHEADER => [
|
|
"Authorization: token {$token}",
|
|
'Content-Type: application/json',
|
|
],
|
|
CURLOPT_TIMEOUT => 30,
|
|
CURLOPT_CUSTOMREQUEST => $method,
|
|
]);
|
|
if ($body !== null) {
|
|
curl_setopt($ch, CURLOPT_POSTFIELDS, $body);
|
|
}
|
|
$response = curl_exec($ch);
|
|
$httpCode = curl_getinfo($ch, CURLINFO_HTTP_CODE);
|
|
curl_close($ch);
|
|
|
|
if ($httpCode < 200 || $httpCode >= 300 || empty($response) || !is_string($response)) {
|
|
return null;
|
|
}
|
|
|
|
$decoded = json_decode($response, true);
|
|
return is_array($decoded) ? $decoded : null;
|
|
}
|
|
|
|
// ── Detect element metadata ─────────────────────────────────────────────────
|
|
|
|
$root = realpath($path) ?: $path;
|
|
|
|
$extElement = '';
|
|
$extType = '';
|
|
$extFolder = '';
|
|
$extName = '';
|
|
$typePrefix = '';
|
|
|
|
// Detect platform from manifest.xml
|
|
$platform = 'generic';
|
|
$manifestXml = "{$root}/.mokogitea/manifest.xml";
|
|
if (file_exists($manifestXml)) {
|
|
$content = file_get_contents($manifestXml);
|
|
if ($content !== false && preg_match('/<platform>([^<]+)<\/platform>/', $content, $pm)) {
|
|
$platform = trim($pm[1]);
|
|
}
|
|
}
|
|
|
|
// Find extension manifest (Joomla XML)
|
|
$extManifest = null;
|
|
$manifestFiles = array_merge(
|
|
glob("{$root}/src/pkg_*.xml") ?: [],
|
|
glob("{$root}/src/*.xml") ?: [],
|
|
glob("{$root}/*.xml") ?: []
|
|
);
|
|
foreach ($manifestFiles as $file) {
|
|
$c = file_get_contents($file);
|
|
if ($c !== false && strpos($c, '<extension') !== false) {
|
|
$extManifest = $file;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Find Dolibarr module file
|
|
$modFile = null;
|
|
$modFiles = array_merge(
|
|
glob("{$root}/src/core/modules/mod*.class.php") ?: [],
|
|
glob("{$root}/htdocs/core/modules/mod*.class.php") ?: [],
|
|
glob("{$root}/core/modules/mod*.class.php") ?: []
|
|
);
|
|
foreach ($modFiles as $file) {
|
|
$c = file_get_contents($file);
|
|
if ($c !== false && strpos($c, 'extends DolibarrModules') !== false) {
|
|
$modFile = $file;
|
|
break;
|
|
}
|
|
}
|
|
|
|
// Extract metadata based on platform
|
|
switch (true) {
|
|
case in_array($platform, ['joomla', 'waas-component'], true) && $extManifest !== null:
|
|
$xml = file_get_contents($extManifest);
|
|
if ($xml === false) {
|
|
break;
|
|
}
|
|
|
|
if (preg_match('/type="([^"]*)"/', $xml, $tm)) {
|
|
$extType = $tm[1];
|
|
}
|
|
if (preg_match('/group="([^"]*)"/', $xml, $gm)) {
|
|
$extFolder = $gm[1];
|
|
}
|
|
|
|
// Element name: <element>, plugin= attribute, <packagename>, or filename
|
|
if (preg_match('/<element>([^<]+)<\/element>/', $xml, $em)) {
|
|
$extElement = $em[1];
|
|
}
|
|
if (empty($extElement) && preg_match('/plugin="([^"]*)"/', $xml, $pm2)) {
|
|
$extElement = $pm2[1];
|
|
}
|
|
if ($extType === 'package' && preg_match('/<packagename>([^<]+)<\/packagename>/', $xml, $pn)) {
|
|
$extElement = $pn[1];
|
|
}
|
|
if (empty($extElement)) {
|
|
$extElement = strtolower(basename($extManifest, '.xml'));
|
|
if (in_array($extElement, ['templatedetails', 'manifest'], true)) {
|
|
$extElement = strtolower(str_replace([' ', '-'], '', $repoName !== '' ? $repoName : basename($root)));
|
|
}
|
|
}
|
|
|
|
// Human-readable name
|
|
if (preg_match('/<name>([^<]+)<\/name>/', $xml, $nm)) {
|
|
$extName = trim($nm[1]);
|
|
}
|
|
break;
|
|
|
|
case in_array($platform, ['dolibarr', 'crm-module'], true) && $modFile !== null:
|
|
$extType = 'dolibarr-module';
|
|
$modBasename = basename($modFile, '.class.php');
|
|
$extElement = strtolower(preg_replace('/^mod/', '', $modBasename) ?? $modBasename);
|
|
|
|
$modContent = file_get_contents($modFile);
|
|
if ($modContent !== false && preg_match('/\$this->name\s*=\s*[\'"]([^\'"]+)[\'"]/', $modContent, $nm2)) {
|
|
$extName = $nm2[1];
|
|
}
|
|
break;
|
|
|
|
default:
|
|
$extElement = strtolower(str_replace([' ', '-'], '', $repoName !== '' ? $repoName : basename($root)));
|
|
$extType = 'generic';
|
|
break;
|
|
}
|
|
|
|
// Strip existing type prefix from element to prevent duplication
|
|
$extElement = preg_replace('/^(pkg_|com_|mod_|plg_[a-z]+_|tpl_|lib_)/', '', $extElement) ?? $extElement;
|
|
|
|
// Compute type prefix
|
|
switch ($extType) {
|
|
case 'plugin':
|
|
$typePrefix = "plg_{$extFolder}_";
|
|
break;
|
|
case 'module':
|
|
$typePrefix = 'mod_';
|
|
break;
|
|
case 'component':
|
|
$typePrefix = 'com_';
|
|
break;
|
|
case 'template':
|
|
$typePrefix = 'tpl_';
|
|
break;
|
|
case 'library':
|
|
$typePrefix = 'lib_';
|
|
break;
|
|
case 'package':
|
|
$typePrefix = 'pkg_';
|
|
break;
|
|
}
|
|
|
|
// Fallback name
|
|
if (empty($extName)) {
|
|
$extName = $repoName !== '' ? $repoName : basename($root);
|
|
}
|
|
|
|
echo "Element: {$extElement}, Type: {$extType}, Prefix: {$typePrefix}, Name: {$extName}\n";
|
|
|
|
// ── Build release name ──────────────────────────────────────────────────────
|
|
|
|
$releaseName = "{$extName} {$version} ({$typePrefix}{$extElement}-{$version})";
|
|
echo "Release name: {$releaseName}\n";
|
|
|
|
// ── Generate release notes ──────────────────────────────────────────────────
|
|
|
|
$releaseNotes = "Release {$version}";
|
|
$releaseNotesScript = dirname(__DIR__) . '/cli/release_notes.php';
|
|
if (file_exists($releaseNotesScript)) {
|
|
$cmd = sprintf(
|
|
'php %s --path %s --version %s',
|
|
escapeshellarg($releaseNotesScript),
|
|
escapeshellarg($root),
|
|
escapeshellarg($version)
|
|
);
|
|
$output = [];
|
|
$exitCode = 0;
|
|
exec($cmd, $output, $exitCode);
|
|
if ($exitCode === 0 && count($output) > 0) {
|
|
$notes = implode("\n", $output);
|
|
if (trim($notes) !== '') {
|
|
$releaseNotes = $notes;
|
|
echo "Release notes: generated from CHANGELOG.md\n";
|
|
}
|
|
}
|
|
}
|
|
|
|
// ── Delete existing release at tag (if present) ─────────────────────────────
|
|
|
|
$existing = giteaApi("{$apiBase}/releases/tags/{$tag}", $token);
|
|
if ($existing !== null && !empty($existing['id'])) {
|
|
$existingId = $existing['id'];
|
|
echo "Deleting existing release: {$tag} (id: {$existingId})\n";
|
|
|
|
// Delete release
|
|
giteaApi("{$apiBase}/releases/{$existingId}", $token, 'DELETE');
|
|
|
|
// Delete tag
|
|
giteaApi("{$apiBase}/tags/{$tag}", $token, 'DELETE');
|
|
}
|
|
|
|
// ── Create new release ──────────────────────────────────────────────────────
|
|
|
|
$payload = json_encode([
|
|
'tag_name' => $tag,
|
|
'target_commitish' => $branch,
|
|
'name' => $releaseName,
|
|
'body' => $releaseNotes,
|
|
'prerelease' => $prerelease,
|
|
]);
|
|
|
|
$newRelease = giteaApi("{$apiBase}/releases", $token, 'POST', $payload !== false ? $payload : '{}');
|
|
if ($newRelease === null || empty($newRelease['id'])) {
|
|
fwrite(STDERR, "Failed to create release at tag: {$tag}\n");
|
|
exit(1);
|
|
}
|
|
|
|
$releaseId = $newRelease['id'];
|
|
echo "Created release: {$tag} (id: {$releaseId})\n";
|
|
|
|
// Output release_id to stdout for CI consumption
|
|
echo "release_id={$releaseId}\n";
|
|
exit(0);
|