Public Access
Compare commits
49 Commits
release-candidate
...
dev
| Author | SHA1 | Date | |
|---|---|---|---|
| 8822469761 | |||
| 908e839c30 | |||
| d5d9c4afce | |||
| ba6003a431 | |||
| f918bccf12 | |||
| 8741964d87 | |||
| c70481b69d | |||
| a5a16fb7e3 | |||
| 1274e2866b | |||
| 314bf91913 | |||
| 5c05ac62d0 | |||
| d206dd5afb | |||
| 50260b5cb5 | |||
| 4eb421d6ba | |||
| df83f36436 | |||
| e4de039cff | |||
| a6338493aa | |||
| 1b113af068 | |||
| a51f0bfb2f | |||
| c7b6f98f93 | |||
| 2dc43de160 | |||
| ea760bb75b | |||
| d065eaf0fd | |||
| e4d9bce5d0 | |||
| e933e7b651 | |||
| 157e87279e | |||
| 7850721f86 | |||
| 8949f69699 | |||
| af2313d936 | |||
| 2e5446ff5e | |||
| ab05bb7008 | |||
| 6bd26698c4 | |||
| 19b504526b | |||
| e7bdf7cbc7 | |||
| ff5794d0cc | |||
| bfba45e8b5 | |||
| 78ea05233b | |||
| ae0d54310d | |||
| 9df59836bf | |||
| 6e40707223 | |||
| ca55e5d2d2 | |||
| 9526d006c4 | |||
| c90a5671bd | |||
| 048a7d71d1 | |||
| c57b5724ac | |||
| 78affd37ff | |||
| b3062c6559 | |||
| 9dab9f1ef6 | |||
| c61d32709c |
@@ -0,0 +1,76 @@
|
|||||||
|
# MokoCLI
|
||||||
|
|
||||||
|
Enterprise automation, validation, sync, and governance engine for all Moko Consulting repositories.
|
||||||
|
|
||||||
|
## Quick Reference
|
||||||
|
|
||||||
|
| Field | Value |
|
||||||
|
|---|---|
|
||||||
|
| **Language** | PHP 8.1+ |
|
||||||
|
| **Version** | 09.01.00 |
|
||||||
|
| **Branch** | develop on `dev`, merge to `main` (protected) |
|
||||||
|
| **Wiki** | [MokoCLI Wiki](https://git.mokoconsulting.tech/MokoConsulting/mokocli/wiki) |
|
||||||
|
|
||||||
|
## Commands
|
||||||
|
|
||||||
|
```bash
|
||||||
|
composer install # Install PHP dependencies
|
||||||
|
php bin/moko health --path . # Repo health check
|
||||||
|
php bin/moko check:syntax --path . # PHP syntax check
|
||||||
|
php bin/moko drift --org MokoConsulting # Scan for standards drift
|
||||||
|
php bin/moko dashboard --token $TOKEN -o dashboard.html # Client dashboard
|
||||||
|
|
||||||
|
# Code quality
|
||||||
|
php vendor/bin/phpcs --standard=phpcs.xml -n lib/ validate/ automation/ cli/
|
||||||
|
php vendor/bin/phpcbf --standard=phpcs.xml lib/ validate/ automation/ cli/
|
||||||
|
php vendor/bin/phpstan analyse -c phpstan.neon --memory-limit=512M
|
||||||
|
composer check # Run all checks
|
||||||
|
```
|
||||||
|
|
||||||
|
## Architecture
|
||||||
|
|
||||||
|
| Directory | Purpose |
|
||||||
|
|---|---|
|
||||||
|
| `cli/` | 32 standalone CLI tools (version, release, build, repo management) |
|
||||||
|
| `validate/` | 20 validation scripts (syntax, structure, manifests, drift) |
|
||||||
|
| `automation/` | 7 bulk operations (sync, push files, templates, cleanup) |
|
||||||
|
| `lib/Enterprise/` | Core library — CliFramework, ApiClient, adapters, validators |
|
||||||
|
| `lib/Enterprise/Plugins/` | 11 platform plugins (Joomla, Dolibarr, Node.js, Python, etc.) |
|
||||||
|
| `deploy/` | SFTP deployment scripts (Joomla, Dolibarr, health checks) |
|
||||||
|
| `templates/` | Universal templates, configs, governance schema |
|
||||||
|
| `.mokogitea/workflows/` | CI/CD workflows (Gitea Actions) |
|
||||||
|
| `bin/moko` | Unified CLI dispatcher — `php bin/moko <command>` |
|
||||||
|
| `monitoring/sites.json` | Sites list for mcp_mokomonitor |
|
||||||
|
|
||||||
|
### CLI Framework
|
||||||
|
|
||||||
|
All CLI tools extend `MokoEnterprise\CliFramework` (`lib/Enterprise/CliFramework.php`).
|
||||||
|
Built-in flags: `--help`, `--verbose`, `--quiet`, `--dry-run`.
|
||||||
|
After adding a CLI tool, register it in `bin/moko` COMMAND_MAP.
|
||||||
|
|
||||||
|
### Platform Adapters
|
||||||
|
|
||||||
|
- `MokoGiteaAdapter` — git.mokoconsulting.tech (primary)
|
||||||
|
- `GitHubAdapter` — github.com mirrors
|
||||||
|
|
||||||
|
### Plugin System
|
||||||
|
|
||||||
|
Platform-specific logic in `lib/Enterprise/Plugins/`. Each implements `ProjectPluginInterface` with health checks, validation, build commands, config schemas.
|
||||||
|
|
||||||
|
## Code Quality
|
||||||
|
|
||||||
|
| Tool | Level | Config |
|
||||||
|
|---|---|---|
|
||||||
|
| PHPCS | PSR-12 (errors only) | `phpcs.xml` |
|
||||||
|
| PHPStan | Level 2 (advisory) | `phpstan.neon` |
|
||||||
|
|
||||||
|
PHPStan runs with `--memory-limit=512M`. CI enforces PHPCS errors; PHPStan is `continue-on-error`.
|
||||||
|
|
||||||
|
## Rules
|
||||||
|
|
||||||
|
- **Never commit** `.claude/`, `.mcp.json`, `TODO.md`, `*.min.css`/`*.min.js`
|
||||||
|
- **Attribution**: `Authored-by: Moko Consulting`
|
||||||
|
- **Workflow directory**: `.mokogitea/` (not `.gitea/` or `.github/`)
|
||||||
|
- **Wiki**: documentation lives in the Gitea wiki, not `docs/` files
|
||||||
|
- **New CLI tools**: extend `CliFramework`, not `CLIApp` (legacy)
|
||||||
|
- **Standards**: [MokoStandards](https://git.mokoconsulting.tech/MokoConsulting/mokocli/wiki/Home)
|
||||||
@@ -7,8 +7,8 @@ contact_links:
|
|||||||
- name: 💬 Ask a Question
|
- name: 💬 Ask a Question
|
||||||
url: https://mokoconsulting.tech/
|
url: https://mokoconsulting.tech/
|
||||||
about: Get help or ask questions through our website
|
about: Get help or ask questions through our website
|
||||||
- name: 📚 moko-platform Documentation
|
- name: 📚 MokoCLI Documentation
|
||||||
url: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
url: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
about: View our coding standards and best practices
|
about: View our coding standards and best practices
|
||||||
- name: 🔒 Report a Security Vulnerability
|
- name: 🔒 Report a Security Vulnerability
|
||||||
url: https://git.mokoconsulting.tech/mokoconsulting-tech/.github-private/security/advisories/new
|
url: https://git.mokoconsulting.tech/mokoconsulting-tech/.github-private/security/advisories/new
|
||||||
|
|||||||
@@ -42,7 +42,7 @@ Suggested text here
|
|||||||
<!-- Add any other context, screenshots, or references -->
|
<!-- Add any other context, screenshots, or references -->
|
||||||
|
|
||||||
## Standards Alignment
|
## Standards Alignment
|
||||||
- [ ] Follows moko-platform documentation guidelines
|
- [ ] Follows MokoCLI documentation guidelines
|
||||||
- [ ] Uses en_US/en_GB localization
|
- [ ] Uses en_US/en_GB localization
|
||||||
- [ ] Includes proper SPDX headers where applicable
|
- [ ] Includes proper SPDX headers where applicable
|
||||||
|
|
||||||
|
|||||||
@@ -37,7 +37,7 @@ If you have ideas about how this could be implemented, share them here:
|
|||||||
Add any other context, mockups, or screenshots about the feature request here.
|
Add any other context, mockups, or screenshots about the feature request here.
|
||||||
|
|
||||||
## Relevant Standards
|
## Relevant Standards
|
||||||
Does this relate to any standards in [moko-platform](https://git.mokoconsulting.tech/MokoConsulting/moko-platform)?
|
Does this relate to any standards in [MokoCLI](https://git.mokoconsulting.tech/MokoConsulting/mokocli)?
|
||||||
- [ ] Accessibility (WCAG 2.1 AA)
|
- [ ] Accessibility (WCAG 2.1 AA)
|
||||||
- [ ] Localization (en_US/en_GB)
|
- [ ] Localization (en_US/en_GB)
|
||||||
- [ ] Security best practices
|
- [ ] Security best practices
|
||||||
|
|||||||
@@ -35,7 +35,7 @@ Use this template only for:
|
|||||||
<!-- Describe how this could be addressed -->
|
<!-- Describe how this could be addressed -->
|
||||||
|
|
||||||
## Standards Reference
|
## Standards Reference
|
||||||
Does this relate to security standards in [moko-platform](https://git.mokoconsulting.tech/MokoConsulting/moko-platform)?
|
Does this relate to security standards in [MokoCLI](https://git.mokoconsulting.tech/MokoConsulting/mokocli)?
|
||||||
- [ ] SPDX license identifiers
|
- [ ] SPDX license identifiers
|
||||||
- [ ] Secret management
|
- [ ] Secret management
|
||||||
- [ ] Dependency security
|
- [ ] Dependency security
|
||||||
|
|||||||
@@ -2,8 +2,8 @@
|
|||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Automation
|
# INGROUP: MokoCLI.Automation
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/branch-protection.yml
|
# PATH: /.gitea/workflows/branch-protection.yml
|
||||||
# BRIEF: Apply standardised branch protection rules to all governed repositories
|
# BRIEF: Apply standardised branch protection rules to all governed repositories
|
||||||
#
|
#
|
||||||
@@ -62,7 +62,7 @@ jobs:
|
|||||||
API="${GITEA_URL}/api/v1"
|
API="${GITEA_URL}/api/v1"
|
||||||
|
|
||||||
# Platform/standards/infra repos to exclude
|
# Platform/standards/infra repos to exclude
|
||||||
EXCLUDE="gitea-org-config org-profile gitea-private .mokogitea-private moko-platform MokoTesting"
|
EXCLUDE="gitea-org-config org-profile gitea-private .mokogitea-private mokocli mokoplatform MokoTesting"
|
||||||
EXCLUDE="$EXCLUDE MokoStandards-Template-Client MokoStandards-Template-Dolibarr MokoStandards-Template-Generic MokoStandards-Template-Joomla MokoDoliProjTemplate"
|
EXCLUDE="$EXCLUDE MokoStandards-Template-Client MokoStandards-Template-Dolibarr MokoStandards-Template-Generic MokoStandards-Template-Joomla MokoDoliProjTemplate"
|
||||||
|
|
||||||
if [ -n "${{ inputs.repos }}" ]; then
|
if [ -n "${{ inputs.repos }}" ]; then
|
||||||
|
|||||||
@@ -2,8 +2,8 @@
|
|||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Automation
|
# INGROUP: MokoCLI.Automation
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/bulk-repo-sync.yml
|
# PATH: /.gitea/workflows/bulk-repo-sync.yml
|
||||||
# BRIEF: Bulk repo sync — runs from API repo, syncs standards to all governed repos
|
# BRIEF: Bulk repo sync — runs from API repo, syncs standards to all governed repos
|
||||||
|
|
||||||
|
|||||||
@@ -2,9 +2,9 @@
|
|||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: moko-platform.CI
|
# DEFGROUP: MokoCLI.CI
|
||||||
# INGROUP: moko-platform
|
# INGROUP: MokoCLI
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/pr-branch-check.yml
|
# PATH: /.gitea/workflows/pr-branch-check.yml
|
||||||
# BRIEF: PR branch merge policy enforcement
|
# BRIEF: PR branch merge policy enforcement
|
||||||
#
|
#
|
||||||
|
|||||||
@@ -4,8 +4,8 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Automation
|
# INGROUP: MokoCLI.Automation
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/renovate.yml
|
# PATH: /.gitea/workflows/renovate.yml
|
||||||
# BRIEF: Run Renovate Bot across all governed repos for dependency updates
|
# BRIEF: Run Renovate Bot across all governed repos for dependency updates
|
||||||
#
|
#
|
||||||
@@ -61,7 +61,7 @@ jobs:
|
|||||||
run: |
|
run: |
|
||||||
API="${GITEA_URL}/api/v1"
|
API="${GITEA_URL}/api/v1"
|
||||||
|
|
||||||
EXCLUDE="gitea-org-config org-profile gitea-private .mokogitea-private moko-platform MokoTesting"
|
EXCLUDE="gitea-org-config org-profile gitea-private .mokogitea-private mokocli mokoplatform MokoTesting"
|
||||||
EXCLUDE="$EXCLUDE MokoStandards-Template-Client MokoStandards-Template-Dolibarr MokoStandards-Template-Generic MokoStandards-Template-Joomla MokoDoliProjTemplate"
|
EXCLUDE="$EXCLUDE MokoStandards-Template-Client MokoStandards-Template-Dolibarr MokoStandards-Template-Generic MokoStandards-Template-Joomla MokoDoliProjTemplate"
|
||||||
|
|
||||||
if [ -n "${{ inputs.repos }}" ]; then
|
if [ -n "${{ inputs.repos }}" ]; then
|
||||||
|
|||||||
@@ -4,8 +4,8 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Maintenance
|
# INGROUP: MokoCLI.Maintenance
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/sync-wikis.yml
|
# PATH: /.gitea/workflows/sync-wikis.yml
|
||||||
# BRIEF: Daily sync of all Gitea wikis to consolidated GitHub wiki repo
|
# BRIEF: Daily sync of all Gitea wikis to consolidated GitHub wiki repo
|
||||||
|
|
||||||
|
|||||||
@@ -4,8 +4,8 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Release
|
# INGROUP: MokoCLI.Release
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.mokogitea/workflows/auto-bump.yml
|
# PATH: /.mokogitea/workflows/auto-bump.yml
|
||||||
# VERSION: 09.23.00
|
# VERSION: 09.23.00
|
||||||
# BRIEF: Auto patch-bump version on every push to dev (skips merge commits)
|
# BRIEF: Auto patch-bump version on every push to dev (skips merge commits)
|
||||||
@@ -43,19 +43,25 @@ jobs:
|
|||||||
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
||||||
fetch-depth: 1
|
fetch-depth: 1
|
||||||
|
|
||||||
- name: Setup moko-platform tools
|
- name: Setup MokoCLI tools
|
||||||
run: |
|
run: |
|
||||||
if ! command -v composer &> /dev/null; then
|
# Check both new (mokocli) and legacy (mokoplatform) install paths
|
||||||
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer >/dev/null 2>&1
|
if [ -f "/opt/mokocli/cli/version_bump.php" ] && [ -f "/opt/mokocli/vendor/autoload.php" ]; then
|
||||||
fi
|
echo "Using pre-installed /opt/mokocli"
|
||||||
if [ -d "/opt/moko-platform/cli" ]; then
|
echo "MOKO_CLI=/opt/mokocli/cli" >> "$GITHUB_ENV"
|
||||||
echo "MOKO_CLI=/opt/moko-platform/cli" >> "$GITHUB_ENV"
|
elif [ -f "/opt/mokoplatform/cli/version_bump.php" ] && [ -f "/opt/mokoplatform/vendor/autoload.php" ]; then
|
||||||
|
echo "Using pre-installed /opt/mokoplatform (legacy path)"
|
||||||
|
echo "MOKO_CLI=/opt/mokoplatform/cli" >> "$GITHUB_ENV"
|
||||||
else
|
else
|
||||||
|
if ! command -v composer &> /dev/null; then
|
||||||
|
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer >/dev/null 2>&1
|
||||||
|
fi
|
||||||
|
rm -rf /tmp/mokocli
|
||||||
git clone --depth 1 --branch main --quiet \
|
git clone --depth 1 --branch main --quiet \
|
||||||
"https://x-access-token:${{ secrets.MOKOGITEA_TOKEN }}@git.mokoconsulting.tech/MokoConsulting/moko-platform.git" \
|
"https://x-access-token:${{ secrets.MOKOGITEA_TOKEN }}@git.mokoconsulting.tech/MokoConsulting/mokocli.git" \
|
||||||
/tmp/moko-platform-api
|
/tmp/mokocli
|
||||||
cd /tmp/moko-platform-api && composer install --no-dev --no-interaction --quiet
|
cd /tmp/mokocli && composer install --no-dev --no-interaction --quiet
|
||||||
echo "MOKO_CLI=/tmp/moko-platform-api/cli" >> "$GITHUB_ENV"
|
echo "MOKO_CLI=/tmp/mokocli/cli" >> "$GITHUB_ENV"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
- name: Bump version
|
- name: Bump version
|
||||||
|
|||||||
@@ -1,285 +1,332 @@
|
|||||||
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
#
|
#
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Release
|
# INGROUP: MokoCLI.Release
|
||||||
# REPO: https://git.mokoconsulting.tech/mokoconsulting-tech/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /templates/workflows/universal/auto-release.yml.template
|
# PATH: /templates/workflows/universal/auto-release.yml.template
|
||||||
# VERSION: 05.00.00
|
# VERSION: 05.00.00
|
||||||
# BRIEF: Universal build & release � detects platform from manifest.xml
|
# BRIEF: Universal build & release � detects platform from manifest.xml
|
||||||
#
|
#
|
||||||
# +========================================================================+
|
# +========================================================================+
|
||||||
# | UNIVERSAL BUILD & RELEASE PIPELINE |
|
# | UNIVERSAL BUILD & RELEASE PIPELINE |
|
||||||
# +========================================================================+
|
# +========================================================================+
|
||||||
# | |
|
# | |
|
||||||
# | Reads manifest.xml (joomla|dolibarr|generic) to branch logic. |
|
# | Reads manifest.xml (joomla|dolibarr|generic) to branch logic. |
|
||||||
# | |
|
# | |
|
||||||
# | Platform-specific: |
|
# | Platform-specific: |
|
||||||
# | joomla: XML manifest, updates.xml, type-prefixed packages |
|
# | joomla: XML manifest, type-prefixed packages |
|
||||||
# | dolibarr: mod*.class.php, update.txt, dev version reset |
|
# | dolibarr: mod*.class.php, update.txt, dev version reset |
|
||||||
# | generic: README-only, no update stream |
|
# | generic: README-only, no update stream |
|
||||||
# | |
|
# | |
|
||||||
# +========================================================================+
|
# +========================================================================+
|
||||||
|
|
||||||
name: "Universal: Build & Release"
|
name: "Universal: Build & Release"
|
||||||
|
|
||||||
on:
|
on:
|
||||||
pull_request:
|
pull_request:
|
||||||
types: [opened, closed]
|
types: [opened, closed]
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
inputs:
|
inputs:
|
||||||
action:
|
action:
|
||||||
description: 'Action to perform'
|
description: 'Action to perform'
|
||||||
required: false
|
required: false
|
||||||
type: choice
|
type: choice
|
||||||
default: release
|
default: release
|
||||||
options:
|
options:
|
||||||
- release
|
- release
|
||||||
- promote-rc
|
- promote-rc
|
||||||
|
|
||||||
env:
|
env:
|
||||||
FORCE_JAVASCRIPT_ACTIONS_TO_NODE24: true
|
FORCE_JAVASCRIPT_ACTIONS_TO_NODE24: true
|
||||||
GITEA_URL: ${{ vars.GITEA_URL || 'https://git.mokoconsulting.tech' }}
|
GITEA_URL: ${{ vars.GITEA_URL || 'https://git.mokoconsulting.tech' }}
|
||||||
GITEA_ORG: ${{ vars.GITEA_ORG || github.repository_owner }}
|
GITEA_ORG: ${{ vars.GITEA_ORG || github.repository_owner }}
|
||||||
GITEA_REPO: ${{ vars.GITEA_REPO || github.event.repository.name }}
|
GITEA_REPO: ${{ vars.GITEA_REPO || github.event.repository.name }}
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
contents: write
|
contents: write
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
# ── PR Opened → Rename branch to RC and build RC release ─────────────────────
|
# ── PR Opened → Rename branch to RC and build RC release ─────────────────────
|
||||||
promote-rc:
|
promote-rc:
|
||||||
name: Promote to RC
|
name: Promote to RC
|
||||||
runs-on: release
|
runs-on: release
|
||||||
if: >-
|
if: >-
|
||||||
(github.event.action == 'opened' && github.event.pull_request.merged != true) ||
|
(github.event.action == 'opened' && github.event.pull_request.merged != true) ||
|
||||||
(github.event_name == 'workflow_dispatch' && inputs.action == 'promote-rc')
|
(github.event_name == 'workflow_dispatch' && inputs.action == 'promote-rc')
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout repository
|
- name: Checkout repository
|
||||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||||
with:
|
with:
|
||||||
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
||||||
fetch-depth: 1
|
fetch-depth: 1
|
||||||
|
|
||||||
- name: Setup moko-platform tools
|
- name: Setup MokoCLI tools
|
||||||
env:
|
env:
|
||||||
MOKO_CLONE_TOKEN: ${{ secrets.MOKOGITEA_TOKEN }}
|
MOKO_CLONE_TOKEN: ${{ secrets.MOKOGITEA_TOKEN }}
|
||||||
MOKO_CLONE_HOST: git.mokoconsulting.tech/MokoConsulting
|
MOKO_CLONE_HOST: git.mokoconsulting.tech/MokoConsulting
|
||||||
run: |
|
run: |
|
||||||
if ! command -v composer &> /dev/null; then
|
# Check both new (mokocli) and legacy (mokoplatform) install paths
|
||||||
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer >/dev/null 2>&1
|
if [ -f /opt/mokocli/cli/version_bump.php ] && [ -f /opt/mokocli/vendor/autoload.php ]; then
|
||||||
fi
|
echo "Using pre-installed /opt/mokocli"
|
||||||
# Always fetch latest CLI tools — never use stale cache from previous runs
|
echo MOKO_CLI=/opt/mokocli/cli >> $GITHUB_ENV
|
||||||
rm -rf /tmp/moko-platform-api
|
elif [ -f /opt/mokoplatform/cli/version_bump.php ] && [ -f /opt/mokoplatform/vendor/autoload.php ]; then
|
||||||
git clone --depth 1 --branch main --quiet \
|
echo "Using pre-installed /opt/mokoplatform (legacy path)"
|
||||||
"https://x-access-token:${MOKO_CLONE_TOKEN}@${MOKO_CLONE_HOST}/moko-platform.git" \
|
echo MOKO_CLI=/opt/mokoplatform/cli >> $GITHUB_ENV
|
||||||
/tmp/moko-platform-api
|
else
|
||||||
cd /tmp/moko-platform-api
|
echo "Falling back to fresh clone"
|
||||||
composer install --no-dev --no-interaction --quiet
|
if ! command -v composer > /dev/null 2>&1; then
|
||||||
|
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer > /dev/null 2>&1
|
||||||
- name: Rename branch to rc
|
fi
|
||||||
run: |
|
rm -rf /tmp/mokocli
|
||||||
php /tmp/moko-platform-api/cli/branch_rename.php \
|
CLONE_URL=https://x-access-token:${MOKO_CLONE_TOKEN}@${MOKO_CLONE_HOST}/mokocli.git
|
||||||
--from "${{ github.event.pull_request.head.ref || 'dev' }}" --to rc \
|
git clone --depth 1 --branch main --quiet $CLONE_URL /tmp/mokocli
|
||||||
--token "${{ secrets.MOKOGITEA_TOKEN }}" \
|
cd /tmp/mokocli
|
||||||
--api-base "${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}" \
|
composer install --no-dev --no-interaction --quiet
|
||||||
--pr "${{ github.event.pull_request.number }}"
|
echo MOKO_CLI=/tmp/mokocli/cli >> $GITHUB_ENV
|
||||||
|
fi
|
||||||
- name: Checkout rc and configure git
|
|
||||||
run: |
|
- name: Rename branch to rc
|
||||||
git fetch origin rc
|
run: |
|
||||||
git checkout rc
|
php ${MOKO_CLI}/branch_rename.php \
|
||||||
git config --local user.email "gitea-actions[bot]@mokoconsulting.tech"
|
--from "${{ github.event.pull_request.head.ref || 'dev' }}" --to rc \
|
||||||
git config --local user.name "gitea-actions[bot]"
|
--token "${{ secrets.MOKOGITEA_TOKEN }}" \
|
||||||
git remote set-url origin "https://x-access-token:${{ secrets.MOKOGITEA_TOKEN }}@git.mokoconsulting.tech/${{ github.repository }}.git"
|
--api-base "${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}" \
|
||||||
|
--pr "${{ github.event.pull_request.number }}"
|
||||||
- name: Publish RC release
|
|
||||||
run: |
|
- name: Checkout rc and configure git
|
||||||
php /tmp/moko-platform-api/cli/release_publish.php \
|
run: |
|
||||||
--path . --stability rc --bump minor --branch rc \
|
git fetch origin rc
|
||||||
--token "${{ secrets.MOKOGITEA_TOKEN }}" \
|
git checkout rc
|
||||||
--skip-update-stream
|
git config --local user.email "gitea-actions[bot]@mokoconsulting.tech"
|
||||||
|
git config --local user.name "gitea-actions[bot]"
|
||||||
- name: Summary
|
git remote set-url origin "https://x-access-token:${{ secrets.MOKOGITEA_TOKEN }}@git.mokoconsulting.tech/${{ github.repository }}.git"
|
||||||
if: always()
|
|
||||||
run: |
|
- name: Publish RC release
|
||||||
echo "## Promoted to Release Candidate" >> $GITHUB_STEP_SUMMARY
|
run: |
|
||||||
echo "Branch renamed to rc, minor bump, RC release built (updates.xml managed by Gitea Pages)" >> $GITHUB_STEP_SUMMARY
|
php ${MOKO_CLI}/release_publish.php \
|
||||||
|
--path . --stability rc --bump minor --branch rc \
|
||||||
# ── Merged PR → Build & Release (or promote RC to stable) ────────────────────
|
--token "${{ secrets.MOKOGITEA_TOKEN }}"
|
||||||
release:
|
|
||||||
name: Build & Release Pipeline
|
- name: Summary
|
||||||
runs-on: release
|
if: always()
|
||||||
if: >-
|
run: |
|
||||||
github.event.pull_request.merged == true ||
|
echo "## Promoted to Release Candidate" >> $GITHUB_STEP_SUMMARY
|
||||||
(github.event_name == 'workflow_dispatch' && inputs.action != 'promote-rc')
|
echo "Branch renamed to rc, minor bump, RC release built" >> $GITHUB_STEP_SUMMARY
|
||||||
|
|
||||||
steps:
|
# ── Merged PR → Build & Release (or promote RC to stable) ────────────────────
|
||||||
- name: Checkout repository
|
release:
|
||||||
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
name: Build & Release Pipeline
|
||||||
with:
|
runs-on: release
|
||||||
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
if: >-
|
||||||
fetch-depth: 0
|
github.event.pull_request.merged == true ||
|
||||||
|
(github.event_name == 'workflow_dispatch' && inputs.action != 'promote-rc')
|
||||||
- name: Configure git for bot pushes
|
|
||||||
run: |
|
steps:
|
||||||
git config --local user.email "gitea-actions[bot]@mokoconsulting.tech"
|
- name: Checkout repository
|
||||||
git config --local user.name "gitea-actions[bot]"
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||||
git remote set-url origin "https://x-access-token:${{ secrets.MOKOGITEA_TOKEN }}@git.mokoconsulting.tech/${{ github.repository }}.git"
|
with:
|
||||||
|
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
||||||
- name: Check for merge conflict markers
|
fetch-depth: 0
|
||||||
run: |
|
|
||||||
CONFLICTS=$(grep -rn '<<<<<<< \|>>>>>>> \|^=======$' --include='*.php' --include='*.xml' --include='*.css' --include='*.js' --include='*.json' --include='*.md' --include='*.yml' --include='*.yaml' --include='*.ini' --include='*.txt' . 2>/dev/null | grep -v '.git/' || true)
|
- name: Configure git for bot pushes
|
||||||
if [ -n "$CONFLICTS" ]; then
|
run: |
|
||||||
echo "::error::Merge conflict markers found — aborting release"
|
git config --local user.email "gitea-actions[bot]@mokoconsulting.tech"
|
||||||
echo "## Release Blocked: Conflict Markers" >> $GITHUB_STEP_SUMMARY
|
git config --local user.name "gitea-actions[bot]"
|
||||||
echo '```' >> $GITHUB_STEP_SUMMARY
|
git remote set-url origin "https://x-access-token:${{ secrets.MOKOGITEA_TOKEN }}@git.mokoconsulting.tech/${{ github.repository }}.git"
|
||||||
echo "$CONFLICTS" >> $GITHUB_STEP_SUMMARY
|
|
||||||
echo '```' >> $GITHUB_STEP_SUMMARY
|
- name: Check for merge conflict markers
|
||||||
exit 1
|
run: |
|
||||||
fi
|
CONFLICTS=$(grep -rn '<<<<<<< \|>>>>>>> \|^=======$' --include='*.php' --include='*.xml' --include='*.css' --include='*.js' --include='*.json' --include='*.md' --include='*.yml' --include='*.yaml' --include='*.ini' --include='*.txt' . 2>/dev/null | grep -v '.git/' || true)
|
||||||
echo "No conflict markers found"
|
if [ -n "$CONFLICTS" ]; then
|
||||||
|
echo "::error::Merge conflict markers found — aborting release"
|
||||||
- name: Setup moko-platform tools
|
echo "## Release Blocked: Conflict Markers" >> $GITHUB_STEP_SUMMARY
|
||||||
env:
|
echo '```' >> $GITHUB_STEP_SUMMARY
|
||||||
MOKO_CLONE_TOKEN: ${{ secrets.MOKOGITEA_TOKEN }}
|
echo "$CONFLICTS" >> $GITHUB_STEP_SUMMARY
|
||||||
MOKO_CLONE_HOST: git.mokoconsulting.tech/MokoConsulting
|
echo '```' >> $GITHUB_STEP_SUMMARY
|
||||||
COMPOSER_AUTH: '{"github-oauth":{"github.com":"${{ secrets.GH_MIRROR_TOKEN }}"}}'
|
exit 1
|
||||||
run: |
|
fi
|
||||||
# Ensure PHP + Composer are available
|
echo "No conflict markers found"
|
||||||
if ! command -v composer &> /dev/null; then
|
|
||||||
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer >/dev/null 2>&1
|
- name: Setup MokoCLI tools
|
||||||
fi
|
env:
|
||||||
# Always fetch latest CLI tools — never use stale cache from previous runs
|
MOKO_CLONE_TOKEN: ${{ secrets.MOKOGITEA_TOKEN }}
|
||||||
rm -rf /tmp/moko-platform-api
|
MOKO_CLONE_HOST: git.mokoconsulting.tech/MokoConsulting
|
||||||
git clone --depth 1 --branch main --quiet \
|
COMPOSER_AUTH: '{"github-oauth":{"github.com":"${{ secrets.GH_MIRROR_TOKEN }}"}}'
|
||||||
"https://x-access-token:${MOKO_CLONE_TOKEN}@${MOKO_CLONE_HOST}/moko-platform.git" \
|
run: |
|
||||||
/tmp/moko-platform-api
|
# Check both new (mokocli) and legacy (mokoplatform) install paths
|
||||||
cd /tmp/moko-platform-api
|
if [ -f /opt/mokocli/cli/version_bump.php ] && [ -f /opt/mokocli/vendor/autoload.php ]; then
|
||||||
composer install --no-dev --no-interaction --quiet
|
echo "Using pre-installed /opt/mokocli"
|
||||||
|
echo MOKO_CLI=/opt/mokocli/cli >> $GITHUB_ENV
|
||||||
|
elif [ -f /opt/mokoplatform/cli/version_bump.php ] && [ -f /opt/mokoplatform/vendor/autoload.php ]; then
|
||||||
- name: "Publish stable release"
|
echo "Using pre-installed /opt/mokoplatform (legacy path)"
|
||||||
run: |
|
echo MOKO_CLI=/opt/mokoplatform/cli >> $GITHUB_ENV
|
||||||
php /tmp/moko-platform-api/cli/release_publish.php \
|
else
|
||||||
--path . --stability stable --bump minor --branch main \
|
echo "Falling back to fresh clone"
|
||||||
--token "${{ secrets.MOKOGITEA_TOKEN }}" \
|
if ! command -v composer > /dev/null 2>&1; then
|
||||||
--skip-update-stream
|
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer > /dev/null 2>&1
|
||||||
|
fi
|
||||||
# -- STEP 9: Mirror to GitHub (stable only) --------------------------------
|
rm -rf /tmp/mokocli
|
||||||
- name: "Step 9: Mirror release to GitHub"
|
CLONE_URL=https://x-access-token:${MOKO_CLONE_TOKEN}@${MOKO_CLONE_HOST}/mokocli.git
|
||||||
if: >-
|
git clone --depth 1 --branch main --quiet $CLONE_URL /tmp/mokocli
|
||||||
steps.version.outputs.skip != 'true' &&
|
cd /tmp/mokocli
|
||||||
secrets.GH_MIRROR_TOKEN != ''
|
composer install --no-dev --no-interaction --quiet
|
||||||
continue-on-error: true
|
echo MOKO_CLI=/tmp/mokocli/cli >> $GITHUB_ENV
|
||||||
run: |
|
fi
|
||||||
VERSION="${{ steps.bump.outputs.version || steps.version.outputs.version }}"
|
|
||||||
RELEASE_TAG="${{ steps.version.outputs.release_tag }}"
|
- name: "Publish stable release"
|
||||||
GH_REPO="${{ vars.GH_MIRROR_REPO || github.repository }}"
|
run: |
|
||||||
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
php ${MOKO_CLI}/release_publish.php \
|
||||||
php /tmp/moko-platform-api/cli/release_mirror.php \
|
--path . --stability stable --bump minor --branch main \
|
||||||
--version "$VERSION" --tag "$RELEASE_TAG" \
|
--token "${{ secrets.MOKOGITEA_TOKEN }}"
|
||||||
--token "${{ secrets.MOKOGITEA_TOKEN }}" --api-base "$API_BASE" \
|
|
||||||
--gh-token "${{ secrets.GH_MIRROR_TOKEN }}" --gh-repo "$GH_REPO" \
|
- name: Update release notes from CHANGELOG.md
|
||||||
--branch main 2>&1 || true
|
run: |
|
||||||
echo "GitHub mirror updated" >> $GITHUB_STEP_SUMMARY
|
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
||||||
|
|
||||||
# -- STEP 10: Sync main branch to GitHub mirror ----------------------------
|
# Extract [Unreleased] section from changelog
|
||||||
- name: "Step 10: Push main to GitHub mirror"
|
if [ -f "CHANGELOG.md" ]; then
|
||||||
if: >-
|
NOTES=$(awk '/^## \[Unreleased\]/{found=1; next} /^## \[/{if(found) exit} found{print}' CHANGELOG.md)
|
||||||
steps.version.outputs.skip != 'true' &&
|
[ -z "$NOTES" ] && NOTES="Stable release"
|
||||||
secrets.GH_MIRROR_TOKEN != ''
|
else
|
||||||
continue-on-error: true
|
NOTES="Stable release"
|
||||||
run: |
|
fi
|
||||||
GH_REPO="${{ vars.GH_MIRROR_REPO || github.repository }}"
|
|
||||||
GH_ORG=$(echo "$GH_REPO" | cut -d/ -f1)
|
# Update release body via API
|
||||||
GH_NAME=$(echo "$GH_REPO" | cut -d/ -f2)
|
RELEASE_ID=$(curl -sf -H "Authorization: token ${{ secrets.MOKOGITEA_TOKEN }}" \
|
||||||
git remote add github "https://x-access-token:${{ secrets.GH_MIRROR_TOKEN }}@github.com/${GH_ORG}/${GH_NAME}.git" 2>/dev/null || \
|
"${API_BASE}/releases/tags/stable" | python3 -c "import json,sys; print(json.load(sys.stdin).get('id',''))" 2>/dev/null || true)
|
||||||
git remote set-url github "https://x-access-token:${{ secrets.GH_MIRROR_TOKEN }}@github.com/${GH_ORG}/${GH_NAME}.git"
|
|
||||||
git fetch origin main --depth=1
|
if [ -n "$RELEASE_ID" ]; then
|
||||||
git push github origin/main:refs/heads/main --force 2>/dev/null \
|
python3 -c "
|
||||||
&& echo "main branch pushed to GitHub mirror" \
|
import json, urllib.request
|
||||||
|| echo "WARNING: GitHub mirror push failed"
|
body = open('/dev/stdin').read()
|
||||||
|
payload = json.dumps({'body': body}).encode()
|
||||||
- name: "Step 11: Delete rc branch and recreate dev from main"
|
req = urllib.request.Request(
|
||||||
if: steps.version.outputs.skip != 'true'
|
'${API_BASE}/releases/${RELEASE_ID}',
|
||||||
continue-on-error: true
|
data=payload, method='PATCH',
|
||||||
run: |
|
headers={
|
||||||
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
'Authorization': 'token ${{ secrets.MOKOGITEA_TOKEN }}',
|
||||||
TOKEN="${{ secrets.MOKOGITEA_TOKEN }}"
|
'Content-Type': 'application/json'
|
||||||
|
})
|
||||||
# Delete rc branch (ephemeral — created by promote-rc)
|
urllib.request.urlopen(req)
|
||||||
curl -sf -X DELETE -H "Authorization: token ${TOKEN}" \
|
" <<< "$NOTES"
|
||||||
"${API_BASE}/branches/rc" 2>/dev/null \
|
echo "Release notes updated from CHANGELOG.md"
|
||||||
&& echo "Deleted rc branch" || echo "rc branch not found"
|
fi
|
||||||
|
|
||||||
# Delete dev branch
|
# -- STEP 9: Mirror to GitHub (stable only) --------------------------------
|
||||||
curl -sf -X DELETE -H "Authorization: token ${TOKEN}" \
|
- name: "Step 9: Mirror release to GitHub"
|
||||||
"${API_BASE}/branches/dev" 2>/dev/null && echo "Deleted dev branch"
|
if: >-
|
||||||
|
steps.version.outputs.skip != 'true' &&
|
||||||
# Recreate dev from main (now includes version bump + changelog promotion)
|
secrets.GH_MIRROR_TOKEN != ''
|
||||||
curl -sf -X POST -H "Authorization: token ${TOKEN}" \
|
continue-on-error: true
|
||||||
-H "Content-Type: application/json" \
|
run: |
|
||||||
"${API_BASE}/branches" \
|
VERSION="${{ steps.bump.outputs.version || steps.version.outputs.version }}"
|
||||||
-d '{"new_branch_name":"dev","old_branch_name":"main"}' 2>/dev/null && echo "Recreated dev from main"
|
RELEASE_TAG="${{ steps.version.outputs.release_tag }}"
|
||||||
|
GH_REPO="${{ vars.GH_MIRROR_REPO || github.repository }}"
|
||||||
echo "Pre-release branches cleaned, dev reset from main" >> $GITHUB_STEP_SUMMARY
|
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
||||||
|
php ${MOKO_CLI}/release_mirror.php \
|
||||||
- name: "Step 12: Create version branch from main"
|
--version "$VERSION" --tag "$RELEASE_TAG" \
|
||||||
if: steps.version.outputs.skip != 'true'
|
--token "${{ secrets.MOKOGITEA_TOKEN }}" --api-base "$API_BASE" \
|
||||||
continue-on-error: true
|
--gh-token "${{ secrets.GH_MIRROR_TOKEN }}" --gh-repo "$GH_REPO" \
|
||||||
run: |
|
--branch main 2>&1 || true
|
||||||
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
echo "GitHub mirror updated" >> $GITHUB_STEP_SUMMARY
|
||||||
TOKEN="${{ secrets.MOKOGITEA_TOKEN }}"
|
|
||||||
VERSION="${{ steps.bump.outputs.version || steps.version.outputs.version }}"
|
# -- STEP 10: Sync main branch to GitHub mirror ----------------------------
|
||||||
BRANCH_NAME="version/${VERSION}"
|
- name: "Step 10: Push main to GitHub mirror"
|
||||||
MAIN_SHA=$(git rev-parse HEAD)
|
if: >-
|
||||||
|
steps.version.outputs.skip != 'true' &&
|
||||||
# Delete old version branch if it exists (same version re-release)
|
secrets.GH_MIRROR_TOKEN != ''
|
||||||
curl -sf -X DELETE -H "Authorization: token ${TOKEN}" "${API_BASE}/branches/${BRANCH_NAME}" 2>/dev/null && echo "Deleted old ${BRANCH_NAME}"
|
continue-on-error: true
|
||||||
|
run: |
|
||||||
# Create version/XX.YY.ZZ from main
|
GH_REPO="${{ vars.GH_MIRROR_REPO || github.repository }}"
|
||||||
curl -sf -X POST -H "Authorization: token ${TOKEN}" -H "Content-Type: application/json" "${API_BASE}/branches" -d "{\"new_branch_name\":\"${BRANCH_NAME}\",\"old_branch_name\":\"main\"}" 2>/dev/null && echo "Created ${BRANCH_NAME} from main (${MAIN_SHA})" || echo "WARNING: ${BRANCH_NAME} creation failed"
|
GH_ORG=$(echo "$GH_REPO" | cut -d/ -f1)
|
||||||
|
GH_NAME=$(echo "$GH_REPO" | cut -d/ -f2)
|
||||||
echo "Version branch created: ${BRANCH_NAME} (${MAIN_SHA})" >> $GITHUB_STEP_SUMMARY
|
git remote add github "https://x-access-token:${{ secrets.GH_MIRROR_TOKEN }}@github.com/${GH_ORG}/${GH_NAME}.git" 2>/dev/null || \
|
||||||
|
git remote set-url github "https://x-access-token:${{ secrets.GH_MIRROR_TOKEN }}@github.com/${GH_ORG}/${GH_NAME}.git"
|
||||||
|
git fetch origin main --depth=1
|
||||||
|
git push github origin/main:refs/heads/main --force 2>/dev/null \
|
||||||
# -- Dolibarr post-release: Reset dev version -----------------------------
|
&& echo "main branch pushed to GitHub mirror" \
|
||||||
- name: "Post-release: Reset dev version"
|
|| echo "WARNING: GitHub mirror push failed"
|
||||||
if: steps.version.outputs.skip != 'true'
|
|
||||||
continue-on-error: true
|
- name: "Step 11: Delete rc branch and recreate dev from main"
|
||||||
run: |
|
if: steps.version.outputs.skip != 'true'
|
||||||
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
continue-on-error: true
|
||||||
php /tmp/moko-platform-api/cli/version_reset_dev.php \
|
run: |
|
||||||
--token "${{ secrets.MOKOGITEA_TOKEN }}" --api-base "${API_BASE}" \
|
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
||||||
--branch dev --path . 2>&1 || true
|
TOKEN="${{ secrets.MOKOGITEA_TOKEN }}"
|
||||||
|
|
||||||
# -- Summary --------------------------------------------------------------
|
# Delete rc branch (ephemeral — created by promote-rc)
|
||||||
- name: Pipeline Summary
|
curl -sf -X DELETE -H "Authorization: token ${TOKEN}" \
|
||||||
if: always()
|
"${API_BASE}/branches/rc" 2>/dev/null \
|
||||||
run: |
|
&& echo "Deleted rc branch" || echo "rc branch not found"
|
||||||
VERSION="${{ steps.bump.outputs.version || steps.version.outputs.version }}"
|
|
||||||
PLATFORM="${{ steps.platform.outputs.platform }}"
|
# Delete dev branch
|
||||||
if [ "${{ steps.version.outputs.skip }}" = "true" ]; then
|
curl -sf -X DELETE -H "Authorization: token ${TOKEN}" \
|
||||||
echo "## Release Skipped" >> $GITHUB_STEP_SUMMARY
|
"${API_BASE}/branches/dev" 2>/dev/null && echo "Deleted dev branch"
|
||||||
echo "No VERSION in README.md" >> $GITHUB_STEP_SUMMARY
|
|
||||||
elif [ "${{ steps.check.outputs.already_released }}" = "true" ]; then
|
# Recreate dev from main (now includes version bump + changelog promotion)
|
||||||
echo "## Already Released — ${VERSION}" >> $GITHUB_STEP_SUMMARY
|
curl -sf -X POST -H "Authorization: token ${TOKEN}" \
|
||||||
else
|
-H "Content-Type: application/json" \
|
||||||
echo "" >> $GITHUB_STEP_SUMMARY
|
"${API_BASE}/branches" \
|
||||||
echo "## Build & Release Complete (${PLATFORM})" >> $GITHUB_STEP_SUMMARY
|
-d '{"new_branch_name":"dev","old_branch_name":"main"}' 2>/dev/null && echo "Recreated dev from main"
|
||||||
echo "" >> $GITHUB_STEP_SUMMARY
|
|
||||||
echo "| Step | Result |" >> $GITHUB_STEP_SUMMARY
|
echo "Pre-release branches cleaned, dev reset from main" >> $GITHUB_STEP_SUMMARY
|
||||||
echo "|------|--------|" >> $GITHUB_STEP_SUMMARY
|
|
||||||
echo "| Platform | \`${PLATFORM}\` |" >> $GITHUB_STEP_SUMMARY
|
- name: "Step 12: Create version branch from main"
|
||||||
echo "| Version | \`${VERSION}\` |" >> $GITHUB_STEP_SUMMARY
|
if: steps.version.outputs.skip != 'true'
|
||||||
echo "| Branch | \`${{ steps.version.outputs.branch }}\` |" >> $GITHUB_STEP_SUMMARY
|
continue-on-error: true
|
||||||
echo "| Tag | \`${{ steps.version.outputs.tag }}\` |" >> $GITHUB_STEP_SUMMARY
|
run: |
|
||||||
echo "| Release | [View](${GITEA_URL}/${GITEA_ORG}/${GITEA_REPO}/releases/tag/${{ steps.version.outputs.tag }}) |" >> $GITHUB_STEP_SUMMARY
|
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
||||||
fi
|
TOKEN="${{ secrets.MOKOGITEA_TOKEN }}"
|
||||||
|
VERSION="${{ steps.bump.outputs.version || steps.version.outputs.version }}"
|
||||||
|
BRANCH_NAME="version/${VERSION}"
|
||||||
|
MAIN_SHA=$(git rev-parse HEAD)
|
||||||
|
|
||||||
|
# Delete old version branch if it exists (same version re-release)
|
||||||
|
curl -sf -X DELETE -H "Authorization: token ${TOKEN}" "${API_BASE}/branches/${BRANCH_NAME}" 2>/dev/null && echo "Deleted old ${BRANCH_NAME}"
|
||||||
|
|
||||||
|
# Create version/XX.YY.ZZ from main
|
||||||
|
curl -sf -X POST -H "Authorization: token ${TOKEN}" -H "Content-Type: application/json" "${API_BASE}/branches" -d "{\"new_branch_name\":\"${BRANCH_NAME}\",\"old_branch_name\":\"main\"}" 2>/dev/null && echo "Created ${BRANCH_NAME} from main (${MAIN_SHA})" || echo "WARNING: ${BRANCH_NAME} creation failed"
|
||||||
|
|
||||||
|
echo "Version branch created: ${BRANCH_NAME} (${MAIN_SHA})" >> $GITHUB_STEP_SUMMARY
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
# -- Dolibarr post-release: Reset dev version -----------------------------
|
||||||
|
- name: "Post-release: Reset dev version"
|
||||||
|
if: steps.version.outputs.skip != 'true'
|
||||||
|
continue-on-error: true
|
||||||
|
run: |
|
||||||
|
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
||||||
|
php ${MOKO_CLI}/version_reset_dev.php \
|
||||||
|
--token "${{ secrets.MOKOGITEA_TOKEN }}" --api-base "${API_BASE}" \
|
||||||
|
--branch dev --path . 2>&1 || true
|
||||||
|
|
||||||
|
# -- Summary --------------------------------------------------------------
|
||||||
|
- name: Pipeline Summary
|
||||||
|
if: always()
|
||||||
|
run: |
|
||||||
|
VERSION="${{ steps.bump.outputs.version || steps.version.outputs.version }}"
|
||||||
|
PLATFORM="${{ steps.platform.outputs.platform }}"
|
||||||
|
if [ "${{ steps.version.outputs.skip }}" = "true" ]; then
|
||||||
|
echo "## Release Skipped" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "No VERSION in README.md" >> $GITHUB_STEP_SUMMARY
|
||||||
|
elif [ "${{ steps.check.outputs.already_released }}" = "true" ]; then
|
||||||
|
echo "## Already Released — ${VERSION}" >> $GITHUB_STEP_SUMMARY
|
||||||
|
else
|
||||||
|
echo "" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "## Build & Release Complete (${PLATFORM})" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Step | Result |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "|------|--------|" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Platform | \`${PLATFORM}\` |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Version | \`${VERSION}\` |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Branch | \`${{ steps.version.outputs.branch }}\` |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Tag | \`${{ steps.version.outputs.tag }}\` |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Release | [View](${GITEA_URL}/${GITEA_ORG}/${GITEA_REPO}/releases/tag/${{ steps.version.outputs.tag }}) |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
fi
|
||||||
|
|||||||
@@ -4,10 +4,10 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: MokoPlatform.Universal
|
# INGROUP: MokoStandards.Universal
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.mokogitea/workflows/branch-cleanup.yml
|
# PATH: /.mokogitea/workflows/branch-cleanup.yml
|
||||||
# VERSION: 09.23.00
|
# VERSION: 01.00.00
|
||||||
# BRIEF: Delete feature branches after PR merge
|
# BRIEF: Delete feature branches after PR merge
|
||||||
|
|
||||||
name: "Branch Cleanup"
|
name: "Branch Cleanup"
|
||||||
|
|||||||
@@ -4,18 +4,18 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.CI
|
# INGROUP: MokoCLI.CI
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/ci-platform.yml
|
# PATH: /.mokogitea/workflows/ci-platform.yml
|
||||||
# VERSION: 09.23.00
|
# VERSION: 09.23.00
|
||||||
# BRIEF: moko-platform CI — the standards engine validates itself
|
# BRIEF: MokoCLI CI — the standards engine validates itself
|
||||||
#
|
#
|
||||||
# +========================================================================+
|
# +========================================================================+
|
||||||
# | MOKO-PLATFORM CI |
|
# | MOKOCLI CI |
|
||||||
# +========================================================================+
|
# +========================================================================+
|
||||||
# | |
|
# | |
|
||||||
# | This is NOT a generic CI workflow. This is the self-validation |
|
# | This is NOT a generic CI workflow. This is the self-validation |
|
||||||
# | pipeline for the central moko-platform enterprise engine. |
|
# | pipeline for the central MokoCLI enterprise engine. |
|
||||||
# | |
|
# | |
|
||||||
# | It dogfoods every tool the platform ships to governed repos: |
|
# | It dogfoods every tool the platform ships to governed repos: |
|
||||||
# | |
|
# | |
|
||||||
@@ -29,7 +29,7 @@
|
|||||||
# | |
|
# | |
|
||||||
# +========================================================================+
|
# +========================================================================+
|
||||||
|
|
||||||
name: "Platform: moko-platform CI"
|
name: "Platform: MokoCLI CI"
|
||||||
|
|
||||||
on:
|
on:
|
||||||
push:
|
push:
|
||||||
@@ -41,7 +41,7 @@ on:
|
|||||||
paths-ignore:
|
paths-ignore:
|
||||||
- '**.md'
|
- '**.md'
|
||||||
- 'wiki/**'
|
- 'wiki/**'
|
||||||
- '.gitea/ISSUE_TEMPLATE/**'
|
- '.mokogitea/ISSUE_TEMPLATE/**'
|
||||||
pull_request:
|
pull_request:
|
||||||
branches:
|
branches:
|
||||||
- main
|
- main
|
||||||
@@ -104,7 +104,7 @@ jobs:
|
|||||||
echo "::error file=${file}::PHP syntax error"
|
echo "::error file=${file}::PHP syntax error"
|
||||||
ERRORS=$((ERRORS + 1))
|
ERRORS=$((ERRORS + 1))
|
||||||
fi
|
fi
|
||||||
done < <(find lib/ validate/ automation/ cli/ src/ deploy/ -name "*.php" -print0 2>/dev/null)
|
done < <(find lib/ validate/ automation/ cli/ source/ src/ deploy/ -name "*.php" -print0 2>/dev/null)
|
||||||
|
|
||||||
{
|
{
|
||||||
echo "### PHP Syntax"
|
echo "### PHP Syntax"
|
||||||
@@ -270,7 +270,7 @@ jobs:
|
|||||||
echo "::warning file=${file}::Missing SPDX header"
|
echo "::warning file=${file}::Missing SPDX header"
|
||||||
MISSING=$((MISSING + 1))
|
MISSING=$((MISSING + 1))
|
||||||
fi
|
fi
|
||||||
done < <(find lib/ validate/ cli/ src/ automation/ deploy/ -name "*.php" -print0 2>/dev/null)
|
done < <(find lib/ validate/ cli/ source/ src/ automation/ deploy/ -name "*.php" -print0 2>/dev/null)
|
||||||
|
|
||||||
{
|
{
|
||||||
echo "### License Headers"
|
echo "### License Headers"
|
||||||
@@ -289,7 +289,7 @@ jobs:
|
|||||||
echo "::error file=${file}::Potential hardcoded secret detected"
|
echo "::error file=${file}::Potential hardcoded secret detected"
|
||||||
FOUND=$((FOUND + 1))
|
FOUND=$((FOUND + 1))
|
||||||
fi
|
fi
|
||||||
done < <(find lib/ validate/ cli/ src/ automation/ deploy/ -name "*.php" -print0 2>/dev/null)
|
done < <(find lib/ validate/ cli/ source/ src/ automation/ deploy/ -name "*.php" -print0 2>/dev/null)
|
||||||
|
|
||||||
{
|
{
|
||||||
echo "### Secret Detection"
|
echo "### Secret Detection"
|
||||||
@@ -421,7 +421,7 @@ jobs:
|
|||||||
- name: Check gate results
|
- name: Check gate results
|
||||||
run: |
|
run: |
|
||||||
{
|
{
|
||||||
echo "# moko-platform CI"
|
echo "# MokoCLI CI"
|
||||||
echo ""
|
echo ""
|
||||||
echo "| Gate | Job | Status |"
|
echo "| Gate | Job | Status |"
|
||||||
echo "|---|---|---|"
|
echo "|---|---|---|"
|
||||||
|
|||||||
@@ -4,9 +4,9 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Maintenance
|
# INGROUP: MokoCLI.Maintenance
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/cleanup.yml
|
# PATH: /.mokogitea/workflows/cleanup.yml
|
||||||
# VERSION: 09.23.00
|
# VERSION: 09.23.00
|
||||||
# BRIEF: Scheduled cleanup — delete merged branches and old workflow runs
|
# BRIEF: Scheduled cleanup — delete merged branches and old workflow runs
|
||||||
|
|
||||||
|
|||||||
@@ -4,8 +4,8 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Security
|
# INGROUP: MokoCLI.Security
|
||||||
# REPO: https://git.mokoconsulting.tech/mokoconsulting-tech/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /templates/workflows/gitleaks.yml.template
|
# PATH: /templates/workflows/gitleaks.yml.template
|
||||||
# VERSION: 09.23.00
|
# VERSION: 09.23.00
|
||||||
# BRIEF: Secret scanning — detect leaked credentials, API keys, and tokens
|
# BRIEF: Secret scanning — detect leaked credentials, API keys, and tokens
|
||||||
|
|||||||
@@ -4,8 +4,8 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Automation
|
# INGROUP: MokoCLI.Automation
|
||||||
# VERSION: 09.24.00
|
# VERSION: 09.25.05
|
||||||
# BRIEF: Auto-create feature branch when an issue is opened
|
# BRIEF: Auto-create feature branch when an issue is opened
|
||||||
|
|
||||||
name: "Universal: Issue Branch"
|
name: "Universal: Issue Branch"
|
||||||
|
|||||||
@@ -4,9 +4,9 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Notifications
|
# INGROUP: MokoCLI.Notifications
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/notify.yml
|
# PATH: /.mokogitea/workflows/notify.yml
|
||||||
# VERSION: 09.23.00
|
# VERSION: 09.23.00
|
||||||
# BRIEF: Push notifications via ntfy on release success or workflow failure
|
# BRIEF: Push notifications via ntfy on release success or workflow failure
|
||||||
|
|
||||||
|
|||||||
@@ -4,8 +4,8 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.CI
|
# INGROUP: MokoCLI.CI
|
||||||
# REPO: https://git.mokoconsulting.tech/mokoconsulting-tech/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /templates/workflows/universal/pr-check.yml.template
|
# PATH: /templates/workflows/universal/pr-check.yml.template
|
||||||
# VERSION: 09.23.00
|
# VERSION: 09.23.00
|
||||||
# BRIEF: PR gate — branch policy + code validation before merge
|
# BRIEF: PR gate — branch policy + code validation before merge
|
||||||
@@ -172,7 +172,8 @@ jobs:
|
|||||||
if: steps.platform.outputs.platform == 'joomla'
|
if: steps.platform.outputs.platform == 'joomla'
|
||||||
run: |
|
run: |
|
||||||
MISSING=0
|
MISSING=0
|
||||||
SOURCE_DIR="src"
|
SOURCE_DIR="source"
|
||||||
|
[ ! -d "$SOURCE_DIR" ] && SOURCE_DIR="src"
|
||||||
[ ! -d "$SOURCE_DIR" ] && exit 0
|
[ ! -d "$SOURCE_DIR" ] && exit 0
|
||||||
while IFS= read -r dir; do
|
while IFS= read -r dir; do
|
||||||
if [ ! -f "${dir}/index.html" ]; then
|
if [ ! -f "${dir}/index.html" ]; then
|
||||||
@@ -220,7 +221,7 @@ jobs:
|
|||||||
echo "joomla.asset.json: valid"
|
echo "joomla.asset.json: valid"
|
||||||
fi
|
fi
|
||||||
|
|
||||||
# Validate all XML files in src/ are well-formed
|
# Validate all XML files in source/src/ are well-formed
|
||||||
XML_ERRORS=0
|
XML_ERRORS=0
|
||||||
if command -v php &> /dev/null; then
|
if command -v php &> /dev/null; then
|
||||||
while IFS= read -r -d '' xmlfile; do
|
while IFS= read -r -d '' xmlfile; do
|
||||||
@@ -451,10 +452,11 @@ jobs:
|
|||||||
|
|
||||||
- name: Verify package source
|
- name: Verify package source
|
||||||
run: |
|
run: |
|
||||||
SOURCE_DIR="src"
|
SOURCE_DIR="source"
|
||||||
|
[ ! -d "$SOURCE_DIR" ] && SOURCE_DIR="src"
|
||||||
[ ! -d "$SOURCE_DIR" ] && SOURCE_DIR="htdocs"
|
[ ! -d "$SOURCE_DIR" ] && SOURCE_DIR="htdocs"
|
||||||
if [ ! -d "$SOURCE_DIR" ]; then
|
if [ ! -d "$SOURCE_DIR" ]; then
|
||||||
echo "::warning::No src/ or htdocs/ directory"
|
echo "::warning::No source/, src/, or htdocs/ directory"
|
||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
FILE_COUNT=$(find "$SOURCE_DIR" -type f | wc -l)
|
FILE_COUNT=$(find "$SOURCE_DIR" -type f | wc -l)
|
||||||
|
|||||||
@@ -4,19 +4,26 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Release
|
# INGROUP: MokoCLI.Release
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /templates/workflows/universal/pre-release.yml.template
|
# PATH: /templates/workflows/universal/pre-release.yml.template
|
||||||
# VERSION: 09.23.00
|
# VERSION: 05.01.00
|
||||||
# BRIEF: Manual pre-release -- builds dev/alpha/beta/rc packages from any branch
|
# BRIEF: Auto pre-release on push to dev/alpha/beta/rc branches
|
||||||
|
|
||||||
name: "Universal: Pre-Release"
|
name: "Universal: Pre-Release"
|
||||||
|
|
||||||
on:
|
on:
|
||||||
pull_request:
|
push:
|
||||||
types: [closed]
|
|
||||||
branches:
|
branches:
|
||||||
- dev
|
- dev
|
||||||
|
- 'fix/**'
|
||||||
|
- 'patch/**'
|
||||||
|
- 'hotfix/**'
|
||||||
|
- 'bugfix/**'
|
||||||
|
- 'chore/**'
|
||||||
|
- alpha
|
||||||
|
- beta
|
||||||
|
- rc
|
||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
inputs:
|
inputs:
|
||||||
stability:
|
stability:
|
||||||
@@ -39,11 +46,11 @@ env:
|
|||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
build:
|
build:
|
||||||
name: "Build Pre-Release (${{ inputs.stability || 'development' }})"
|
name: "Build Pre-Release (${{ inputs.stability || github.ref_name }})"
|
||||||
runs-on: release
|
runs-on: release
|
||||||
if: >-
|
if: >-
|
||||||
github.event_name == 'workflow_dispatch' ||
|
github.event_name == 'workflow_dispatch' ||
|
||||||
(github.event.pull_request.merged == true && github.event.pull_request.base.ref == 'dev')
|
github.event_name == 'push'
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
- name: Checkout
|
- name: Checkout
|
||||||
@@ -51,56 +58,84 @@ jobs:
|
|||||||
with:
|
with:
|
||||||
fetch-depth: 0
|
fetch-depth: 0
|
||||||
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
||||||
|
ref: ${{ github.ref_name }}
|
||||||
|
|
||||||
- name: Setup moko-platform tools
|
- name: Setup MokoCLI tools
|
||||||
env:
|
env:
|
||||||
MOKO_CLONE_TOKEN: ${{ secrets.MOKOGITEA_TOKEN }}
|
MOKO_CLONE_TOKEN: ${{ secrets.MOKOGITEA_TOKEN }}
|
||||||
MOKO_CLONE_HOST: git.mokoconsulting.tech/MokoConsulting
|
MOKO_CLONE_HOST: git.mokoconsulting.tech/MokoConsulting
|
||||||
run: |
|
run: |
|
||||||
if ! command -v composer &> /dev/null; then
|
# Check both new (mokocli) and legacy (mokoplatform) install paths
|
||||||
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer >/dev/null 2>&1
|
if [ -f /opt/mokocli/cli/version_bump.php ] && [ -f /opt/mokocli/cli/manifest_element.php ] && [ -f /opt/mokocli/vendor/autoload.php ]; then
|
||||||
|
echo "Using pre-installed /opt/mokocli"
|
||||||
|
echo MOKO_CLI=/opt/mokocli/cli >> $GITHUB_ENV
|
||||||
|
elif [ -f /opt/mokoplatform/cli/version_bump.php ] && [ -f /opt/mokoplatform/vendor/autoload.php ]; then
|
||||||
|
echo "Using pre-installed /opt/mokoplatform (legacy path)"
|
||||||
|
echo MOKO_CLI=/opt/mokoplatform/cli >> $GITHUB_ENV
|
||||||
|
else
|
||||||
|
echo "Falling back to fresh clone"
|
||||||
|
if ! command -v composer > /dev/null 2>&1; then
|
||||||
|
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer > /dev/null 2>&1
|
||||||
|
fi
|
||||||
|
rm -rf /tmp/mokocli
|
||||||
|
CLONE_URL=https://x-access-token:${MOKO_CLONE_TOKEN}@${MOKO_CLONE_HOST}/mokocli.git
|
||||||
|
git clone --depth 1 --branch main --quiet $CLONE_URL /tmp/mokocli
|
||||||
|
cd /tmp/mokocli && composer install --no-dev --no-interaction --quiet
|
||||||
|
echo MOKO_CLI=/tmp/mokocli/cli >> $GITHUB_ENV
|
||||||
fi
|
fi
|
||||||
# Always fetch latest CLI tools — never use stale cache from previous runs
|
|
||||||
rm -rf /tmp/moko-platform-api
|
|
||||||
git clone --depth 1 --branch main --quiet \
|
|
||||||
"https://x-access-token:${MOKO_CLONE_TOKEN}@${MOKO_CLONE_HOST}/moko-platform.git" \
|
|
||||||
/tmp/moko-platform-api
|
|
||||||
cd /tmp/moko-platform-api && composer install --no-dev --no-interaction --quiet
|
|
||||||
echo "MOKO_CLI=/tmp/moko-platform-api/cli" >> "$GITHUB_ENV"
|
|
||||||
|
|
||||||
- name: Detect platform
|
- name: Detect platform
|
||||||
id: platform
|
id: platform
|
||||||
run: |
|
run: |
|
||||||
|
# Auto-detect and update platform if not set in manifest
|
||||||
|
php ${MOKO_CLI}/platform_detect.php --path . --github-output 2>/dev/null || true
|
||||||
php ${MOKO_CLI}/manifest_read.php --path . --github-output
|
php ${MOKO_CLI}/manifest_read.php --path . --github-output
|
||||||
|
|
||||||
- name: Resolve metadata and bump version
|
- name: Resolve metadata and bump version
|
||||||
id: meta
|
id: meta
|
||||||
run: |
|
run: |
|
||||||
STABILITY="${{ inputs.stability || 'development' }}"
|
# Auto-detect stability from branch name on push, or use input on dispatch
|
||||||
|
if [ "${{ github.event_name }}" = "push" ]; then
|
||||||
|
case "${{ github.ref_name }}" in
|
||||||
|
rc) STABILITY="release-candidate" ;;
|
||||||
|
alpha) STABILITY="alpha" ;;
|
||||||
|
beta) STABILITY="beta" ;;
|
||||||
|
*) STABILITY="development" ;;
|
||||||
|
esac
|
||||||
|
else
|
||||||
|
STABILITY="${{ inputs.stability || 'development' }}"
|
||||||
|
fi
|
||||||
|
|
||||||
case "$STABILITY" in
|
case "$STABILITY" in
|
||||||
development) TAG="development" ;;
|
development) SUFFIX="-dev"; TAG="development" ;;
|
||||||
alpha) TAG="alpha" ;;
|
alpha) SUFFIX="-alpha"; TAG="alpha" ;;
|
||||||
beta) TAG="beta" ;;
|
beta) SUFFIX="-beta"; TAG="beta" ;;
|
||||||
release-candidate) TAG="release-candidate" ;;
|
release-candidate) SUFFIX="-rc"; TAG="release-candidate" ;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
# Bump version: patch for dev/alpha/beta, minor for RC
|
# Bump version via CLI: patch for dev/alpha/beta, minor for RC
|
||||||
case "$STABILITY" in
|
case "$STABILITY" in
|
||||||
release-candidate) php ${MOKO_CLI}/version_bump.php --path . --minor 2>/dev/null || true ;;
|
release-candidate) BUMP="minor" ;;
|
||||||
*) php ${MOKO_CLI}/version_bump.php --path . 2>/dev/null || true ;;
|
*) BUMP="patch" ;;
|
||||||
esac
|
esac
|
||||||
|
|
||||||
# Set stability suffix and fix consistency
|
php ${MOKO_CLI}/version_bump.php --path . $([ "$BUMP" = "minor" ] && echo "--minor") 2>/dev/null || true
|
||||||
VERSION=$(php ${MOKO_CLI}/version_read.php --path . 2>/dev/null || echo '00.00.01')
|
|
||||||
|
# Set stability suffix and verify consistency
|
||||||
|
VERSION=$(php ${MOKO_CLI}/version_read.php --path . 2>/dev/null || echo "00.00.01")
|
||||||
VERSION=$(echo "$VERSION" | sed 's/-\(dev\|alpha\|beta\|rc\)$//')
|
VERSION=$(echo "$VERSION" | sed 's/-\(dev\|alpha\|beta\|rc\)$//')
|
||||||
|
|
||||||
php ${MOKO_CLI}/version_set_platform.php \
|
php ${MOKO_CLI}/version_set_platform.php \
|
||||||
--path . --version "$VERSION" --branch "${{ github.ref_name }}" --stability "$STABILITY" 2>/dev/null || true
|
--path . --version "$VERSION" --branch "${{ github.ref_name }}" --stability "$STABILITY" 2>/dev/null || true
|
||||||
php ${MOKO_CLI}/version_check.php --path . --fix 2>/dev/null || true
|
php ${MOKO_CLI}/version_check.php --path . --fix 2>/dev/null || true
|
||||||
|
|
||||||
# Read final version with suffix
|
# Ensure licensing tags (updateservers, dlid) if enabled in manifest.xml
|
||||||
VERSION=$(php ${MOKO_CLI}/version_read.php --path . 2>/dev/null)
|
php ${MOKO_CLI}/manifest_licensing.php --path . --fix 2>/dev/null || true
|
||||||
[ -z "$VERSION" ] && VERSION="00.00.01"
|
|
||||||
|
# Append suffix for output
|
||||||
|
if [ -n "$SUFFIX" ]; then
|
||||||
|
VERSION="${VERSION}${SUFFIX}"
|
||||||
|
fi
|
||||||
|
|
||||||
# Commit version bump
|
# Commit version bump
|
||||||
git config --local user.email "gitea-actions[bot]@mokoconsulting.tech"
|
git config --local user.email "gitea-actions[bot]@mokoconsulting.tech"
|
||||||
@@ -125,11 +160,12 @@ jobs:
|
|||||||
|
|
||||||
echo "version=${VERSION}" >> "$GITHUB_OUTPUT"
|
echo "version=${VERSION}" >> "$GITHUB_OUTPUT"
|
||||||
echo "stability=${STABILITY}" >> "$GITHUB_OUTPUT"
|
echo "stability=${STABILITY}" >> "$GITHUB_OUTPUT"
|
||||||
|
echo "suffix=${SUFFIX}" >> "$GITHUB_OUTPUT"
|
||||||
echo "tag=${TAG}" >> "$GITHUB_OUTPUT"
|
echo "tag=${TAG}" >> "$GITHUB_OUTPUT"
|
||||||
echo "zip_name=${ZIP_NAME}" >> "$GITHUB_OUTPUT"
|
echo "zip_name=${ZIP_NAME}" >> "$GITHUB_OUTPUT"
|
||||||
echo "ext_element=${EXT_ELEMENT}" >> "$GITHUB_OUTPUT"
|
echo "ext_element=${EXT_ELEMENT}" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
echo "=== Pre-Release: ${EXT_ELEMENT} ${VERSION} ==="
|
echo "=== Pre-Release: ${EXT_ELEMENT} ${VERSION}${SUFFIX} ==="
|
||||||
|
|
||||||
- name: Create release
|
- name: Create release
|
||||||
id: release
|
id: release
|
||||||
@@ -140,7 +176,42 @@ jobs:
|
|||||||
php ${MOKO_CLI}/release_create.php \
|
php ${MOKO_CLI}/release_create.php \
|
||||||
--path . --version "$VERSION" --tag "$TAG" \
|
--path . --version "$VERSION" --tag "$TAG" \
|
||||||
--token "${{ secrets.MOKOGITEA_TOKEN }}" --api-base "$API_BASE" \
|
--token "${{ secrets.MOKOGITEA_TOKEN }}" --api-base "$API_BASE" \
|
||||||
--repo "${GITEA_REPO}" --branch dev --prerelease
|
--repo "${GITEA_REPO}" --branch "${{ github.ref_name }}" --prerelease
|
||||||
|
|
||||||
|
- name: Update release notes from CHANGELOG.md
|
||||||
|
run: |
|
||||||
|
TAG="${{ steps.meta.outputs.tag }}"
|
||||||
|
VERSION="${{ steps.meta.outputs.version }}"
|
||||||
|
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
||||||
|
|
||||||
|
# Extract [Unreleased] section from changelog (everything between [Unreleased] and next ## heading)
|
||||||
|
if [ -f "CHANGELOG.md" ]; then
|
||||||
|
NOTES=$(awk '/^## \[Unreleased\]/{found=1; next} /^## \[/{if(found) exit} found{print}' CHANGELOG.md)
|
||||||
|
[ -z "$NOTES" ] && NOTES="Release ${VERSION}"
|
||||||
|
else
|
||||||
|
NOTES="Release ${VERSION}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
# Update release body via API
|
||||||
|
RELEASE_ID=$(curl -sf -H "Authorization: token ${{ secrets.MOKOGITEA_TOKEN }}" \
|
||||||
|
"${API_BASE}/releases/tags/${TAG}" | python3 -c "import json,sys; print(json.load(sys.stdin).get('id',''))" 2>/dev/null || true)
|
||||||
|
|
||||||
|
if [ -n "$RELEASE_ID" ]; then
|
||||||
|
python3 -c "
|
||||||
|
import json, urllib.request
|
||||||
|
body = open('/dev/stdin').read()
|
||||||
|
payload = json.dumps({'body': body}).encode()
|
||||||
|
req = urllib.request.Request(
|
||||||
|
'${API_BASE}/releases/${RELEASE_ID}',
|
||||||
|
data=payload, method='PATCH',
|
||||||
|
headers={
|
||||||
|
'Authorization': 'token ${{ secrets.MOKOGITEA_TOKEN }}',
|
||||||
|
'Content-Type': 'application/json'
|
||||||
|
})
|
||||||
|
urllib.request.urlopen(req)
|
||||||
|
" <<< "$NOTES"
|
||||||
|
echo "Release notes updated from CHANGELOG.md"
|
||||||
|
fi
|
||||||
|
|
||||||
- name: Build package and upload
|
- name: Build package and upload
|
||||||
id: package
|
id: package
|
||||||
|
|||||||
File diff suppressed because it is too large
Load Diff
@@ -4,9 +4,9 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: moko-platform.Security
|
# INGROUP: MokoCLI.Security
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /.gitea/workflows/security-audit.yml
|
# PATH: /.mokogitea/workflows/security-audit.yml
|
||||||
# VERSION: 09.23.00
|
# VERSION: 09.23.00
|
||||||
# BRIEF: Dependency vulnerability scanning for composer and npm packages
|
# BRIEF: Dependency vulnerability scanning for composer and npm packages
|
||||||
|
|
||||||
|
|||||||
@@ -1,302 +0,0 @@
|
|||||||
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
|
||||||
#
|
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
|
||||||
#
|
|
||||||
# FILE INFORMATION
|
|
||||||
# DEFGROUP: Gitea.Workflow
|
|
||||||
# INGROUP: moko-platform.Universal
|
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
|
||||||
# PATH: /templates/workflows/update-server.yml
|
|
||||||
# VERSION: 09.23.00
|
|
||||||
# BRIEF: Pre-release build + update server XML for dev/alpha/beta/rc branches
|
|
||||||
#
|
|
||||||
# Thin wrapper around moko-platform CLI tools.
|
|
||||||
# Builds packages, updates updates.xml, and optionally deploys via SFTP.
|
|
||||||
#
|
|
||||||
# Joomla filters update entries by the user's "Minimum Stability" setting.
|
|
||||||
|
|
||||||
name: "Update Server"
|
|
||||||
|
|
||||||
on:
|
|
||||||
push:
|
|
||||||
branches:
|
|
||||||
- 'dev'
|
|
||||||
- 'dev/**'
|
|
||||||
- 'alpha/**'
|
|
||||||
- 'beta/**'
|
|
||||||
- 'rc/**'
|
|
||||||
paths:
|
|
||||||
- 'src/**'
|
|
||||||
- 'htdocs/**'
|
|
||||||
pull_request:
|
|
||||||
types: [closed]
|
|
||||||
branches:
|
|
||||||
- 'dev'
|
|
||||||
- 'dev/**'
|
|
||||||
- 'alpha/**'
|
|
||||||
- 'beta/**'
|
|
||||||
- 'rc/**'
|
|
||||||
paths:
|
|
||||||
- 'src/**'
|
|
||||||
- 'htdocs/**'
|
|
||||||
workflow_dispatch:
|
|
||||||
inputs:
|
|
||||||
stability:
|
|
||||||
description: 'Stability tag'
|
|
||||||
required: true
|
|
||||||
default: 'development'
|
|
||||||
type: choice
|
|
||||||
options:
|
|
||||||
- development
|
|
||||||
- alpha
|
|
||||||
- beta
|
|
||||||
- rc
|
|
||||||
- stable
|
|
||||||
|
|
||||||
env:
|
|
||||||
FORCE_JAVASCRIPT_ACTIONS_TO_NODE24: true
|
|
||||||
GITEA_URL: ${{ vars.GITEA_URL || 'https://git.mokoconsulting.tech' }}
|
|
||||||
GITEA_ORG: ${{ vars.GITEA_ORG || github.repository_owner }}
|
|
||||||
GITEA_REPO: ${{ vars.GITEA_REPO || github.event.repository.name }}
|
|
||||||
|
|
||||||
permissions:
|
|
||||||
contents: write
|
|
||||||
|
|
||||||
jobs:
|
|
||||||
update-xml:
|
|
||||||
name: Update Server
|
|
||||||
runs-on: release
|
|
||||||
if: >-
|
|
||||||
github.event.pull_request.merged == true || github.event_name == 'workflow_dispatch' || github.event_name == 'push'
|
|
||||||
|
|
||||||
steps:
|
|
||||||
- name: Checkout repository
|
|
||||||
uses: actions/checkout@v4
|
|
||||||
with:
|
|
||||||
token: ${{ secrets.MOKOGITEA_TOKEN }}
|
|
||||||
fetch-depth: 0
|
|
||||||
|
|
||||||
- name: Setup moko-platform tools
|
|
||||||
env:
|
|
||||||
MOKO_CLONE_TOKEN: ${{ secrets.MOKOGITEA_TOKEN }}
|
|
||||||
MOKO_CLONE_HOST: git.mokoconsulting.tech/MokoConsulting
|
|
||||||
COMPOSER_AUTH: '{"http-basic":{"git.mokoconsulting.tech":{"username":"token","password":"${{ secrets.MOKOGITEA_TOKEN }}"}}}'
|
|
||||||
run: |
|
|
||||||
if ! command -v composer &> /dev/null; then
|
|
||||||
sudo apt-get update -qq && sudo apt-get install -y -qq php-cli php-mbstring php-xml php-zip php-curl composer >/dev/null 2>&1
|
|
||||||
fi
|
|
||||||
# Always fetch latest CLI tools — never use stale cache from previous runs
|
|
||||||
rm -rf /tmp/moko-platform
|
|
||||||
git clone --depth 1 --branch main --quiet \
|
|
||||||
"https://x-access-token:${MOKO_CLONE_TOKEN}@${MOKO_CLONE_HOST}/moko-platform.git" \
|
|
||||||
/tmp/moko-platform 2>/dev/null || true
|
|
||||||
if [ -d "/tmp/moko-platform" ] && [ -f "/tmp/moko-platform/composer.json" ]; then
|
|
||||||
cd /tmp/moko-platform && composer install --no-dev --no-interaction --quiet 2>/dev/null || true
|
|
||||||
fi
|
|
||||||
echo "MOKO_CLI=/tmp/moko-platform/cli" >> "$GITHUB_ENV"
|
|
||||||
|
|
||||||
- name: Detect platform
|
|
||||||
id: platform
|
|
||||||
run: php ${MOKO_CLI}/manifest_read.php --path . --github-output
|
|
||||||
|
|
||||||
- name: Resolve stability and bump version
|
|
||||||
id: meta
|
|
||||||
run: |
|
|
||||||
BRANCH="${{ github.ref_name }}"
|
|
||||||
|
|
||||||
# Configure git for bot pushes
|
|
||||||
git config --local user.email "gitea-actions[bot]@mokoconsulting.tech"
|
|
||||||
git config --local user.name "gitea-actions[bot]"
|
|
||||||
git remote set-url origin "https://x-access-token:${{ secrets.MOKOGITEA_TOKEN }}@git.mokoconsulting.tech/${{ github.repository }}.git"
|
|
||||||
|
|
||||||
# Determine stability from branch or manual input
|
|
||||||
if [ "${{ github.event_name }}" = "workflow_dispatch" ]; then
|
|
||||||
STABILITY="${{ inputs.stability }}"
|
|
||||||
elif [[ "$BRANCH" == rc/* ]]; then
|
|
||||||
STABILITY="rc"
|
|
||||||
elif [[ "$BRANCH" == beta/* ]]; then
|
|
||||||
STABILITY="beta"
|
|
||||||
elif [[ "$BRANCH" == alpha/* ]]; then
|
|
||||||
STABILITY="alpha"
|
|
||||||
else
|
|
||||||
STABILITY="development"
|
|
||||||
fi
|
|
||||||
|
|
||||||
# Gitea release tag per stability
|
|
||||||
case "$STABILITY" in
|
|
||||||
development) TAG="development" ;;
|
|
||||||
alpha) TAG="alpha" ;;
|
|
||||||
beta) TAG="beta" ;;
|
|
||||||
rc) TAG="release-candidate" ;;
|
|
||||||
*) TAG="stable" ;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
# Bump patch, set platform suffix, fix consistency — version_bump preserves suffix
|
|
||||||
php ${MOKO_CLI}/version_set_platform.php \
|
|
||||||
--path . --version "$(php ${MOKO_CLI}/version_read.php --path . 2>/dev/null || echo '00.00.01')" \
|
|
||||||
--branch "$BRANCH" --stability "$STABILITY" 2>/dev/null || true
|
|
||||||
php ${MOKO_CLI}/version_bump.php --path . 2>/dev/null || true
|
|
||||||
php ${MOKO_CLI}/version_check.php --path . --fix 2>/dev/null || true
|
|
||||||
|
|
||||||
# Read final version (includes suffix, e.g. 01.02.15-dev)
|
|
||||||
VERSION=$(php ${MOKO_CLI}/version_read.php --path . 2>/dev/null || echo "00.00.01")
|
|
||||||
|
|
||||||
echo "version=${VERSION}" >> "$GITHUB_OUTPUT"
|
|
||||||
echo "stability=${STABILITY}" >> "$GITHUB_OUTPUT"
|
|
||||||
echo "tag=${TAG}" >> "$GITHUB_OUTPUT"
|
|
||||||
|
|
||||||
# Commit version bump if changed
|
|
||||||
git add -A
|
|
||||||
git diff --cached --quiet || {
|
|
||||||
git commit -m "chore(version): auto-bump ${VERSION} [skip ci]" \
|
|
||||||
--author="gitea-actions[bot] <gitea-actions[bot]@mokoconsulting.tech>"
|
|
||||||
git push
|
|
||||||
}
|
|
||||||
|
|
||||||
- name: Create release and upload package
|
|
||||||
id: package
|
|
||||||
run: |
|
|
||||||
VERSION="${{ steps.meta.outputs.version }}"
|
|
||||||
TAG="${{ steps.meta.outputs.tag }}"
|
|
||||||
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
|
||||||
|
|
||||||
# Create or update Gitea release
|
|
||||||
php ${MOKO_CLI}/release_create.php \
|
|
||||||
--path . --version "$VERSION" --tag "$TAG" \
|
|
||||||
--token "${{ secrets.MOKOGITEA_TOKEN }}" --api-base "$API_BASE" \
|
|
||||||
--repo "${GITEA_REPO}" --branch "${{ github.ref_name }}" --prerelease
|
|
||||||
|
|
||||||
# Build package and upload
|
|
||||||
php ${MOKO_CLI}/release_package.php \
|
|
||||||
--path . --version "$VERSION" --tag "$TAG" \
|
|
||||||
--token "${{ secrets.MOKOGITEA_TOKEN }}" --api-base "$API_BASE" \
|
|
||||||
--repo "${GITEA_REPO}" --output /tmp || true
|
|
||||||
|
|
||||||
- name: Update updates.xml
|
|
||||||
if: steps.platform.outputs.platform == 'joomla'
|
|
||||||
run: |
|
|
||||||
VERSION="${{ steps.meta.outputs.version }}"
|
|
||||||
STABILITY="${{ steps.meta.outputs.stability }}"
|
|
||||||
SHA256="${{ steps.package.outputs.sha256_zip }}"
|
|
||||||
|
|
||||||
if [ ! -f "updates.xml" ]; then
|
|
||||||
echo "No updates.xml — skipping"
|
|
||||||
exit 0
|
|
||||||
fi
|
|
||||||
|
|
||||||
SHA_FLAG=""
|
|
||||||
[ -n "$SHA256" ] && SHA_FLAG="--sha ${SHA256}"
|
|
||||||
|
|
||||||
php ${MOKO_CLI}/updates_xml_build.php \
|
|
||||||
--path . --version "${VERSION}" --stability "${STABILITY}" \
|
|
||||||
--gitea-url "${GITEA_URL}" --org "${GITEA_ORG}" --repo "${GITEA_REPO}" \
|
|
||||||
${SHA_FLAG}
|
|
||||||
|
|
||||||
# Commit and push updates.xml
|
|
||||||
git add updates.xml
|
|
||||||
git diff --cached --quiet || {
|
|
||||||
git commit -m "chore: update ${STABILITY} channel ${VERSION} [skip ci]"
|
|
||||||
git push
|
|
||||||
}
|
|
||||||
|
|
||||||
- name: Sync updates.xml to main
|
|
||||||
if: github.ref_name != 'main' && steps.platform.outputs.platform == 'joomla'
|
|
||||||
run: |
|
|
||||||
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
|
||||||
GITEA_TOKEN="${{ secrets.MOKOGITEA_TOKEN }}"
|
|
||||||
|
|
||||||
FILE_SHA=$(curl -sf -H "Authorization: token ${GITEA_TOKEN}" \
|
|
||||||
"${API_BASE}/contents/updates.xml?ref=main" | python3 -c "import sys,json; print(json.load(sys.stdin).get('sha',''))" 2>/dev/null || true)
|
|
||||||
|
|
||||||
if [ -n "$FILE_SHA" ] && [ -f "updates.xml" ]; then
|
|
||||||
python3 -c "
|
|
||||||
import base64, json, urllib.request, sys
|
|
||||||
with open('updates.xml', 'rb') as f:
|
|
||||||
content = base64.b64encode(f.read()).decode()
|
|
||||||
payload = json.dumps({
|
|
||||||
'content': content,
|
|
||||||
'sha': '${FILE_SHA}',
|
|
||||||
'message': 'chore: sync updates.xml from ${{ steps.meta.outputs.stability }} [skip ci]',
|
|
||||||
'branch': 'main'
|
|
||||||
}).encode()
|
|
||||||
req = urllib.request.Request(
|
|
||||||
'${API_BASE}/contents/updates.xml',
|
|
||||||
data=payload, method='PUT',
|
|
||||||
headers={
|
|
||||||
'Authorization': 'token ${GITEA_TOKEN}',
|
|
||||||
'Content-Type': 'application/json'
|
|
||||||
})
|
|
||||||
try:
|
|
||||||
urllib.request.urlopen(req)
|
|
||||||
print('updates.xml synced to main')
|
|
||||||
except Exception as e:
|
|
||||||
print(f'WARNING: sync to main failed: {e}', file=sys.stderr)
|
|
||||||
"
|
|
||||||
fi
|
|
||||||
|
|
||||||
- name: SFTP deploy to dev server
|
|
||||||
if: contains(github.ref, 'dev/') || github.ref == 'refs/heads/dev'
|
|
||||||
env:
|
|
||||||
DEV_HOST: ${{ vars.DEV_FTP_HOST }}
|
|
||||||
DEV_PATH: ${{ vars.DEV_FTP_PATH }}
|
|
||||||
DEV_SUFFIX: ${{ vars.DEV_FTP_SUFFIX }}
|
|
||||||
DEV_USER: ${{ vars.DEV_FTP_USERNAME }}
|
|
||||||
DEV_PORT: ${{ vars.DEV_FTP_PORT }}
|
|
||||||
DEV_KEY: ${{ secrets.DEV_FTP_KEY }}
|
|
||||||
DEV_PASS: ${{ secrets.DEV_FTP_PASSWORD }}
|
|
||||||
run: |
|
|
||||||
# Permission check: admin or maintain role required
|
|
||||||
ACTOR="${{ github.actor }}"
|
|
||||||
API_BASE="${GITEA_URL}/api/v1/repos/${GITEA_ORG}/${GITEA_REPO}"
|
|
||||||
|
|
||||||
PERMISSION=$(curl -sf -H "Authorization: token ${{ secrets.MOKOGITEA_TOKEN }}" \
|
|
||||||
"${API_BASE}/collaborators/${ACTOR}/permission" 2>/dev/null | \
|
|
||||||
python3 -c "import sys,json; print(json.load(sys.stdin).get('permission','read'))" 2>/dev/null || echo "read")
|
|
||||||
case "$PERMISSION" in
|
|
||||||
admin|maintain|write) ;;
|
|
||||||
*)
|
|
||||||
echo "Deploy denied: ${ACTOR} has '${PERMISSION}' — requires admin, maintain, or write"
|
|
||||||
exit 0
|
|
||||||
;;
|
|
||||||
esac
|
|
||||||
|
|
||||||
[ -z "$DEV_HOST" ] || [ -z "$DEV_PATH" ] && { echo "DEV FTP not configured — skipping SFTP"; exit 0; }
|
|
||||||
|
|
||||||
SOURCE_DIR="src"
|
|
||||||
[ ! -d "$SOURCE_DIR" ] && SOURCE_DIR="htdocs"
|
|
||||||
[ ! -d "$SOURCE_DIR" ] && exit 0
|
|
||||||
|
|
||||||
PORT="${DEV_PORT:-22}"
|
|
||||||
REMOTE="${DEV_PATH%/}"
|
|
||||||
[ -n "$DEV_SUFFIX" ] && REMOTE="${REMOTE}/${DEV_SUFFIX#/}"
|
|
||||||
|
|
||||||
printf '{"host":"%s","port":%s,"username":"%s","remotePath":"%s"' \
|
|
||||||
"$DEV_HOST" "$PORT" "$DEV_USER" "$REMOTE" > /tmp/sftp-config.json
|
|
||||||
if [ -n "$DEV_KEY" ]; then
|
|
||||||
echo "$DEV_KEY" > /tmp/deploy_key && chmod 600 /tmp/deploy_key
|
|
||||||
printf ',"privateKeyPath":"/tmp/deploy_key"}' >> /tmp/sftp-config.json
|
|
||||||
else
|
|
||||||
printf ',"password":"%s"}' "$DEV_PASS" >> /tmp/sftp-config.json
|
|
||||||
fi
|
|
||||||
|
|
||||||
PLATFORM=$(php ${MOKO_CLI}/platform_detect.php --path . 2>/dev/null || true)
|
|
||||||
if [ "$PLATFORM" = "waas-component" ] && [ -f "${MOKO_CLI}/../deploy/deploy-joomla.php" ]; then
|
|
||||||
php ${MOKO_CLI}/../deploy/deploy-joomla.php --path . --src-dir "$SOURCE_DIR" --config /tmp/sftp-config.json
|
|
||||||
elif [ -f "${MOKO_CLI}/../deploy/deploy-sftp.php" ]; then
|
|
||||||
php ${MOKO_CLI}/../deploy/deploy-sftp.php --path . --src-dir "$SOURCE_DIR" --config /tmp/sftp-config.json
|
|
||||||
fi
|
|
||||||
rm -f /tmp/deploy_key /tmp/sftp-config.json
|
|
||||||
echo "SFTP deploy to dev complete" >> $GITHUB_STEP_SUMMARY
|
|
||||||
|
|
||||||
- name: Summary
|
|
||||||
if: always()
|
|
||||||
run: |
|
|
||||||
VERSION="${{ steps.meta.outputs.version }}"
|
|
||||||
STABILITY="${{ steps.meta.outputs.stability }}"
|
|
||||||
DISPLAY="${VERSION}"
|
|
||||||
echo "## Update Server" >> $GITHUB_STEP_SUMMARY
|
|
||||||
echo "" >> $GITHUB_STEP_SUMMARY
|
|
||||||
echo "| Field | Value |" >> $GITHUB_STEP_SUMMARY
|
|
||||||
echo "|-------|-------|" >> $GITHUB_STEP_SUMMARY
|
|
||||||
echo "| Stability | \`${STABILITY}\` |" >> $GITHUB_STEP_SUMMARY
|
|
||||||
echo "| Version | \`${DISPLAY}\` |" >> $GITHUB_STEP_SUMMARY
|
|
||||||
@@ -1,7 +1,7 @@
|
|||||||
{
|
{
|
||||||
"metadata": {
|
"metadata": {
|
||||||
"generated_at": "2026-03-10T19:51:42.238134Z",
|
"generated_at": "2026-03-10T19:51:42.238134Z",
|
||||||
"repository": "MokoConsulting/moko-platform",
|
"repository": "MokoConsulting/mokocli",
|
||||||
"version": "1.0.0"
|
"version": "1.0.0"
|
||||||
},
|
},
|
||||||
"scripts": [
|
"scripts": [
|
||||||
|
|||||||
+15
-8
@@ -4,15 +4,28 @@ SPDX-License-Identifier: GPL-3.0-or-later
|
|||||||
FILE INFORMATION
|
FILE INFORMATION
|
||||||
DEFGROUP: MokoStandards.Root
|
DEFGROUP: MokoStandards.Root
|
||||||
INGROUP: MokoStandards
|
INGROUP: MokoStandards
|
||||||
REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
PATH: /CHANGELOG.md
|
PATH: /CHANGELOG.md
|
||||||
BRIEF: Release changelog
|
BRIEF: Release changelog
|
||||||
-->
|
-->
|
||||||
|
|
||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- `workflow_sync.php` — cascading workflow sync from Generic → platform templates → live repos based on manifest.platform
|
||||||
|
- `platform_detect.php` — auto-detect repo platform type (joomla/dolibarr/go/mcp/platform/generic) from file structure, optionally update manifest
|
||||||
|
- Version prefix support in `version_read.php` and `version_bump.php` — repos with `<version_prefix>` in manifest (e.g. MokoGitea: `1.26.1+moko.`) get prefix-aware version scanning and bumping
|
||||||
|
- Platform types: joomla, dolibarr, go, mcp, platform, generic
|
||||||
|
- Template-Go and Template-MCP repos created
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- `auto-release.yml` — patch branches (fix/*, patch/*, hotfix/*, bugfix/*) use `--bump none` (pre-release already bumped); feature/dev branches bump minor
|
||||||
|
- `pre-release.yml` — triggers on push to dev, fix/**, patch/**, hotfix/**, bugfix/**, alpha, beta, rc branches
|
||||||
|
- Version format standardized: `[prefix]XX.YY.ZZ` in source files, suffix (`-dev`, `-rc`) added by release system only
|
||||||
|
|
||||||
|
## [09.25.00] --- 2026-06-04
|
||||||
|
|
||||||
## [09.23] --- 2026-05-31
|
## [09.23] --- 2026-05-31
|
||||||
|
|
||||||
## [09.22] --- 2026-05-31
|
## [09.22] --- 2026-05-31
|
||||||
@@ -30,9 +43,3 @@ BRIEF: Release changelog
|
|||||||
## [09.21] --- 2026-05-30
|
## [09.21] --- 2026-05-30
|
||||||
|
|
||||||
## [09.20] --- 2026-05-30
|
## [09.20] --- 2026-05-30
|
||||||
|
|
||||||
## [09.19] --- 2026-05-30
|
|
||||||
|
|
||||||
## [09.18] --- 2026-05-30
|
|
||||||
|
|
||||||
## [09.17] --- 2026-05-30
|
|
||||||
|
|||||||
@@ -1,102 +0,0 @@
|
|||||||
# CLAUDE.md
|
|
||||||
|
|
||||||
This file provides guidance to Claude Code when working with this repository.
|
|
||||||
|
|
||||||
## Project Overview
|
|
||||||
|
|
||||||
**moko-platform** — Enterprise automation, validation, sync, and governance engine for all Moko Consulting repositories
|
|
||||||
|
|
||||||
| Field | Value |
|
|
||||||
|---|---|
|
|
||||||
| **Language** | PHP 8.1+ |
|
|
||||||
| **Default branch** | main |
|
|
||||||
| **License** | GPL-3.0-or-later |
|
|
||||||
| **Version** | 09.01.00 |
|
|
||||||
| **Wiki** | [moko-platform Wiki](https://git.mokoconsulting.tech/MokoConsulting/moko-platform/wiki) |
|
|
||||||
|
|
||||||
## Common Commands
|
|
||||||
|
|
||||||
```bash
|
|
||||||
composer install # Install PHP dependencies
|
|
||||||
php bin/moko health --path . # Run repo health check
|
|
||||||
php bin/moko check:syntax --path . # PHP syntax check
|
|
||||||
php bin/moko drift --org MokoConsulting # Scan for standards drift
|
|
||||||
php bin/moko dashboard --token $TOKEN -o dashboard.html # Generate client dashboard
|
|
||||||
|
|
||||||
# Code quality
|
|
||||||
php vendor/bin/phpcs --standard=phpcs.xml -n lib/ validate/ automation/ cli/
|
|
||||||
php vendor/bin/phpcbf --standard=phpcs.xml lib/ validate/ automation/ cli/
|
|
||||||
php vendor/bin/phpstan analyse -c phpstan.neon --memory-limit=512M
|
|
||||||
|
|
||||||
# Run all checks
|
|
||||||
composer check
|
|
||||||
```
|
|
||||||
|
|
||||||
## Architecture
|
|
||||||
|
|
||||||
### Directory Layout
|
|
||||||
|
|
||||||
| Directory | Purpose |
|
|
||||||
|-----------|---------|
|
|
||||||
| `cli/` | 32 standalone CLI tools (version, release, build, repo management) |
|
|
||||||
| `validate/` | 20 validation scripts (syntax, structure, manifests, drift) |
|
|
||||||
| `automation/` | 7 bulk operations (sync, push files, templates, cleanup) |
|
|
||||||
| `lib/Enterprise/` | Core library — CliFramework, ApiClient, adapters, validators |
|
|
||||||
| `lib/Enterprise/Plugins/` | 11 platform plugins (Joomla, Dolibarr, Node.js, Python, etc.) |
|
|
||||||
| `deploy/` | SFTP deployment scripts (Joomla, Dolibarr, health checks) |
|
|
||||||
| `templates/` | Universal templates, configs, governance schema |
|
|
||||||
| `.mokogitea/workflows/` | CI/CD workflows (Gitea Actions) |
|
|
||||||
| `bin/moko` | Unified CLI dispatcher — runs any tool via `php bin/moko <command>` |
|
|
||||||
|
|
||||||
### CLI Framework
|
|
||||||
|
|
||||||
All CLI tools extend `MokoEnterprise\CliFramework` (defined in `lib/Enterprise/CliFramework.php`).
|
|
||||||
|
|
||||||
Pattern for new tools:
|
|
||||||
```php
|
|
||||||
class MyTool extends CliFramework {
|
|
||||||
protected function configure(): void {
|
|
||||||
$this->setDescription('What this tool does');
|
|
||||||
$this->addArgument('--name', 'Description', 'default');
|
|
||||||
}
|
|
||||||
protected function run(): int {
|
|
||||||
$name = $this->getArgument('--name');
|
|
||||||
// ... business logic ...
|
|
||||||
return 0;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
$app = new MyTool();
|
|
||||||
exit($app->execute());
|
|
||||||
```
|
|
||||||
|
|
||||||
Built-in flags: `--help`, `--verbose`, `--quiet`, `--dry-run`
|
|
||||||
|
|
||||||
### Platform Adapters
|
|
||||||
|
|
||||||
Git operations are abstracted via `GitPlatformAdapter` interface:
|
|
||||||
- `MokoGiteaAdapter` — for git.mokoconsulting.tech (primary)
|
|
||||||
- `GitHubAdapter` — for github.com mirrors
|
|
||||||
|
|
||||||
### Plugin System
|
|
||||||
|
|
||||||
Platform-specific logic lives in `lib/Enterprise/Plugins/`. Each plugin implements `ProjectPluginInterface` with methods for health checks, validation, build commands, and config schemas.
|
|
||||||
|
|
||||||
## Code Quality
|
|
||||||
|
|
||||||
| Tool | Level | Config |
|
|
||||||
|------|-------|--------|
|
|
||||||
| PHPCS | PSR-12 (errors only) | `phpcs.xml` |
|
|
||||||
| PHPStan | Level 2 | `phpstan.neon` |
|
|
||||||
|
|
||||||
PHPStan runs with `--memory-limit=512M` due to large codebase. CI enforces PHPCS errors; PHPStan is advisory (`continue-on-error`).
|
|
||||||
|
|
||||||
## Rules
|
|
||||||
|
|
||||||
- **Workflow directory**: `.mokogitea/` (not `.gitea/` or `.github/`)
|
|
||||||
- **Never commit** `.claude/`, `.mcp.json`, `TODO.md`, or `*.min.css`/`*.min.js`
|
|
||||||
- **Attribution**: use `Authored-by: Moko Consulting` in commits
|
|
||||||
- **Branch strategy**: develop on `dev`, merge to `main` for release
|
|
||||||
- **Minification**: handled at build time (CI) and runtime (MokoMinifyHelper for Joomla templates)
|
|
||||||
- **Wiki**: documentation lives in the Gitea wiki, not in `docs/` files
|
|
||||||
- **New CLI tools**: extend `CliFramework`, not `CLIApp` (legacy)
|
|
||||||
- **After adding a CLI tool**: register it in `bin/moko` COMMAND_MAP
|
|
||||||
+161
-161
@@ -1,161 +1,161 @@
|
|||||||
# Contributing to Moko Consulting Projects
|
# Contributing to Moko Consulting Projects
|
||||||
|
|
||||||
Thank you for your interest in contributing. All Moko Consulting repositories follow this universal workflow and version policy.
|
Thank you for your interest in contributing. All Moko Consulting repositories follow this universal workflow and version policy.
|
||||||
|
|
||||||
## Branching Workflow
|
## Branching Workflow
|
||||||
|
|
||||||
```
|
```
|
||||||
feature/* ──PR──> dev ──draft PR──> (renamed to rc) ──merge──> main
|
feature/* ──PR──> dev ──draft PR──> (renamed to rc) ──merge──> main
|
||||||
```
|
```
|
||||||
|
|
||||||
### Step by step
|
### Step by step
|
||||||
|
|
||||||
1. **Create a feature branch** from `dev`:
|
1. **Create a feature branch** from `dev`:
|
||||||
```bash
|
```bash
|
||||||
git checkout dev && git pull
|
git checkout dev && git pull
|
||||||
git checkout -b feature/my-change
|
git checkout -b feature/my-change
|
||||||
```
|
```
|
||||||
|
|
||||||
2. **Work and commit** on your feature branch. Push to origin.
|
2. **Work and commit** on your feature branch. Push to origin.
|
||||||
|
|
||||||
3. **Open a PR**: `feature/my-change` → `dev`. After review and checks, merge it.
|
3. **Open a PR**: `feature/my-change` → `dev`. After review and checks, merge it.
|
||||||
|
|
||||||
4. **When ready for release**, open a **draft PR**: `dev` → `main`.
|
4. **When ready for release**, open a **draft PR**: `dev` → `main`.
|
||||||
- This automatically renames the source branch to `rc` (release candidate)
|
- This automatically renames the source branch to `rc` (release candidate)
|
||||||
- An RC pre-release is built and uploaded
|
- An RC pre-release is built and uploaded
|
||||||
|
|
||||||
5. **Alpha and beta branches** are created by manually renaming the branch before the RC stage:
|
5. **Alpha and beta branches** are created by manually renaming the branch before the RC stage:
|
||||||
- Rename `dev` to `alpha` for early testing → alpha pre-release is built
|
- Rename `dev` to `alpha` for early testing → alpha pre-release is built
|
||||||
- Rename `alpha` to `beta` for feature-complete testing → beta pre-release is built
|
- Rename `alpha` to `beta` for feature-complete testing → beta pre-release is built
|
||||||
- When the draft PR is created, the branch is renamed to `rc`
|
- When the draft PR is created, the branch is renamed to `rc`
|
||||||
|
|
||||||
6. **Once PR checks pass** on the `rc` branch, mark the PR as ready and merge to `main`.
|
6. **Once PR checks pass** on the `rc` branch, mark the PR as ready and merge to `main`.
|
||||||
|
|
||||||
7. **Merging to main** triggers the stable release pipeline:
|
7. **Merging to main** triggers the stable release pipeline:
|
||||||
- Minor version bump (e.g., `02.09.xx` → `02.10.00`)
|
- Minor version bump (e.g., `02.09.xx` → `02.10.00`)
|
||||||
- Stability suffix stripped (clean version)
|
- Stability suffix stripped (clean version)
|
||||||
- Gitea release created with ZIP/tar.gz packages
|
- Gitea release created with ZIP/tar.gz packages
|
||||||
- `updates.xml` updated (Joomla extensions)
|
- `updates.xml` updated (Joomla extensions)
|
||||||
- `dev` branch recreated from `main`
|
- `dev` branch recreated from `main`
|
||||||
|
|
||||||
### Branch summary
|
### Branch summary
|
||||||
|
|
||||||
| Branch | Purpose | Created by |
|
| Branch | Purpose | Created by |
|
||||||
|--------|---------|-----------|
|
|--------|---------|-----------|
|
||||||
| `feature/*` | New features and fixes | Developer |
|
| `feature/*` | New features and fixes | Developer |
|
||||||
| `dev` | Integration branch | Auto-recreated after release |
|
| `dev` | Integration branch | Auto-recreated after release |
|
||||||
| `alpha` | Alpha pre-release testing | Manual rename from `dev` |
|
| `alpha` | Alpha pre-release testing | Manual rename from `dev` |
|
||||||
| `beta` | Beta pre-release testing | Manual rename from `alpha` |
|
| `beta` | Beta pre-release testing | Manual rename from `alpha` |
|
||||||
| `rc` | Release candidate | Auto-renamed on draft PR to main |
|
| `rc` | Release candidate | Auto-renamed on draft PR to main |
|
||||||
| `main` | Stable releases | Protected, merge only |
|
| `main` | Stable releases | Protected, merge only |
|
||||||
| `version/XX.YY.ZZ` | Archived release snapshots | Auto-created by CI |
|
| `version/XX.YY.ZZ` | Archived release snapshots | Auto-created by CI |
|
||||||
|
|
||||||
### Protected branches
|
### Protected branches
|
||||||
|
|
||||||
| Branch | Direct push | Merge via |
|
| Branch | Direct push | Merge via |
|
||||||
|--------|------------|-----------|
|
|--------|------------|-----------|
|
||||||
| `main` | Blocked (CI bot whitelisted) | PR merge only |
|
| `main` | Blocked (CI bot whitelisted) | PR merge only |
|
||||||
| `dev` | Blocked (CI bot whitelisted) | PR merge from feature/* |
|
| `dev` | Blocked (CI bot whitelisted) | PR merge from feature/* |
|
||||||
| `rc` | Blocked (CI bot whitelisted) | Auto-created on draft PR |
|
| `rc` | Blocked (CI bot whitelisted) | Auto-created on draft PR |
|
||||||
| `alpha` | Blocked (CI bot whitelisted) | Manual rename |
|
| `alpha` | Blocked (CI bot whitelisted) | Manual rename |
|
||||||
| `beta` | Blocked (CI bot whitelisted) | Manual rename |
|
| `beta` | Blocked (CI bot whitelisted) | Manual rename |
|
||||||
| `feature/*` | Open | N/A (source branch) |
|
| `feature/*` | Open | N/A (source branch) |
|
||||||
|
|
||||||
## Version Policy
|
## Version Policy
|
||||||
|
|
||||||
### Format
|
### Format
|
||||||
|
|
||||||
All versions use `XX.YY.ZZ` — three two-digit segments, zero-padded:
|
All versions use `XX.YY.ZZ` — three two-digit segments, zero-padded:
|
||||||
|
|
||||||
- **XX** — Major version (breaking changes)
|
- **XX** — Major version (breaking changes)
|
||||||
- **YY** — Minor version (new features, bumped on release to main)
|
- **YY** — Minor version (new features, bumped on release to main)
|
||||||
- **ZZ** — Patch version (auto-incremented on every push to dev/feature branches)
|
- **ZZ** — Patch version (auto-incremented on every push to dev/feature branches)
|
||||||
|
|
||||||
Rollover: patch `99` → `00` increments minor; minor `99` → `00` increments major.
|
Rollover: patch `99` → `00` increments minor; minor `99` → `00` increments major.
|
||||||
|
|
||||||
### Stability suffixes
|
### Stability suffixes
|
||||||
|
|
||||||
Each branch appends a suffix to indicate stability:
|
Each branch appends a suffix to indicate stability:
|
||||||
|
|
||||||
| Branch | Suffix | Example |
|
| Branch | Suffix | Example |
|
||||||
|--------|--------|---------|
|
|--------|--------|---------|
|
||||||
| `main` | (none) | `02.09.00` |
|
| `main` | (none) | `02.09.00` |
|
||||||
| `dev` | `-dev` | `02.09.01-dev` |
|
| `dev` | `-dev` | `02.09.01-dev` |
|
||||||
| `feature/*` | `-dev` | `02.09.01-dev` |
|
| `feature/*` | `-dev` | `02.09.01-dev` |
|
||||||
| `alpha` | `-alpha` | `02.09.01-alpha` |
|
| `alpha` | `-alpha` | `02.09.01-alpha` |
|
||||||
| `beta` | `-beta` | `02.09.01-beta` |
|
| `beta` | `-beta` | `02.09.01-beta` |
|
||||||
| `rc` | `-rc` | `02.09.01-rc` |
|
| `rc` | `-rc` | `02.09.01-rc` |
|
||||||
|
|
||||||
### Auto version bump
|
### Auto version bump
|
||||||
|
|
||||||
On every push to `dev`, `feature/*`, or `patch/*`:
|
On every push to `dev`, `feature/*`, or `patch/*`:
|
||||||
|
|
||||||
1. Patch version incremented
|
1. Patch version incremented
|
||||||
2. Stability suffix `-dev` applied
|
2. Stability suffix `-dev` applied
|
||||||
3. All version-bearing files updated (manifests, CHANGELOG, PHP headers, etc.)
|
3. All version-bearing files updated (manifests, CHANGELOG, PHP headers, etc.)
|
||||||
4. Commit created with `[skip ci]` to avoid loops
|
4. Commit created with `[skip ci]` to avoid loops
|
||||||
|
|
||||||
### Release version flow
|
### Release version flow
|
||||||
|
|
||||||
Version bumps happen at specific release events:
|
Version bumps happen at specific release events:
|
||||||
|
|
||||||
| Event | Bump | Example |
|
| Event | Bump | Example |
|
||||||
|-------|------|---------|
|
|-------|------|---------|
|
||||||
| Feature merged to dev | Patch bump after dev release | `02.09.01-dev` → release → `02.09.02-dev` |
|
| Feature merged to dev | Patch bump after dev release | `02.09.01-dev` → release → `02.09.02-dev` |
|
||||||
| Dev promoted to RC | Minor bump | `02.09.02-dev` → `02.10.00-rc` |
|
| Dev promoted to RC | Minor bump | `02.09.02-dev` → `02.10.00-rc` |
|
||||||
| RC merged to main | Minor bump | `02.10.00-rc` → `02.11.00` (stable) |
|
| RC merged to main | Minor bump | `02.10.00-rc` → `02.11.00` (stable) |
|
||||||
| Dev recreated from main | Patch bump | `02.11.00` → `02.11.01-dev` |
|
| Dev recreated from main | Patch bump | `02.11.00` → `02.11.01-dev` |
|
||||||
|
|
||||||
### Release stream copies
|
### Release stream copies
|
||||||
|
|
||||||
When a higher-stability release is published, copies are created for all lesser streams with the same base version:
|
When a higher-stability release is published, copies are created for all lesser streams with the same base version:
|
||||||
|
|
||||||
- **RC `02.10.00-rc`** also creates: `02.10.00-dev`, `02.10.00-alpha`, `02.10.00-beta`
|
- **RC `02.10.00-rc`** also creates: `02.10.00-dev`, `02.10.00-alpha`, `02.10.00-beta`
|
||||||
- **Stable `02.11.00`** also creates: `02.11.00-dev`, `02.11.00-alpha`, `02.11.00-beta`, `02.11.00-rc`
|
- **Stable `02.11.00`** also creates: `02.11.00-dev`, `02.11.00-alpha`, `02.11.00-beta`, `02.11.00-rc`
|
||||||
|
|
||||||
This ensures Joomla sites on ANY stability channel see the update (Joomla only shows versions higher than what's installed).
|
This ensures Joomla sites on ANY stability channel see the update (Joomla only shows versions higher than what's installed).
|
||||||
|
|
||||||
### Version files
|
### Version files
|
||||||
|
|
||||||
The version tools update all files containing version stamps:
|
The version tools update all files containing version stamps:
|
||||||
|
|
||||||
- `.mokogitea/manifest.xml` (canonical source)
|
- `.mokogitea/manifest.xml` (canonical source)
|
||||||
- Joomla XML manifests (`<version>` tag)
|
- Joomla XML manifests (`<version>` tag)
|
||||||
- `README.md`, `CHANGELOG.md` (`VERSION:` pattern)
|
- `README.md`, `CHANGELOG.md` (`VERSION:` pattern)
|
||||||
- `package.json`, `pyproject.toml`
|
- `package.json`, `pyproject.toml`
|
||||||
- Any text file with a `VERSION: XX.YY.ZZ` label
|
- Any text file with a `VERSION: XX.YY.ZZ` label
|
||||||
|
|
||||||
Files synced from other repos (with a `# REPO:` header) are not touched.
|
Files synced from other repos (with a `# REPO:` header) are not touched.
|
||||||
|
|
||||||
## Code Standards
|
## Code Standards
|
||||||
|
|
||||||
- **PHP**: PSR-12, tabs for indentation
|
- **PHP**: PSR-12, tabs for indentation
|
||||||
- **Copyright**: all files must include the Moko Consulting copyright header
|
- **Copyright**: all files must include the Moko Consulting copyright header
|
||||||
- **License**: SPDX identifier `GPL-3.0-or-later` (or as specified per repo)
|
- **License**: SPDX identifier `GPL-3.0-or-later` (or as specified per repo)
|
||||||
- **Attribution**: use `Authored-by: Moko Consulting` in commits, not individual names
|
- **Attribution**: use `Authored-by: Moko Consulting` in commits, not individual names
|
||||||
|
|
||||||
## Commit Messages
|
## Commit Messages
|
||||||
|
|
||||||
Use conventional commit format:
|
Use conventional commit format:
|
||||||
|
|
||||||
```
|
```
|
||||||
type(scope): short description
|
type(scope): short description
|
||||||
|
|
||||||
Optional body with context.
|
Optional body with context.
|
||||||
|
|
||||||
Authored-by: Moko Consulting
|
Authored-by: Moko Consulting
|
||||||
```
|
```
|
||||||
|
|
||||||
Types: `feat`, `fix`, `chore`, `docs`, `style`, `refactor`, `test`, `ci`
|
Types: `feat`, `fix`, `chore`, `docs`, `style`, `refactor`, `test`, `ci`
|
||||||
|
|
||||||
Special flags in commit messages:
|
Special flags in commit messages:
|
||||||
- `[skip ci]` — skip all CI workflows
|
- `[skip ci]` — skip all CI workflows
|
||||||
- `[skip bump]` — skip auto version bump only
|
- `[skip bump]` — skip auto version bump only
|
||||||
|
|
||||||
## Reporting Issues
|
## Reporting Issues
|
||||||
|
|
||||||
Use the repository's issue tracker with the appropriate template.
|
Use the repository's issue tracker with the appropriate template.
|
||||||
|
|
||||||
---
|
---
|
||||||
|
|
||||||
*Moko Consulting <hello@mokoconsulting.tech>*
|
*Moko Consulting <hello@mokoconsulting.tech>*
|
||||||
|
|||||||
+4
-4
@@ -2,16 +2,16 @@
|
|||||||
Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
FILE INFORMATION
|
FILE INFORMATION
|
||||||
DEFGROUP: MokoPlatform.Root
|
DEFGROUP: MokoCLI.Root
|
||||||
INGROUP: MokoPlatform
|
INGROUP: MokoCLI
|
||||||
REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
PATH: /PLUGIN_SCRIPTS.md
|
PATH: /PLUGIN_SCRIPTS.md
|
||||||
BRIEF: Plugin system CLI documentation
|
BRIEF: Plugin system CLI documentation
|
||||||
-->
|
-->
|
||||||
|
|
||||||
# Plugin System CLI Scripts
|
# Plugin System CLI Scripts
|
||||||
|
|
||||||
Command-line scripts for validating, health checking, and managing projects using the moko-platform plugin system.
|
Command-line scripts for validating, health checking, and managing projects using the MokoCLI plugin system.
|
||||||
|
|
||||||
## Available Scripts
|
## Available Scripts
|
||||||
|
|
||||||
|
|||||||
@@ -2,19 +2,19 @@
|
|||||||
Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
FILE INFORMATION
|
FILE INFORMATION
|
||||||
DEFGROUP: MokoPlatform.Root
|
DEFGROUP: MokoCLI.Root
|
||||||
INGROUP: MokoPlatform
|
INGROUP: MokoCLI
|
||||||
REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
PATH: /README.md
|
PATH: /README.md
|
||||||
VERSION: 09.24.00
|
VERSION: 09.25.05
|
||||||
BRIEF: Project overview and documentation
|
BRIEF: Project overview and documentation
|
||||||
-->
|
-->
|
||||||
|
|
||||||
# moko-platform Enterprise API
|
# MokoCLI Enterprise API
|
||||||
|
|
||||||
  
|
  
|
||||||
|
|
||||||
PHP implementation of moko-platform — enterprise standards, automation framework, workflow templates, and bulk sync tooling.
|
PHP implementation of MokoCLI — enterprise standards, automation framework, workflow templates, and bulk sync tooling.
|
||||||
|
|
||||||
> **Primary platform**: [Gitea — git.mokoconsulting.tech](https://git.mokoconsulting.tech/MokoConsulting/MokoStandards-API)
|
> **Primary platform**: [Gitea — git.mokoconsulting.tech](https://git.mokoconsulting.tech/MokoConsulting/MokoStandards-API)
|
||||||
> **Backup mirror**: [GitHub](https://github.com/MokoConsulting/MokoStandards-API) *(read-only mirror)*
|
> **Backup mirror**: [GitHub](https://github.com/MokoConsulting/MokoStandards-API) *(read-only mirror)*
|
||||||
|
|||||||
+3
-3
@@ -2,9 +2,9 @@
|
|||||||
Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
FILE INFORMATION
|
FILE INFORMATION
|
||||||
DEFGROUP: MokoPlatform.Index
|
DEFGROUP: MokoCLI.Index
|
||||||
INGROUP: MokoPlatform.Analysis
|
INGROUP: MokoCLI.Analysis
|
||||||
REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
PATH: /analysis/index.md
|
PATH: /analysis/index.md
|
||||||
BRIEF: Analysis directory index
|
BRIEF: Analysis directory index
|
||||||
-->
|
-->
|
||||||
|
|||||||
@@ -9,9 +9,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform.Scripts
|
* INGROUP: MokoCLI.Scripts
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/bulk_joomla_template.php
|
* PATH: /automation/bulk_joomla_template.php
|
||||||
* BRIEF: Bulk scaffold and sync Joomla template repositories
|
* BRIEF: Bulk scaffold and sync Joomla template repositories
|
||||||
*
|
*
|
||||||
@@ -42,7 +42,7 @@ use MokoEnterprise\{
|
|||||||
*
|
*
|
||||||
* Provides three operations for Joomla template projects:
|
* Provides three operations for Joomla template projects:
|
||||||
* --scaffold: Create a new template repository with the full directory structure
|
* --scaffold: Create a new template repository with the full directory structure
|
||||||
* --sync: Push moko-platform files to existing template repositories
|
* --sync: Push MokoCLI files to existing template repositories
|
||||||
* --list: List all repositories tagged as joomla-template
|
* --list: List all repositories tagged as joomla-template
|
||||||
*
|
*
|
||||||
* Works with both GitHub and Gitea via the PlatformAdapterFactory.
|
* Works with both GitHub and Gitea via the PlatformAdapterFactory.
|
||||||
@@ -318,7 +318,7 @@ class BulkJoomlaTemplate extends CliFramework
|
|||||||
$name,
|
$name,
|
||||||
$path,
|
$path,
|
||||||
$content,
|
$content,
|
||||||
"chore: update {$path} from moko-platform",
|
"chore: update {$path} from MokoCLI",
|
||||||
$existingSha,
|
$existingSha,
|
||||||
$branch
|
$branch
|
||||||
);
|
);
|
||||||
|
|||||||
+42
-42
@@ -9,9 +9,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform.Scripts
|
* INGROUP: MokoCLI.Scripts
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/bulk_sync.php
|
* PATH: /automation/bulk_sync.php
|
||||||
* BRIEF: Enterprise-grade bulk repository synchronization
|
* BRIEF: Enterprise-grade bulk repository synchronization
|
||||||
*/
|
*/
|
||||||
@@ -42,7 +42,7 @@ use MokoEnterprise\{
|
|||||||
/**
|
/**
|
||||||
* Bulk Repository Synchronization Tool
|
* Bulk Repository Synchronization Tool
|
||||||
*
|
*
|
||||||
* Synchronizes moko-platform files across multiple repositories using
|
* Synchronizes MokoCLI files across multiple repositories using
|
||||||
* the Enterprise library for robust, audited operations.
|
* the Enterprise library for robust, audited operations.
|
||||||
*/
|
*/
|
||||||
class BulkSync extends CliFramework
|
class BulkSync extends CliFramework
|
||||||
@@ -95,7 +95,7 @@ class BulkSync extends CliFramework
|
|||||||
*/
|
*/
|
||||||
protected function run(): int
|
protected function run(): int
|
||||||
{
|
{
|
||||||
$this->log("🚀 moko-platform Bulk Synchronization v" . self::VERSION, 'INFO');
|
$this->log("🚀 MokoCLI Bulk Synchronization v" . self::VERSION, 'INFO');
|
||||||
|
|
||||||
// Initialize enterprise components
|
// Initialize enterprise components
|
||||||
if (!$this->initializeComponents()) {
|
if (!$this->initializeComponents()) {
|
||||||
@@ -180,7 +180,7 @@ class BulkSync extends CliFramework
|
|||||||
$results['health'] = $this->runHealthChecksAll($org, $repositories);
|
$results['health'] = $this->runHealthChecksAll($org, $repositories);
|
||||||
}
|
}
|
||||||
|
|
||||||
// Create/update tracking issue in moko-platform
|
// Create/update tracking issue in MokoCLI
|
||||||
$this->createSyncIssue($org, $results);
|
$this->createSyncIssue($org, $results);
|
||||||
|
|
||||||
// Create/update a failure issue when any repos failed
|
// Create/update a failure issue when any repos failed
|
||||||
@@ -244,7 +244,7 @@ class BulkSync extends CliFramework
|
|||||||
* Filter repositories based on include/exclude lists
|
* Filter repositories based on include/exclude lists
|
||||||
*/
|
*/
|
||||||
/** Repositories that are permanently excluded from bulk sync. */
|
/** Repositories that are permanently excluded from bulk sync. */
|
||||||
private const ALWAYS_EXCLUDE = ['moko-platform', '.github-private'];
|
private const ALWAYS_EXCLUDE = ['MokoCLI', '.github-private'];
|
||||||
|
|
||||||
private function filterRepositories(array $repositories, array $include, array $exclude): array
|
private function filterRepositories(array $repositories, array $include, array $exclude): array
|
||||||
{
|
{
|
||||||
@@ -426,7 +426,7 @@ class BulkSync extends CliFramework
|
|||||||
$this->log("", 'ERROR');
|
$this->log("", 'ERROR');
|
||||||
$this->log("Required Implementation:", 'ERROR');
|
$this->log("Required Implementation:", 'ERROR');
|
||||||
$this->log(" 1. Clone/fetch target repository", 'ERROR');
|
$this->log(" 1. Clone/fetch target repository", 'ERROR');
|
||||||
$this->log(" 2. Apply file updates based on moko-platform configuration", 'ERROR');
|
$this->log(" 2. Apply file updates based on MokoCLI configuration", 'ERROR');
|
||||||
$this->log(" 3. Create pull request with changes", 'ERROR');
|
$this->log(" 3. Create pull request with changes", 'ERROR');
|
||||||
$this->log(" 4. Handle merge conflicts and validation", 'ERROR');
|
$this->log(" 4. Handle merge conflicts and validation", 'ERROR');
|
||||||
$this->log("", 'ERROR');
|
$this->log("", 'ERROR');
|
||||||
@@ -837,7 +837,7 @@ class BulkSync extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Ensure all standard moko-platform labels exist on a target repository.
|
* Ensure all standard MokoCLI labels exist on a target repository.
|
||||||
*
|
*
|
||||||
* Fetches existing labels first (GET) and only POSTs the ones that are
|
* Fetches existing labels first (GET) and only POSTs the ones that are
|
||||||
* missing. This avoids the 422 "already exists" responses that would
|
* missing. This avoids the 422 "already exists" responses that would
|
||||||
@@ -872,7 +872,7 @@ class BulkSync extends CliFramework
|
|||||||
|
|
||||||
// Workflow / Process
|
// Workflow / Process
|
||||||
['automation', '8B4513', 'Automated processes or scripts'],
|
['automation', '8B4513', 'Automated processes or scripts'],
|
||||||
['moko-platform', 'B60205', 'moko-platform compliance'],
|
['MokoCLI', 'B60205', 'MokoCLI compliance'],
|
||||||
['needs-review', 'FBCA04', 'Awaiting code review'],
|
['needs-review', 'FBCA04', 'Awaiting code review'],
|
||||||
['work-in-progress', 'D93F0B', 'Work in progress, not ready for merge'],
|
['work-in-progress', 'D93F0B', 'Work in progress, not ready for merge'],
|
||||||
['breaking-change', 'D73A4A', 'Breaking API or functionality change'],
|
['breaking-change', 'D73A4A', 'Breaking API or functionality change'],
|
||||||
@@ -912,8 +912,8 @@ class BulkSync extends CliFramework
|
|||||||
['health: poor', 'FF6B6B', 'Health score below 50'],
|
['health: poor', 'FF6B6B', 'Health score below 50'],
|
||||||
|
|
||||||
// Sync / Automation (used by bulk_sync, scan_drift, check_repo_health)
|
// Sync / Automation (used by bulk_sync, scan_drift, check_repo_health)
|
||||||
['standards-update', 'B60205', 'moko-platform sync update'],
|
['standards-update', 'B60205', 'MokoCLI sync update'],
|
||||||
['standards-drift', 'FBCA04', 'Repository drifted from moko-platform'],
|
['standards-drift', 'FBCA04', 'Repository drifted from MokoCLI'],
|
||||||
['sync-report', '0075CA', 'Bulk sync run report'],
|
['sync-report', '0075CA', 'Bulk sync run report'],
|
||||||
['sync-failure', 'D73A4A', 'Bulk sync failure requiring attention'],
|
['sync-failure', 'D73A4A', 'Bulk sync failure requiring attention'],
|
||||||
['push-failure', 'D73A4A', 'File push failure requiring attention'],
|
['push-failure', 'D73A4A', 'File push failure requiring attention'],
|
||||||
@@ -925,10 +925,10 @@ class BulkSync extends CliFramework
|
|||||||
['type: version', '0E8A16', 'Version-related change'],
|
['type: version', '0E8A16', 'Version-related change'],
|
||||||
];
|
];
|
||||||
|
|
||||||
// Quick check: if the repo already has the 'moko-platform' label, it was
|
// Quick check: if the repo already has the 'MokoCLI' label, it was
|
||||||
// provisioned previously — skip the expensive full label provisioning.
|
// provisioned previously — skip the expensive full label provisioning.
|
||||||
try {
|
try {
|
||||||
$probe = $this->api->get("/repos/{$org}/{$repo}/labels/moko-platform");
|
$probe = $this->api->get("/repos/{$org}/{$repo}/labels/MokoCLI");
|
||||||
if (!empty($probe['name'])) {
|
if (!empty($probe['name'])) {
|
||||||
return; // already provisioned
|
return; // already provisioned
|
||||||
}
|
}
|
||||||
@@ -1024,7 +1024,7 @@ class BulkSync extends CliFramework
|
|||||||
*/
|
*/
|
||||||
private function updateOpenBranches(string $org, string $repo): void
|
private function updateOpenBranches(string $org, string $repo): void
|
||||||
{
|
{
|
||||||
$syncBranchPrefix = 'chore/sync-moko-platform-';
|
$syncBranchPrefix = 'chore/sync-MokoCLI-';
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$defaultBranch = 'main';
|
$defaultBranch = 'main';
|
||||||
@@ -1055,7 +1055,7 @@ class BulkSync extends CliFramework
|
|||||||
$this->api->post("/repos/{$org}/{$repo}/merges", [
|
$this->api->post("/repos/{$org}/{$repo}/merges", [
|
||||||
'base' => $branch,
|
'base' => $branch,
|
||||||
'head' => $defaultBranch,
|
'head' => $defaultBranch,
|
||||||
'commit_message' => "chore: merge {$defaultBranch} into {$branch} (moko-platform sync)",
|
'commit_message' => "chore: merge {$defaultBranch} into {$branch} (MokoCLI sync)",
|
||||||
]);
|
]);
|
||||||
$this->log(" 🔀 Merged {$defaultBranch} → {$branch} (PR #{$prNum})", 'INFO');
|
$this->log(" 🔀 Merged {$defaultBranch} → {$branch} (PR #{$prNum})", 'INFO');
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
@@ -1076,7 +1076,7 @@ class BulkSync extends CliFramework
|
|||||||
|
|
||||||
/**
|
/**
|
||||||
* Records which sync run touched the repo, the PR number, and the
|
* Records which sync run touched the repo, the PR number, and the
|
||||||
* moko-platform version that was applied — giving each repo a clear audit
|
* MokoCLI version that was applied — giving each repo a clear audit
|
||||||
* trail of what was changed and why.
|
* trail of what was changed and why.
|
||||||
*/
|
*/
|
||||||
/**
|
/**
|
||||||
@@ -1119,16 +1119,16 @@ class BulkSync extends CliFramework
|
|||||||
$minor = self::VERSION_MINOR;
|
$minor = self::VERSION_MINOR;
|
||||||
$force = isset($this->options['force']) ? ' *(--force)*' : '';
|
$force = isset($this->options['force']) ? ' *(--force)*' : '';
|
||||||
$prLink = $this->adapter->getPullRequestWebUrl($org, $repo, $prNumber);
|
$prLink = $this->adapter->getPullRequestWebUrl($org, $repo, $prNumber);
|
||||||
$source = $this->adapter->getRepoWebUrl($org, 'moko-platform');
|
$source = $this->adapter->getRepoWebUrl($org, 'MokoCLI');
|
||||||
$branchName = 'chore/sync-moko-platform-v' . $minor;
|
$branchName = 'chore/sync-MokoCLI-v' . $minor;
|
||||||
$branchLink = $this->adapter->getBranchWebUrl($org, $repo, $branchName);
|
$branchLink = $this->adapter->getBranchWebUrl($org, $repo, $branchName);
|
||||||
|
|
||||||
$title = "chore: moko-platform v{$minor} sync tracking";
|
$title = "chore: MokoCLI v{$minor} sync tracking";
|
||||||
|
|
||||||
$body = <<<MD
|
$body = <<<MD
|
||||||
## moko-platform Sync Applied
|
## MokoCLI Sync Applied
|
||||||
|
|
||||||
A moko-platform bulk sync run has updated files in this repository.
|
A MokoCLI bulk sync run has updated files in this repository.
|
||||||
|
|
||||||
| Field | Value |
|
| Field | Value |
|
||||||
|-------|-------|
|
|-------|-------|
|
||||||
@@ -1144,13 +1144,13 @@ class BulkSync extends CliFramework
|
|||||||
Protected files (README, CHANGELOG, GOVERNANCE, etc.) were not overwritten.
|
Protected files (README, CHANGELOG, GOVERNANCE, etc.) were not overwritten.
|
||||||
|
|
||||||
---
|
---
|
||||||
*Updated automatically by [moko-platform]({$source}) `bulk_sync.php`*
|
*Updated automatically by [MokoCLI]({$source}) `bulk_sync.php`*
|
||||||
MD;
|
MD;
|
||||||
|
|
||||||
// Dedent heredoc
|
// Dedent heredoc
|
||||||
$body = preg_replace('/^ /m', '', $body);
|
$body = preg_replace('/^ /m', '', $body);
|
||||||
|
|
||||||
$labelNames = ['standards-update', 'moko-platform', 'type: chore', 'automation'];
|
$labelNames = ['standards-update', 'MokoCLI', 'type: chore', 'automation'];
|
||||||
$labels = $this->resolveLabelIds($org, $repo, $labelNames);
|
$labels = $this->resolveLabelIds($org, $repo, $labelNames);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
@@ -1213,7 +1213,7 @@ class BulkSync extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create a tracking issue in moko-platform for this sync run.
|
* Create a tracking issue in MokoCLI for this sync run.
|
||||||
*/
|
*/
|
||||||
private function createSyncIssue(string $org, array $results): void
|
private function createSyncIssue(string $org, array $results): void
|
||||||
{
|
{
|
||||||
@@ -1232,7 +1232,7 @@ class BulkSync extends CliFramework
|
|||||||
$issues = $results['issues'] ?? [];
|
$issues = $results['issues'] ?? [];
|
||||||
|
|
||||||
// Stable title — no timestamp so repeated runs update a single issue
|
// Stable title — no timestamp so repeated runs update a single issue
|
||||||
$title = "sync: moko-platform v" . self::VERSION_MINOR . " bulk sync report";
|
$title = "sync: MokoCLI v" . self::VERSION_MINOR . " bulk sync report";
|
||||||
|
|
||||||
$protection = $results['protection'] ?? [];
|
$protection = $results['protection'] ?? [];
|
||||||
$hasProtect = !empty($protection);
|
$hasProtect = !empty($protection);
|
||||||
@@ -1281,7 +1281,7 @@ class BulkSync extends CliFramework
|
|||||||
: "|---|---|---|---|";
|
: "|---|---|---|---|";
|
||||||
|
|
||||||
$body = <<<MD
|
$body = <<<MD
|
||||||
## moko-platform Bulk Sync Report
|
## MokoCLI Bulk Sync Report
|
||||||
|
|
||||||
**Organisation:** `{$org}`
|
**Organisation:** `{$org}`
|
||||||
**Triggered:** {$now}{$force}
|
**Triggered:** {$now}{$force}
|
||||||
@@ -1301,7 +1301,7 @@ class BulkSync extends CliFramework
|
|||||||
|
|
||||||
try {
|
try {
|
||||||
// Search for existing issue by label — any state so we can reopen closed ones
|
// Search for existing issue by label — any state so we can reopen closed ones
|
||||||
$existing = $this->api->get("/repos/{$org}/moko-platform/issues", [
|
$existing = $this->api->get("/repos/{$org}/MokoCLI/issues", [
|
||||||
'labels' => 'sync-report',
|
'labels' => 'sync-report',
|
||||||
'state' => 'all',
|
'state' => 'all',
|
||||||
'per_page' => 1,
|
'per_page' => 1,
|
||||||
@@ -1309,8 +1309,8 @@ class BulkSync extends CliFramework
|
|||||||
'direction' => 'desc',
|
'direction' => 'desc',
|
||||||
]);
|
]);
|
||||||
|
|
||||||
$labelNames = ['sync-report', 'moko-platform', 'type: chore', 'automation'];
|
$labelNames = ['sync-report', 'MokoCLI', 'type: chore', 'automation'];
|
||||||
$labels = $this->resolveLabelIds($org, 'moko-platform', $labelNames);
|
$labels = $this->resolveLabelIds($org, 'MokoCLI', $labelNames);
|
||||||
$existing = array_values($existing);
|
$existing = array_values($existing);
|
||||||
|
|
||||||
if (!empty($existing) && isset($existing[0]['number'])) {
|
if (!empty($existing) && isset($existing[0]['number'])) {
|
||||||
@@ -1319,22 +1319,22 @@ class BulkSync extends CliFramework
|
|||||||
if (($existing[0]['state'] ?? 'open') === 'closed') {
|
if (($existing[0]['state'] ?? 'open') === 'closed') {
|
||||||
$patch['state'] = 'open';
|
$patch['state'] = 'open';
|
||||||
}
|
}
|
||||||
$this->api->patch("/repos/{$org}/moko-platform/issues/{$issueNumber}", $patch);
|
$this->api->patch("/repos/{$org}/MokoCLI/issues/{$issueNumber}", $patch);
|
||||||
try {
|
try {
|
||||||
$this->api->post("/repos/{$org}/moko-platform/issues/{$issueNumber}/labels", ['labels' => $labels]);
|
$this->api->post("/repos/{$org}/MokoCLI/issues/{$issueNumber}/labels", ['labels' => $labels]);
|
||||||
} catch (\Exception $le) {
|
} catch (\Exception $le) {
|
||||||
/* non-fatal */
|
/* non-fatal */
|
||||||
}
|
}
|
||||||
$this->log("📋 Sync report issue updated: {$org}/moko-platform#{$issueNumber}", 'INFO');
|
$this->log("📋 Sync report issue updated: {$org}/MokoCLI#{$issueNumber}", 'INFO');
|
||||||
} else {
|
} else {
|
||||||
$issue = $this->api->post("/repos/{$org}/moko-platform/issues", [
|
$issue = $this->api->post("/repos/{$org}/MokoCLI/issues", [
|
||||||
'title' => $title,
|
'title' => $title,
|
||||||
'body' => $body,
|
'body' => $body,
|
||||||
'labels' => $labels,
|
'labels' => $labels,
|
||||||
'assignees' => ['jmiller'],
|
'assignees' => ['jmiller'],
|
||||||
]);
|
]);
|
||||||
$issueNumber = $issue['number'] ?? '?';
|
$issueNumber = $issue['number'] ?? '?';
|
||||||
$this->log("📋 Sync report issue created: {$org}/moko-platform#{$issueNumber}", 'INFO');
|
$this->log("📋 Sync report issue created: {$org}/MokoCLI#{$issueNumber}", 'INFO');
|
||||||
}
|
}
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
$this->log("⚠️ Failed to create/update sync report issue: " . $e->getMessage(), 'WARN');
|
$this->log("⚠️ Failed to create/update sync report issue: " . $e->getMessage(), 'WARN');
|
||||||
@@ -1342,7 +1342,7 @@ class BulkSync extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create or update a failure issue in moko-platform when repos fail to sync.
|
* Create or update a failure issue in MokoCLI when repos fail to sync.
|
||||||
* Uses the 'sync-failure' label so it is distinct from the run-report issue.
|
* Uses the 'sync-failure' label so it is distinct from the run-report issue.
|
||||||
* Reopens a closed issue rather than creating a duplicate.
|
* Reopens a closed issue rather than creating a duplicate.
|
||||||
*/
|
*/
|
||||||
@@ -1388,7 +1388,7 @@ class BulkSync extends CliFramework
|
|||||||
$body = preg_replace('/^ /m', '', $body);
|
$body = preg_replace('/^ /m', '', $body);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$existing = $this->api->get("/repos/{$org}/moko-platform/issues", [
|
$existing = $this->api->get("/repos/{$org}/MokoCLI/issues", [
|
||||||
'labels' => 'sync-failure',
|
'labels' => 'sync-failure',
|
||||||
'state' => 'all',
|
'state' => 'all',
|
||||||
'per_page' => 1,
|
'per_page' => 1,
|
||||||
@@ -1403,17 +1403,17 @@ class BulkSync extends CliFramework
|
|||||||
if (($existing[0]['state'] ?? 'open') === 'closed') {
|
if (($existing[0]['state'] ?? 'open') === 'closed') {
|
||||||
$patch['state'] = 'open';
|
$patch['state'] = 'open';
|
||||||
}
|
}
|
||||||
$this->api->patch("/repos/{$org}/moko-platform/issues/{$num}", $patch);
|
$this->api->patch("/repos/{$org}/MokoCLI/issues/{$num}", $patch);
|
||||||
$this->log("🚨 Failure issue #{$num} updated: {$org}/moko-platform#{$num}", 'WARN');
|
$this->log("🚨 Failure issue #{$num} updated: {$org}/MokoCLI#{$num}", 'WARN');
|
||||||
} else {
|
} else {
|
||||||
$issue = $this->api->post("/repos/{$org}/moko-platform/issues", [
|
$issue = $this->api->post("/repos/{$org}/MokoCLI/issues", [
|
||||||
'title' => $title,
|
'title' => $title,
|
||||||
'body' => $body,
|
'body' => $body,
|
||||||
'labels' => $this->resolveLabelIds($org, 'moko-platform', ['sync-failure']),
|
'labels' => $this->resolveLabelIds($org, 'MokoCLI', ['sync-failure']),
|
||||||
'assignees' => ['jmiller'],
|
'assignees' => ['jmiller'],
|
||||||
]);
|
]);
|
||||||
$num = $issue['number'] ?? '?';
|
$num = $issue['number'] ?? '?';
|
||||||
$this->log("🚨 Failure issue created: {$org}/moko-platform#{$num}", 'WARN');
|
$this->log("🚨 Failure issue created: {$org}/MokoCLI#{$num}", 'WARN');
|
||||||
}
|
}
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
$this->log("⚠️ Could not create/update failure issue: " . $e->getMessage(), 'WARN');
|
$this->log("⚠️ Could not create/update failure issue: " . $e->getMessage(), 'WARN');
|
||||||
|
|||||||
+123
-123
@@ -1,123 +1,123 @@
|
|||||||
#!/usr/bin/env bash
|
#!/usr/bin/env bash
|
||||||
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
# BRIEF: Trigger a workflow across all client-waas repos in a Gitea org
|
# BRIEF: Trigger a workflow across all client-waas repos in a Gitea org
|
||||||
|
|
||||||
set -euo pipefail
|
set -euo pipefail
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
# Usage
|
# Usage
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
usage() {
|
usage() {
|
||||||
cat <<EOF
|
cat <<EOF
|
||||||
Usage: $(basename "$0") GITEA_URL TOKEN ORG WORKFLOW [REF] [INPUTS]
|
Usage: $(basename "$0") GITEA_URL TOKEN ORG WORKFLOW [REF] [INPUTS]
|
||||||
|
|
||||||
Arguments:
|
Arguments:
|
||||||
GITEA_URL Base URL of the Gitea instance (e.g. https://git.mokoconsulting.tech)
|
GITEA_URL Base URL of the Gitea instance (e.g. https://git.mokoconsulting.tech)
|
||||||
TOKEN Gitea API token with repo/action permissions
|
TOKEN Gitea API token with repo/action permissions
|
||||||
ORG Organisation or user that owns the repos
|
ORG Organisation or user that owns the repos
|
||||||
WORKFLOW Workflow filename to trigger (e.g. dependency-audit.yml)
|
WORKFLOW Workflow filename to trigger (e.g. dependency-audit.yml)
|
||||||
REF Branch ref to run against (default: main)
|
REF Branch ref to run against (default: main)
|
||||||
INPUTS Optional JSON object of workflow inputs (e.g. '{"dry_run":"true"}')
|
INPUTS Optional JSON object of workflow inputs (e.g. '{"dry_run":"true"}')
|
||||||
|
|
||||||
Example:
|
Example:
|
||||||
$(basename "$0") https://git.mokoconsulting.tech abc123 MokoConsulting dependency-audit.yml main '{"notify":"true"}'
|
$(basename "$0") https://git.mokoconsulting.tech abc123 MokoConsulting dependency-audit.yml main '{"notify":"true"}'
|
||||||
EOF
|
EOF
|
||||||
exit 1
|
exit 1
|
||||||
}
|
}
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
# Argument parsing
|
# Argument parsing
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
if [ $# -lt 4 ]; then
|
if [ $# -lt 4 ]; then
|
||||||
usage
|
usage
|
||||||
fi
|
fi
|
||||||
|
|
||||||
GITEA_URL="${1%/}"
|
GITEA_URL="${1%/}"
|
||||||
TOKEN="$2"
|
TOKEN="$2"
|
||||||
ORG="$3"
|
ORG="$3"
|
||||||
WORKFLOW="$4"
|
WORKFLOW="$4"
|
||||||
REF="${5:-main}"
|
REF="${5:-main}"
|
||||||
INPUTS="${6:-{\}}"
|
INPUTS="${6:-{\}}"
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
# Fetch all repos in the org, paginated
|
# Fetch all repos in the org, paginated
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
echo "Fetching repos for org '${ORG}' on ${GITEA_URL} ..."
|
echo "Fetching repos for org '${ORG}' on ${GITEA_URL} ..."
|
||||||
|
|
||||||
PAGE=1
|
PAGE=1
|
||||||
LIMIT=50
|
LIMIT=50
|
||||||
ALL_REPOS=""
|
ALL_REPOS=""
|
||||||
|
|
||||||
while true; do
|
while true; do
|
||||||
RESPONSE=$(curl -s \
|
RESPONSE=$(curl -s \
|
||||||
-H "Authorization: token ${TOKEN}" \
|
-H "Authorization: token ${TOKEN}" \
|
||||||
-H "Accept: application/json" \
|
-H "Accept: application/json" \
|
||||||
"${GITEA_URL}/api/v1/orgs/${ORG}/repos?page=${PAGE}&limit=${LIMIT}")
|
"${GITEA_URL}/api/v1/orgs/${ORG}/repos?page=${PAGE}&limit=${LIMIT}")
|
||||||
|
|
||||||
# Break if empty array
|
# Break if empty array
|
||||||
COUNT=$(echo "$RESPONSE" | jq -r 'length')
|
COUNT=$(echo "$RESPONSE" | jq -r 'length')
|
||||||
if [ "$COUNT" -eq 0 ]; then
|
if [ "$COUNT" -eq 0 ]; then
|
||||||
break
|
break
|
||||||
fi
|
fi
|
||||||
|
|
||||||
NAMES=$(echo "$RESPONSE" | jq -r '.[].name')
|
NAMES=$(echo "$RESPONSE" | jq -r '.[].name')
|
||||||
ALL_REPOS="${ALL_REPOS}${NAMES}"$'\n'
|
ALL_REPOS="${ALL_REPOS}${NAMES}"$'\n'
|
||||||
|
|
||||||
if [ "$COUNT" -lt "$LIMIT" ]; then
|
if [ "$COUNT" -lt "$LIMIT" ]; then
|
||||||
break
|
break
|
||||||
fi
|
fi
|
||||||
|
|
||||||
PAGE=$((PAGE + 1))
|
PAGE=$((PAGE + 1))
|
||||||
done
|
done
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
# Filter for client-waas repos
|
# Filter for client-waas repos
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
CLIENT_REPOS=$(echo "$ALL_REPOS" | grep 'client-waas' | sort || true)
|
CLIENT_REPOS=$(echo "$ALL_REPOS" | grep 'client-waas' | sort || true)
|
||||||
|
|
||||||
if [ -z "$CLIENT_REPOS" ]; then
|
if [ -z "$CLIENT_REPOS" ]; then
|
||||||
echo "No client-waas repos found in org '${ORG}'."
|
echo "No client-waas repos found in org '${ORG}'."
|
||||||
exit 0
|
exit 0
|
||||||
fi
|
fi
|
||||||
|
|
||||||
TOTAL=$(echo "$CLIENT_REPOS" | wc -l | tr -d ' ')
|
TOTAL=$(echo "$CLIENT_REPOS" | wc -l | tr -d ' ')
|
||||||
echo "Found ${TOTAL} client-waas repo(s). Triggering workflow '${WORKFLOW}' (ref: ${REF}) ..."
|
echo "Found ${TOTAL} client-waas repo(s). Triggering workflow '${WORKFLOW}' (ref: ${REF}) ..."
|
||||||
echo ""
|
echo ""
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
# Trigger workflow for each repo
|
# Trigger workflow for each repo
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
SUCCESS=0
|
SUCCESS=0
|
||||||
FAIL=0
|
FAIL=0
|
||||||
|
|
||||||
while IFS= read -r REPO; do
|
while IFS= read -r REPO; do
|
||||||
[ -z "$REPO" ] && continue
|
[ -z "$REPO" ] && continue
|
||||||
|
|
||||||
PAYLOAD=$(jq -n --arg ref "$REF" --argjson inputs "$INPUTS" '{ref: $ref, inputs: $inputs}')
|
PAYLOAD=$(jq -n --arg ref "$REF" --argjson inputs "$INPUTS" '{ref: $ref, inputs: $inputs}')
|
||||||
|
|
||||||
HTTP_CODE=$(curl -s -o /dev/null -w '%{http_code}' \
|
HTTP_CODE=$(curl -s -o /dev/null -w '%{http_code}' \
|
||||||
-X POST \
|
-X POST \
|
||||||
-H "Authorization: token ${TOKEN}" \
|
-H "Authorization: token ${TOKEN}" \
|
||||||
-H "Content-Type: application/json" \
|
-H "Content-Type: application/json" \
|
||||||
-d "$PAYLOAD" \
|
-d "$PAYLOAD" \
|
||||||
"${GITEA_URL}/api/v1/repos/${ORG}/${REPO}/actions/workflows/${WORKFLOW}/dispatches")
|
"${GITEA_URL}/api/v1/repos/${ORG}/${REPO}/actions/workflows/${WORKFLOW}/dispatches")
|
||||||
|
|
||||||
if [ "$HTTP_CODE" -eq 204 ] || [ "$HTTP_CODE" -eq 201 ]; then
|
if [ "$HTTP_CODE" -eq 204 ] || [ "$HTTP_CODE" -eq 201 ]; then
|
||||||
echo " [OK] ${ORG}/${REPO} (HTTP ${HTTP_CODE})"
|
echo " [OK] ${ORG}/${REPO} (HTTP ${HTTP_CODE})"
|
||||||
SUCCESS=$((SUCCESS + 1))
|
SUCCESS=$((SUCCESS + 1))
|
||||||
else
|
else
|
||||||
echo " [FAIL] ${ORG}/${REPO} (HTTP ${HTTP_CODE})"
|
echo " [FAIL] ${ORG}/${REPO} (HTTP ${HTTP_CODE})"
|
||||||
FAIL=$((FAIL + 1))
|
FAIL=$((FAIL + 1))
|
||||||
fi
|
fi
|
||||||
done <<< "$CLIENT_REPOS"
|
done <<< "$CLIENT_REPOS"
|
||||||
|
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
# Summary
|
# Summary
|
||||||
# ---------------------------------------------------------------------------
|
# ---------------------------------------------------------------------------
|
||||||
echo ""
|
echo ""
|
||||||
echo "Done. Success: ${SUCCESS} | Failed: ${FAIL} | Total: ${TOTAL}"
|
echo "Done. Success: ${SUCCESS} | Failed: ${FAIL} | Total: ${TOTAL}"
|
||||||
|
|
||||||
if [ "$FAIL" -gt 0 ]; then
|
if [ "$FAIL" -gt 0 ]; then
|
||||||
exit 1
|
exit 1
|
||||||
fi
|
fi
|
||||||
|
|||||||
@@ -6,8 +6,8 @@
|
|||||||
#
|
#
|
||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Automation.CI
|
# DEFGROUP: Automation.CI
|
||||||
# INGROUP: moko-platform.Automation
|
# INGROUP: MokoCLI.Automation
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /automation/ci-issue-reporter.sh
|
# PATH: /automation/ci-issue-reporter.sh
|
||||||
# VERSION: 09.23.00
|
# VERSION: 09.23.00
|
||||||
# BRIEF: Creates or updates a Gitea issue when a CI gate fails.
|
# BRIEF: Creates or updates a Gitea issue when a CI gate fails.
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/enrich_manifest_xml.php
|
* PATH: /automation/enrich_manifest_xml.php
|
||||||
* BRIEF: Enrich XML manifests with repo-specific build and deploy details
|
* BRIEF: Enrich XML manifests with repo-specific build and deploy details
|
||||||
*
|
*
|
||||||
@@ -46,7 +46,7 @@ class EnrichManifestXmlCli extends CliFramework
|
|||||||
$parser = new MokoStandardsParser();
|
$parser = new MokoStandardsParser();
|
||||||
$tmpBase = sys_get_temp_dir() . '/moko-enrich-' . getmypid();
|
$tmpBase = sys_get_temp_dir() . '/moko-enrich-' . getmypid();
|
||||||
|
|
||||||
echo "=== moko-platform XML Manifest Enrichment ===\n";
|
echo "=== MokoCLI XML Manifest Enrichment ===\n";
|
||||||
echo "Mode: " . ($this->dryRun ? "DRY RUN" : "LIVE") . "\n";
|
echo "Mode: " . ($this->dryRun ? "DRY RUN" : "LIVE") . "\n";
|
||||||
if (!empty($skipRepos)) {
|
if (!empty($skipRepos)) {
|
||||||
echo "Skipping: " . implode(', ', $skipRepos) . "\n";
|
echo "Skipping: " . implode(', ', $skipRepos) . "\n";
|
||||||
@@ -97,7 +97,7 @@ class EnrichManifestXmlCli extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
$manifestPath = "{$workDir}/.mokogitea/manifest.xml";
|
$manifestPath = "{$workDir}/.mokogitea/manifest.xml";
|
||||||
if (!file_exists($manifestPath) || !str_contains(file_get_contents($manifestPath), '<moko-platform')) {
|
if (!file_exists($manifestPath) || !str_contains(file_get_contents($manifestPath), '<MokoCLI')) {
|
||||||
echo "SKIP (no XML manifest)\n";
|
echo "SKIP (no XML manifest)\n";
|
||||||
$stats['skipped']++;
|
$stats['skipped']++;
|
||||||
$this->rmTree($workDir);
|
$this->rmTree($workDir);
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/enrich_mokostandards_xml.php
|
* PATH: /automation/enrich_mokostandards_xml.php
|
||||||
* BRIEF: Enrich XML manifests with repo-specific build and deploy details
|
* BRIEF: Enrich XML manifests with repo-specific build and deploy details
|
||||||
*
|
*
|
||||||
@@ -46,7 +46,7 @@ class EnrichMokostandardsXmlCli extends CliFramework
|
|||||||
$parser = new MokoStandardsParser();
|
$parser = new MokoStandardsParser();
|
||||||
$tmpBase = sys_get_temp_dir() . '/moko-enrich-' . getmypid();
|
$tmpBase = sys_get_temp_dir() . '/moko-enrich-' . getmypid();
|
||||||
|
|
||||||
echo "=== moko-platform XML Manifest Enrichment ===\n";
|
echo "=== MokoCLI XML Manifest Enrichment ===\n";
|
||||||
echo "Mode: " . ($this->dryRun ? "DRY RUN" : "LIVE") . "\n";
|
echo "Mode: " . ($this->dryRun ? "DRY RUN" : "LIVE") . "\n";
|
||||||
if (!empty($skipRepos)) {
|
if (!empty($skipRepos)) {
|
||||||
echo "Skipping: " . implode(', ', $skipRepos) . "\n";
|
echo "Skipping: " . implode(', ', $skipRepos) . "\n";
|
||||||
@@ -97,7 +97,7 @@ class EnrichMokostandardsXmlCli extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
$manifestPath = "{$workDir}/.mokogitea/manifest.xml";
|
$manifestPath = "{$workDir}/.mokogitea/manifest.xml";
|
||||||
if (!file_exists($manifestPath) || !str_contains(file_get_contents($manifestPath), '<moko-platform')) {
|
if (!file_exists($manifestPath) || !str_contains(file_get_contents($manifestPath), '<MokoCLI')) {
|
||||||
echo "SKIP (no XML manifest)\n";
|
echo "SKIP (no XML manifest)\n";
|
||||||
$stats['skipped']++;
|
$stats['skipped']++;
|
||||||
$this->rmTree($workDir);
|
$this->rmTree($workDir);
|
||||||
|
|||||||
+3
-3
@@ -2,9 +2,9 @@
|
|||||||
Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
SPDX-License-Identifier: GPL-3.0-or-later
|
SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
FILE INFORMATION
|
FILE INFORMATION
|
||||||
DEFGROUP: MokoPlatform.Index
|
DEFGROUP: MokoCLI.Index
|
||||||
INGROUP: MokoPlatform.Automation
|
INGROUP: MokoCLI.Automation
|
||||||
REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
PATH: /automation/index.md
|
PATH: /automation/index.md
|
||||||
BRIEF: Automation directory index
|
BRIEF: Automation directory index
|
||||||
-->
|
-->
|
||||||
|
|||||||
@@ -8,16 +8,16 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/migrate_to_gitea.php
|
* PATH: /automation/migrate_to_gitea.php
|
||||||
* BRIEF: Migrate repositories from GitHub to self-hosted Gitea instance
|
* BRIEF: Migrate repositories from GitHub to self-hosted Gitea instance
|
||||||
*
|
*
|
||||||
* USAGE
|
* USAGE
|
||||||
* php automation/migrate_to_gitea.php --dry-run
|
* php automation/migrate_to_gitea.php --dry-run
|
||||||
* php automation/migrate_to_gitea.php --repos MokoCRM MokoDoliMods
|
* php automation/migrate_to_gitea.php --repos MokoCRM MokoDoliMods
|
||||||
* php automation/migrate_to_gitea.php --exclude moko-platform --skip-archived
|
* php automation/migrate_to_gitea.php --exclude MokoCLI --skip-archived
|
||||||
* php automation/migrate_to_gitea.php --resume
|
* php automation/migrate_to_gitea.php --resume
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@@ -278,7 +278,7 @@ class MigrateToGitea extends CliFramework
|
|||||||
try {
|
try {
|
||||||
$this->gitea->createIssue(
|
$this->gitea->createIssue(
|
||||||
$giteaOrg,
|
$giteaOrg,
|
||||||
'moko-platform',
|
'MokoCLI',
|
||||||
'chore: GitHub → Gitea migration report — ' . count($results['migrated']) . ' repos migrated',
|
'chore: GitHub → Gitea migration report — ' . count($results['migrated']) . ' repos migrated',
|
||||||
$report,
|
$report,
|
||||||
['labels' => ['automation', 'type: chore']]
|
['labels' => ['automation', 'type: chore']]
|
||||||
|
|||||||
+22
-22
@@ -9,9 +9,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform.Scripts
|
* INGROUP: MokoCLI.Scripts
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/push_files.php
|
* PATH: /automation/push_files.php
|
||||||
* BRIEF: Push one or more specific files to one or more remote repositories
|
* BRIEF: Push one or more specific files to one or more remote repositories
|
||||||
*/
|
*/
|
||||||
@@ -35,7 +35,7 @@ use MokoEnterprise\{
|
|||||||
/**
|
/**
|
||||||
* Targeted File Push Tool
|
* Targeted File Push Tool
|
||||||
*
|
*
|
||||||
* Pushes one or more specific files from moko-platform templates to one or
|
* Pushes one or more specific files from MokoCLI templates to one or
|
||||||
* more remote repositories — without running a full sync.
|
* more remote repositories — without running a full sync.
|
||||||
*
|
*
|
||||||
* Files are specified by their destination path as they appear in the target
|
* Files are specified by their destination path as they appear in the target
|
||||||
@@ -81,7 +81,7 @@ class PushFiles extends CliFramework
|
|||||||
*/
|
*/
|
||||||
protected function run(): int
|
protected function run(): int
|
||||||
{
|
{
|
||||||
$this->log('📦 moko-platform File Push v' . self::VERSION, 'INFO');
|
$this->log('📦 MokoCLI File Push v' . self::VERSION, 'INFO');
|
||||||
|
|
||||||
if (!$this->initializeComponents()) {
|
if (!$this->initializeComponents()) {
|
||||||
return 1;
|
return 1;
|
||||||
@@ -337,7 +337,7 @@ class PushFiles extends CliFramework
|
|||||||
|
|
||||||
$prNumber = null;
|
$prNumber = null;
|
||||||
if (!$direct) {
|
if (!$direct) {
|
||||||
$prTitle = "chore: push " . count($entries) . " file(s) from moko-platform";
|
$prTitle = "chore: push " . count($entries) . " file(s) from MokoCLI";
|
||||||
$prBody = $this->buildPRBody($entries);
|
$prBody = $this->buildPRBody($entries);
|
||||||
$pr = $this->adapter->createPullRequest(
|
$pr = $this->adapter->createPullRequest(
|
||||||
$org,
|
$org,
|
||||||
@@ -414,7 +414,7 @@ class PushFiles extends CliFramework
|
|||||||
|
|
||||||
$message = !empty($customMessage)
|
$message = !empty($customMessage)
|
||||||
? $customMessage
|
? $customMessage
|
||||||
: "chore: update {$destPath} from moko-platform";
|
: "chore: update {$destPath} from MokoCLI";
|
||||||
|
|
||||||
// Fetch existing file SHA (needed for updates)
|
// Fetch existing file SHA (needed for updates)
|
||||||
$existingSha = null;
|
$existingSha = null;
|
||||||
@@ -457,9 +457,9 @@ class PushFiles extends CliFramework
|
|||||||
): void {
|
): void {
|
||||||
$now = gmdate('Y-m-d H:i:s') . ' UTC';
|
$now = gmdate('Y-m-d H:i:s') . ' UTC';
|
||||||
$version = self::VERSION;
|
$version = self::VERSION;
|
||||||
$source = $this->adapter->getRepoWebUrl($org, 'moko-platform');
|
$source = $this->adapter->getRepoWebUrl($org, 'MokoCLI');
|
||||||
|
|
||||||
$title = "chore: moko-platform file push tracking";
|
$title = "chore: MokoCLI file push tracking";
|
||||||
|
|
||||||
$deliveryLine = $prNumber !== null
|
$deliveryLine = $prNumber !== null
|
||||||
? "| **Pull request** | [#{$prNumber}](" . $this->adapter->getPullRequestWebUrl($org, $repo, $prNumber) . ") |"
|
? "| **Pull request** | [#{$prNumber}](" . $this->adapter->getPullRequestWebUrl($org, $repo, $prNumber) . ") |"
|
||||||
@@ -471,9 +471,9 @@ class PushFiles extends CliFramework
|
|||||||
));
|
));
|
||||||
|
|
||||||
$body = <<<MD
|
$body = <<<MD
|
||||||
## moko-platform File Push
|
## MokoCLI File Push
|
||||||
|
|
||||||
One or more files were pushed to this repository from moko-platform.
|
One or more files were pushed to this repository from MokoCLI.
|
||||||
|
|
||||||
| Field | Value |
|
| Field | Value |
|
||||||
|-------|-------|
|
|-------|-------|
|
||||||
@@ -487,12 +487,12 @@ class PushFiles extends CliFramework
|
|||||||
{$fileRows}
|
{$fileRows}
|
||||||
|
|
||||||
---
|
---
|
||||||
*Generated automatically by [moko-platform]({$source}) `push_files.php`*
|
*Generated automatically by [MokoCLI]({$source}) `push_files.php`*
|
||||||
MD;
|
MD;
|
||||||
|
|
||||||
$body = preg_replace('/^ /m', '', $body);
|
$body = preg_replace('/^ /m', '', $body);
|
||||||
|
|
||||||
$labels = ['standards-update', 'moko-platform', 'type: chore', 'automation'];
|
$labels = ['standards-update', 'MokoCLI', 'type: chore', 'automation'];
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$existing = $this->api->get("/repos/{$org}/{$repo}/issues", [
|
$existing = $this->api->get("/repos/{$org}/{$repo}/issues", [
|
||||||
@@ -550,7 +550,7 @@ class PushFiles extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Create or update a failure issue in moko-platform when repos fail to receive files.
|
* Create or update a failure issue in MokoCLI when repos fail to receive files.
|
||||||
* Uses the 'push-failure' label. Reopens a closed issue rather than creating a duplicate.
|
* Uses the 'push-failure' label. Reopens a closed issue rather than creating a duplicate.
|
||||||
*/
|
*/
|
||||||
private function createFailureIssue(string $org, array $results): void
|
private function createFailureIssue(string $org, array $results): void
|
||||||
@@ -598,7 +598,7 @@ class PushFiles extends CliFramework
|
|||||||
$body = preg_replace('/^ /m', '', $body);
|
$body = preg_replace('/^ /m', '', $body);
|
||||||
|
|
||||||
try {
|
try {
|
||||||
$existing = $this->api->get("/repos/{$org}/moko-platform/issues", [
|
$existing = $this->api->get("/repos/{$org}/MokoCLI/issues", [
|
||||||
'labels' => 'push-failure',
|
'labels' => 'push-failure',
|
||||||
'state' => 'all',
|
'state' => 'all',
|
||||||
'per_page' => 1,
|
'per_page' => 1,
|
||||||
@@ -613,17 +613,17 @@ class PushFiles extends CliFramework
|
|||||||
if (($existing[0]['state'] ?? 'open') === 'closed') {
|
if (($existing[0]['state'] ?? 'open') === 'closed') {
|
||||||
$patch['state'] = 'open';
|
$patch['state'] = 'open';
|
||||||
}
|
}
|
||||||
$this->api->patch("/repos/{$org}/moko-platform/issues/{$num}", $patch);
|
$this->api->patch("/repos/{$org}/MokoCLI/issues/{$num}", $patch);
|
||||||
$this->log("🚨 Failure issue #{$num} updated: {$org}/moko-platform#{$num}", 'WARN');
|
$this->log("🚨 Failure issue #{$num} updated: {$org}/MokoCLI#{$num}", 'WARN');
|
||||||
} else {
|
} else {
|
||||||
$issue = $this->api->post("/repos/{$org}/moko-platform/issues", [
|
$issue = $this->api->post("/repos/{$org}/MokoCLI/issues", [
|
||||||
'title' => $title,
|
'title' => $title,
|
||||||
'body' => $body,
|
'body' => $body,
|
||||||
'labels' => ['push-failure'],
|
'labels' => ['push-failure'],
|
||||||
'assignees' => ['jmiller'],
|
'assignees' => ['jmiller'],
|
||||||
]);
|
]);
|
||||||
$num = $issue['number'] ?? '?';
|
$num = $issue['number'] ?? '?';
|
||||||
$this->log("🚨 Failure issue created: {$org}/moko-platform#{$num}", 'WARN');
|
$this->log("🚨 Failure issue created: {$org}/MokoCLI#{$num}", 'WARN');
|
||||||
}
|
}
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
$this->log("⚠️ Could not create/update failure issue: " . $e->getMessage(), 'WARN');
|
$this->log("⚠️ Could not create/update failure issue: " . $e->getMessage(), 'WARN');
|
||||||
@@ -638,14 +638,14 @@ class PushFiles extends CliFramework
|
|||||||
private function buildPRBody(array $entries): string
|
private function buildPRBody(array $entries): string
|
||||||
{
|
{
|
||||||
$now = gmdate('Y-m-d H:i:s') . ' UTC';
|
$now = gmdate('Y-m-d H:i:s') . ' UTC';
|
||||||
$lines = ["## moko-platform File Push\n", "**Pushed:** {$now}\n", '### Files\n'];
|
$lines = ["## MokoCLI File Push\n", "**Pushed:** {$now}\n", '### Files\n'];
|
||||||
|
|
||||||
foreach ($entries as $entry) {
|
foreach ($entries as $entry) {
|
||||||
$lines[] = "- `{$entry['destination']}`";
|
$lines[] = "- `{$entry['destination']}`";
|
||||||
}
|
}
|
||||||
|
|
||||||
$sourceUrl = $this->adapter->getRepoWebUrl(self::DEFAULT_ORG, 'moko-platform');
|
$sourceUrl = $this->adapter->getRepoWebUrl(self::DEFAULT_ORG, 'MokoCLI');
|
||||||
$lines[] = "\n---\n*Generated by [moko-platform]({$sourceUrl}) `push_files.php`*";
|
$lines[] = "\n---\n*Generated by [MokoCLI]({$sourceUrl}) `push_files.php`*";
|
||||||
|
|
||||||
return implode("\n", $lines);
|
return implode("\n", $lines);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/push_manifest_xml.php
|
* PATH: /automation/push_manifest_xml.php
|
||||||
* BRIEF: Push XML manifests to all governed repositories
|
* BRIEF: Push XML manifests to all governed repositories
|
||||||
*/
|
*/
|
||||||
@@ -47,7 +47,7 @@ class PushManifestXmlCli extends CliFramework
|
|||||||
$parser = new MokoStandardsParser();
|
$parser = new MokoStandardsParser();
|
||||||
$tmpBase = sys_get_temp_dir() . '/moko-manifest-push-' . getmypid();
|
$tmpBase = sys_get_temp_dir() . '/moko-manifest-push-' . getmypid();
|
||||||
|
|
||||||
echo "=== moko-platform XML Manifest Push ===\n";
|
echo "=== MokoCLI XML Manifest Push ===\n";
|
||||||
echo "Org: {$giteaOrg}\n";
|
echo "Org: {$giteaOrg}\n";
|
||||||
echo "Mode: " . ($this->dryRun ? "DRY RUN" : "LIVE") . "\n";
|
echo "Mode: " . ($this->dryRun ? "DRY RUN" : "LIVE") . "\n";
|
||||||
if ($repoFilter) {
|
if ($repoFilter) {
|
||||||
@@ -125,7 +125,7 @@ class PushManifestXmlCli extends CliFramework
|
|||||||
|
|
||||||
// Check if already XML and up-to-date
|
// Check if already XML and up-to-date
|
||||||
$manifestPath = "{$workDir}/.mokogitea/manifest.xml";
|
$manifestPath = "{$workDir}/.mokogitea/manifest.xml";
|
||||||
$existingIsXml = file_exists($manifestPath) && str_contains(file_get_contents($manifestPath), '<moko-platform');
|
$existingIsXml = file_exists($manifestPath) && str_contains(file_get_contents($manifestPath), '<MokoCLI');
|
||||||
if ($existingIsXml && !$force) {
|
if ($existingIsXml && !$force) {
|
||||||
$existingPlatform = $parser->extractPlatform(file_get_contents($manifestPath));
|
$existingPlatform = $parser->extractPlatform(file_get_contents($manifestPath));
|
||||||
if ($existingPlatform === $platform) {
|
if ($existingPlatform === $platform) {
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/push_mokostandards_xml.php
|
* PATH: /automation/push_mokostandards_xml.php
|
||||||
* BRIEF: Push XML manifests to all governed repositories
|
* BRIEF: Push XML manifests to all governed repositories
|
||||||
*/
|
*/
|
||||||
@@ -47,7 +47,7 @@ class PushMokostandardsXmlCli extends CliFramework
|
|||||||
$parser = new MokoStandardsParser();
|
$parser = new MokoStandardsParser();
|
||||||
$tmpBase = sys_get_temp_dir() . '/moko-manifest-push-' . getmypid();
|
$tmpBase = sys_get_temp_dir() . '/moko-manifest-push-' . getmypid();
|
||||||
|
|
||||||
echo "=== moko-platform XML Manifest Push ===\n";
|
echo "=== MokoCLI XML Manifest Push ===\n";
|
||||||
echo "Org: {$giteaOrg}\n";
|
echo "Org: {$giteaOrg}\n";
|
||||||
echo "Mode: " . ($this->dryRun ? "DRY RUN" : "LIVE") . "\n";
|
echo "Mode: " . ($this->dryRun ? "DRY RUN" : "LIVE") . "\n";
|
||||||
if ($repoFilter) {
|
if ($repoFilter) {
|
||||||
@@ -125,7 +125,7 @@ class PushMokostandardsXmlCli extends CliFramework
|
|||||||
|
|
||||||
// Check if already XML and up-to-date
|
// Check if already XML and up-to-date
|
||||||
$manifestPath = "{$workDir}/.mokogitea/manifest.xml";
|
$manifestPath = "{$workDir}/.mokogitea/manifest.xml";
|
||||||
$existingIsXml = file_exists($manifestPath) && str_contains(file_get_contents($manifestPath), '<moko-platform');
|
$existingIsXml = file_exists($manifestPath) && str_contains(file_get_contents($manifestPath), '<MokoCLI');
|
||||||
if ($existingIsXml && !$force) {
|
if ($existingIsXml && !$force) {
|
||||||
$existingPlatform = $parser->extractPlatform(file_get_contents($manifestPath));
|
$existingPlatform = $parser->extractPlatform(file_get_contents($manifestPath));
|
||||||
if ($existingPlatform === $platform) {
|
if ($existingPlatform === $platform) {
|
||||||
|
|||||||
+11
-11
@@ -9,9 +9,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Automation
|
* DEFGROUP: MokoCLI.Automation
|
||||||
* INGROUP: MokoPlatform.Scripts
|
* INGROUP: MokoCLI.Scripts
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /automation/repo_cleanup.php
|
* PATH: /automation/repo_cleanup.php
|
||||||
* BRIEF: Enterprise repository cleanup — branches, PRs, issues, workflows, labels, logs
|
* BRIEF: Enterprise repository cleanup — branches, PRs, issues, workflows, labels, logs
|
||||||
*/
|
*/
|
||||||
@@ -39,14 +39,14 @@ use MokoEnterprise\{ApiClient, AuditLogger, CliFramework, Config, GitPlatformAda
|
|||||||
class RepoCleanup extends CliFramework
|
class RepoCleanup extends CliFramework
|
||||||
{
|
{
|
||||||
private const VERSION = '09.23.00';
|
private const VERSION = '09.23.00';
|
||||||
private const SYNC_PREFIX = 'chore/sync-moko-platform-';
|
private const SYNC_PREFIX = 'chore/sync-MokoCLI-';
|
||||||
private const CURRENT_BRANCH = 'chore/sync-moko-platform-v04.02.00';
|
private const CURRENT_BRANCH = 'chore/sync-MokoCLI-v04.02.00';
|
||||||
|
|
||||||
/** Workflow files that have been retired and should be deleted from governed repos. */
|
/** Workflow files that have been retired and should be deleted from governed repos. */
|
||||||
private const RETIRED_WORKFLOWS = [
|
private const RETIRED_WORKFLOWS = [
|
||||||
'build.yml', 'code-quality.yml', 'release-cycle.yml', 'release-pipeline.yml',
|
'build.yml', 'code-quality.yml', 'release-cycle.yml', 'release-pipeline.yml',
|
||||||
'branch-cleanup.yml', 'auto-update-changelog.yml', 'enterprise-issue-manager.yml',
|
'branch-cleanup.yml', 'auto-update-changelog.yml', 'enterprise-issue-manager.yml',
|
||||||
'flush-actions-cache.yml', 'moko-platform-script-runner.yml', 'unified-ci.yml',
|
'flush-actions-cache.yml', 'MokoCLI-script-runner.yml', 'unified-ci.yml',
|
||||||
'unified-platform-testing.yml', 'reusable-build.yml', 'reusable-ci-validation.yml',
|
'unified-platform-testing.yml', 'reusable-build.yml', 'reusable-ci-validation.yml',
|
||||||
'reusable-deploy.yml', 'reusable-php-quality.yml', 'reusable-platform-testing.yml',
|
'reusable-deploy.yml', 'reusable-php-quality.yml', 'reusable-platform-testing.yml',
|
||||||
'reusable-project-detector.yml', 'reusable-release.yml', 'reusable-script-executor.yml',
|
'reusable-project-detector.yml', 'reusable-release.yml', 'reusable-script-executor.yml',
|
||||||
@@ -98,7 +98,7 @@ class RepoCleanup extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
$this->logMsg("🧹 moko-platform Repository Cleanup v" . self::VERSION);
|
$this->logMsg("🧹 MokoCLI Repository Cleanup v" . self::VERSION);
|
||||||
$this->logMsg("Organization: {$org}");
|
$this->logMsg("Organization: {$org}");
|
||||||
$this->logMsg("Current sync branch: " . self::CURRENT_BRANCH);
|
$this->logMsg("Current sync branch: " . self::CURRENT_BRANCH);
|
||||||
if ($this->dryRun) {
|
if ($this->dryRun) {
|
||||||
@@ -225,7 +225,7 @@ class RepoCleanup extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
$allRepos = $this->adapter->listOrgRepos($org, $skipArchived);
|
$allRepos = $this->adapter->listOrgRepos($org, $skipArchived);
|
||||||
return array_filter($allRepos, fn($r) => !in_array($r['name'], ['moko-platform', '.github-private'], true));
|
return array_filter($allRepos, fn($r) => !in_array($r['name'], ['MokoCLI', '.github-private'], true));
|
||||||
}
|
}
|
||||||
|
|
||||||
// ─── Cleanup operations ──────────────────────────────────────────────
|
// ─── Cleanup operations ──────────────────────────────────────────────
|
||||||
@@ -463,9 +463,9 @@ class RepoCleanup extends CliFramework
|
|||||||
private function checkLabels(string $org, string $repo, array &$results): void
|
private function checkLabels(string $org, string $repo, array &$results): void
|
||||||
{
|
{
|
||||||
try {
|
try {
|
||||||
$this->api->get("/repos/{$org}/{$repo}/labels/moko-platform");
|
$this->api->get("/repos/{$org}/{$repo}/labels/MokoCLI");
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
$this->logMsg(" ⚠️ Missing 'moko-platform' label");
|
$this->logMsg(" ⚠️ Missing 'MokoCLI' label");
|
||||||
$results['labels_missing']++;
|
$results['labels_missing']++;
|
||||||
$this->api->resetCircuitBreaker();
|
$this->api->resetCircuitBreaker();
|
||||||
}
|
}
|
||||||
@@ -479,7 +479,7 @@ class RepoCleanup extends CliFramework
|
|||||||
if (preg_match('/^\s*VERSION:\s*(\d{2}\.\d{2}\.\d{2})/m', $content, $m)) {
|
if (preg_match('/^\s*VERSION:\s*(\d{2}\.\d{2}\.\d{2})/m', $content, $m)) {
|
||||||
$version = $m[1];
|
$version = $m[1];
|
||||||
|
|
||||||
// Check manifest.xml for the tracked moko-platform version
|
// Check manifest.xml for the tracked MokoCLI version
|
||||||
try {
|
try {
|
||||||
$mokoFile = $this->api->get("/repos/{$org}/{$repo}/contents/.mokogitea/manifest.xml");
|
$mokoFile = $this->api->get("/repos/{$org}/{$repo}/contents/.mokogitea/manifest.xml");
|
||||||
$mokoContent = base64_decode($mokoFile['content'] ?? '');
|
$mokoContent = base64_decode($mokoFile['content'] ?? '');
|
||||||
|
|||||||
@@ -4,9 +4,9 @@
|
|||||||
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
# SPDX-License-Identifier: GPL-3.0-or-later
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
#
|
#
|
||||||
# DEFGROUP: MokoPlatform.Automation.ServerAutoheal
|
# DEFGROUP: MokoCLI.Automation.ServerAutoheal
|
||||||
# INGROUP: MokoPlatform.Automation
|
# INGROUP: MokoCLI.Automation
|
||||||
# REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
# PATH: /automation/server-autoheal.sh
|
# PATH: /automation/server-autoheal.sh
|
||||||
# BRIEF: Server auto-heal on unclean restart + split system/content backups
|
# BRIEF: Server auto-heal on unclean restart + split system/content backups
|
||||||
#
|
#
|
||||||
|
|||||||
@@ -9,11 +9,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoStandards.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: MokoStandards
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /bin/moko
|
* PATH: /bin/moko
|
||||||
* BRIEF: Unified CLI dispatcher — run any MokoStandards script without needing GitHub Actions
|
* BRIEF: Unified CLI dispatcher — run any MokoCLI script without needing GitHub Actions
|
||||||
*
|
*
|
||||||
* USAGE
|
* USAGE
|
||||||
* php bin/moko <command> [options] (all platforms)
|
* php bin/moko <command> [options] (all platforms)
|
||||||
@@ -292,10 +292,10 @@ function printHelp(): void
|
|||||||
{
|
{
|
||||||
echo <<<'HELP'
|
echo <<<'HELP'
|
||||||
╔══════════════════════════════════════════════════════════╗
|
╔══════════════════════════════════════════════════════════╗
|
||||||
║ MokoStandards CLI (bin/moko) ║
|
║ MokoCLI (bin/moko) ║
|
||||||
╚══════════════════════════════════════════════════════════╝
|
╚══════════════════════════════════════════════════════════╝
|
||||||
|
|
||||||
Run any MokoStandards script locally without GitHub Actions.
|
Run any MokoCLI script locally without GitHub Actions.
|
||||||
|
|
||||||
USAGE
|
USAGE
|
||||||
php bin/moko <command> [options] (all platforms)
|
php bin/moko <command> [options] (all platforms)
|
||||||
|
|||||||
@@ -8,9 +8,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/archive_repo.php
|
* PATH: /cli/archive_repo.php
|
||||||
* BRIEF: Gracefully retire a governed repository — archive, close issues/PRs, remove sync def
|
* BRIEF: Gracefully retire a governed repository — archive, close issues/PRs, remove sync def
|
||||||
*/
|
*/
|
||||||
@@ -135,7 +135,7 @@ class ArchiveRepoCli extends CliFramework
|
|||||||
try {
|
try {
|
||||||
$issue = $adapter->createIssue(
|
$issue = $adapter->createIssue(
|
||||||
$org,
|
$org,
|
||||||
'moko-platform',
|
'MokoCLI',
|
||||||
"chore: archived repository {$repoName}",
|
"chore: archived repository {$repoName}",
|
||||||
"## Repository Archived\n\n"
|
"## Repository Archived\n\n"
|
||||||
. "**Repository:** `{$org}/{$repoName}`\n"
|
. "**Repository:** `{$org}/{$repoName}`\n"
|
||||||
@@ -150,7 +150,7 @@ class ArchiveRepoCli extends CliFramework
|
|||||||
]
|
]
|
||||||
);
|
);
|
||||||
if (isset($issue['number'])) {
|
if (isset($issue['number'])) {
|
||||||
echo " Archival record: moko-platform#{$issue['number']}\n";
|
echo " Archival record: MokoCLI#{$issue['number']}\n";
|
||||||
}
|
}
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
echo " Warning: could not create archival record: " . $e->getMessage() . "\n";
|
echo " Warning: could not create archival record: " . $e->getMessage() . "\n";
|
||||||
|
|||||||
+3
-3
@@ -14,9 +14,9 @@
|
|||||||
* (at your option) any later version.
|
* (at your option) any later version.
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Enterprise.CLI
|
* DEFGROUP: MokoCLI.Enterprise.CLI
|
||||||
* INGROUP: MokoPlatform.Enterprise
|
* INGROUP: MokoCLI.Enterprise
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/audit_query.php
|
* PATH: /cli/audit_query.php
|
||||||
* BRIEF: Search, filter, and export audit logs
|
* BRIEF: Search, filter, and export audit logs
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/badge_update.php
|
* PATH: /cli/badge_update.php
|
||||||
* BRIEF: Update [VERSION: XX.XX.XX] badges in all markdown files
|
* BRIEF: Update [VERSION: XX.XX.XX] badges in all markdown files
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/branch_rename.php
|
* PATH: /cli/branch_rename.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Rename a git branch via Gitea API (create new, update PR, delete old)
|
* BRIEF: Rename a git branch via Gitea API (create new, update PR, delete old)
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/bulk_workflow_push.php
|
* PATH: /cli/bulk_workflow_push.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Push a workflow file to all governed repos via the Gitea Contents API
|
* BRIEF: Push a workflow file to all governed repos via the Gitea Contents API
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@@ -154,7 +154,7 @@ class BulkWorkflowPushCli extends CliFramework
|
|||||||
'content' => $encodedContent,
|
'content' => $encodedContent,
|
||||||
'sha' => $remoteSha,
|
'sha' => $remoteSha,
|
||||||
'message' => "chore: sync {$destPath} "
|
'message' => "chore: sync {$destPath} "
|
||||||
. "from moko-platform [skip ci]",
|
. "from MokoCLI [skip ci]",
|
||||||
'branch' => $branch,
|
'branch' => $branch,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
@@ -184,7 +184,7 @@ class BulkWorkflowPushCli extends CliFramework
|
|||||||
$payload = json_encode([
|
$payload = json_encode([
|
||||||
'content' => $encodedContent,
|
'content' => $encodedContent,
|
||||||
'message' => "chore: add {$destPath} "
|
'message' => "chore: add {$destPath} "
|
||||||
. "from moko-platform [skip ci]",
|
. "from MokoCLI [skip ci]",
|
||||||
'branch' => $branch,
|
'branch' => $branch,
|
||||||
]);
|
]);
|
||||||
|
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/bulk_workflow_trigger.php
|
* PATH: /cli/bulk_workflow_trigger.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Trigger a workflow across multiple repos at once
|
* BRIEF: Trigger a workflow across multiple repos at once
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/changelog_promote.php
|
* PATH: /cli/changelog_promote.php
|
||||||
* BRIEF: Promote [Unreleased] section in CHANGELOG.md to a versioned entry
|
* BRIEF: Promote [Unreleased] section in CHANGELOG.md to a versioned entry
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/changelog_prune.php
|
* PATH: /cli/changelog_prune.php
|
||||||
* BRIEF: Prune old CHANGELOG.md entries — keeps [Unreleased] + last N releases
|
* BRIEF: Prune old CHANGELOG.md entries — keeps [Unreleased] + last N releases
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/client_dashboard.php
|
* PATH: /cli/client_dashboard.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Generate unified client dashboard HTML
|
* BRIEF: Generate unified client dashboard HTML
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/client_health_check.php
|
* PATH: /cli/client_health_check.php
|
||||||
* BRIEF: Verify a client site's update server, installed version, and release availability
|
* BRIEF: Verify a client site's update server, installed version, and release availability
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/client_inventory.php
|
* PATH: /cli/client_inventory.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Discover and list all client-waas repos with their server configuration status
|
* BRIEF: Discover and list all client-waas repos with their server configuration status
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/client_provision.php
|
* PATH: /cli/client_provision.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Provision a new client environment end-to-end
|
* BRIEF: Provision a new client environment end-to-end
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
+3
-3
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/completion.php
|
* PATH: /cli/completion.php
|
||||||
* BRIEF: Generate bash/zsh tab completion scripts for bin/moko
|
* BRIEF: Generate bash/zsh tab completion scripts for bin/moko
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -8,9 +8,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/create_project.php
|
* PATH: /cli/create_project.php
|
||||||
* BRIEF: Create baseline GitHub Projects for repositories with standard fields and views
|
* BRIEF: Create baseline GitHub Projects for repositories with standard fields and views
|
||||||
*/
|
*/
|
||||||
@@ -24,7 +24,7 @@ use MokoEnterprise\CliFramework;
|
|||||||
class CreateProjectCli extends CliFramework
|
class CreateProjectCli extends CliFramework
|
||||||
{
|
{
|
||||||
/** @var string[] */
|
/** @var string[] */
|
||||||
private array $ALWAYS_EXCLUDE = ['moko-platform', '.github-private'];
|
private array $ALWAYS_EXCLUDE = ['MokoCLI', '.github-private'];
|
||||||
|
|
||||||
/** @var array<string, string> */
|
/** @var array<string, string> */
|
||||||
private array $PLATFORM_TO_TYPE = [
|
private array $PLATFORM_TO_TYPE = [
|
||||||
@@ -183,7 +183,7 @@ class CreateProjectCli extends CliFramework
|
|||||||
CURLOPT_HTTPHEADER => [
|
CURLOPT_HTTPHEADER => [
|
||||||
'Authorization: bearer ' . $token,
|
'Authorization: bearer ' . $token,
|
||||||
'Content-Type: application/json',
|
'Content-Type: application/json',
|
||||||
'User-Agent: moko-platform-CreateProject',
|
'User-Agent: MokoCLI-CreateProject',
|
||||||
],
|
],
|
||||||
]);
|
]);
|
||||||
$body = (string) curl_exec($ch);
|
$body = (string) curl_exec($ch);
|
||||||
@@ -422,14 +422,14 @@ class CreateProjectCli extends CliFramework
|
|||||||
updateProjectV2(input: {
|
updateProjectV2(input: {
|
||||||
projectId: $projectId,
|
projectId: $projectId,
|
||||||
shortDescription: $shortDescription,
|
shortDescription: $shortDescription,
|
||||||
readme: "Managed by moko-platform. Run `php cli/create_project.php` to regenerate."
|
readme: "Managed by MokoCLI. Run `php cli/create_project.php` to regenerate."
|
||||||
}) {
|
}) {
|
||||||
projectV2 { id }
|
projectV2 { id }
|
||||||
}
|
}
|
||||||
}',
|
}',
|
||||||
[
|
[
|
||||||
'projectId' => $projectId,
|
'projectId' => $projectId,
|
||||||
'shortDescription' => "Standard project board for {$repo}. Auto-created by moko-platform.",
|
'shortDescription' => "Standard project board for {$repo}. Auto-created by MokoCLI.",
|
||||||
],
|
],
|
||||||
$token
|
$token
|
||||||
);
|
);
|
||||||
|
|||||||
+17
-17
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/create_repo.php
|
* PATH: /cli/create_repo.php
|
||||||
* BRIEF: Scaffold a new governed repository with full moko-platform baseline
|
* BRIEF: Scaffold a new governed repository with full MokoCLI baseline
|
||||||
*/
|
*/
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
@@ -28,7 +28,7 @@ class CreateRepoCli extends CliFramework
|
|||||||
{
|
{
|
||||||
protected function configure(): void
|
protected function configure(): void
|
||||||
{
|
{
|
||||||
$this->setDescription('Scaffold a new governed repository with full moko-platform baseline');
|
$this->setDescription('Scaffold a new governed repository with full MokoCLI baseline');
|
||||||
$this->addArgument('--name', 'Repository name', null);
|
$this->addArgument('--name', 'Repository name', null);
|
||||||
$this->addArgument('--type', 'Project type', null);
|
$this->addArgument('--type', 'Project type', null);
|
||||||
$this->addArgument('--description', 'Repository description', '');
|
$this->addArgument('--description', 'Repository description', '');
|
||||||
@@ -60,16 +60,16 @@ class CreateRepoCli extends CliFramework
|
|||||||
'generic' => 'generic',
|
'generic' => 'generic',
|
||||||
];
|
];
|
||||||
$TYPE_TO_TOPICS = [
|
$TYPE_TO_TOPICS = [
|
||||||
'dolibarr' => ['dolibarr', 'erp', 'crm', 'php', 'moko-platform'],
|
'dolibarr' => ['dolibarr', 'erp', 'crm', 'php', 'MokoCLI'],
|
||||||
'joomla' => ['joomla', 'cms', 'php', 'moko-platform'],
|
'joomla' => ['joomla', 'cms', 'php', 'MokoCLI'],
|
||||||
'nodejs' => ['nodejs', 'javascript', 'typescript', 'moko-platform'],
|
'nodejs' => ['nodejs', 'javascript', 'typescript', 'MokoCLI'],
|
||||||
'terraform' => ['terraform', 'infrastructure', 'iac', 'moko-platform'],
|
'terraform' => ['terraform', 'infrastructure', 'iac', 'MokoCLI'],
|
||||||
'python' => ['python', 'moko-platform'],
|
'python' => ['python', 'MokoCLI'],
|
||||||
'wordpress' => ['wordpress', 'php', 'cms', 'moko-platform'],
|
'wordpress' => ['wordpress', 'php', 'cms', 'MokoCLI'],
|
||||||
'generic' => ['moko-platform'],
|
'generic' => ['MokoCLI'],
|
||||||
];
|
];
|
||||||
$platform = $TYPE_TO_PLATFORM[$type] ?? 'generic';
|
$platform = $TYPE_TO_PLATFORM[$type] ?? 'generic';
|
||||||
$topics = $TYPE_TO_TOPICS[$type] ?? ['moko-platform'];
|
$topics = $TYPE_TO_TOPICS[$type] ?? ['MokoCLI'];
|
||||||
$platformName = $adapter->getPlatformName();
|
$platformName = $adapter->getPlatformName();
|
||||||
$vis = $private ? 'private' : 'public';
|
$vis = $private ? 'private' : 'public';
|
||||||
echo "Scaffolding new repository: {$org}/{$name}"
|
echo "Scaffolding new repository: {$org}/{$name}"
|
||||||
@@ -84,7 +84,7 @@ class CreateRepoCli extends CliFramework
|
|||||||
if (!$this->dryRun) {
|
if (!$this->dryRun) {
|
||||||
try {
|
try {
|
||||||
$data = $adapter->createOrgRepo($org, $name, [
|
$data = $adapter->createOrgRepo($org, $name, [
|
||||||
'description' => $description ?: "Managed by moko-platform ({$type})",
|
'description' => $description ?: "Managed by MokoCLI ({$type})",
|
||||||
'private' => $private,
|
'private' => $private,
|
||||||
'has_issues' => true,
|
'has_issues' => true,
|
||||||
'has_projects' => true,
|
'has_projects' => true,
|
||||||
@@ -143,7 +143,7 @@ class CreateRepoCli extends CliFramework
|
|||||||
. "Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>\n"
|
. "Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>\n"
|
||||||
. "SPDX-License-Identifier: GPL-3.0-or-later\n"
|
. "SPDX-License-Identifier: GPL-3.0-or-later\n"
|
||||||
. "DEFGROUP: {$name}\n"
|
. "DEFGROUP: {$name}\n"
|
||||||
. "INGROUP: moko-platform\n"
|
. "INGROUP: MokoCLI\n"
|
||||||
. "REPO: {$repoUrl}\n"
|
. "REPO: {$repoUrl}\n"
|
||||||
. "PATH: /README.md\n"
|
. "PATH: /README.md\n"
|
||||||
. "BRIEF: {$description}\n"
|
. "BRIEF: {$description}\n"
|
||||||
@@ -152,7 +152,7 @@ class CreateRepoCli extends CliFramework
|
|||||||
. "{$description}\n\n"
|
. "{$description}\n\n"
|
||||||
. "## Getting Started\n\n"
|
. "## Getting Started\n\n"
|
||||||
. "This repository is governed by"
|
. "This repository is governed by"
|
||||||
. " [moko-platform]({$standardsUrl}).\n\n"
|
. " [MokoCLI]({$standardsUrl}).\n\n"
|
||||||
. "## License\n\n"
|
. "## License\n\n"
|
||||||
. "GPL-3.0-or-later. See [LICENSE](LICENSE)"
|
. "GPL-3.0-or-later. See [LICENSE](LICENSE)"
|
||||||
. " for details.\n";
|
. " for details.\n";
|
||||||
@@ -169,7 +169,7 @@ class CreateRepoCli extends CliFramework
|
|||||||
$name,
|
$name,
|
||||||
'README.md',
|
'README.md',
|
||||||
$readmeContent,
|
$readmeContent,
|
||||||
'docs: initialize README with moko-platform header [skip ci]',
|
'docs: initialize README with MokoCLI header [skip ci]',
|
||||||
$sha
|
$sha
|
||||||
);
|
);
|
||||||
echo " README.md created\n";
|
echo " README.md created\n";
|
||||||
|
|||||||
@@ -8,9 +8,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: MokoPlatform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/deploy_joomla.php
|
* PATH: /cli/deploy_joomla.php
|
||||||
* BRIEF: Smart Joomla deploy — routes files to correct server directories by extension type
|
* BRIEF: Smart Joomla deploy — routes files to correct server directories by extension type
|
||||||
*
|
*
|
||||||
@@ -31,7 +31,7 @@ require_once __DIR__ . '/../vendor/autoload.php';
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
use phpseclib3\Net\SFTP;
|
use phpseclib3\Net\SFTP;
|
||||||
use phpseclib3\Crypt\PublicKeyLoader;
|
use phpseclib3\Crypt\PublicKeyLoader;
|
||||||
|
|
||||||
@@ -866,11 +866,11 @@ class DeployJoomla extends CliFramework
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
// 3-5. Fallback chain
|
// 3-5. Fallback chain (source/ → src/ → htdocs/)
|
||||||
foreach (['src', 'htdocs'] as $candidate) {
|
$resolved = SourceResolver::resolveAbsolute($repoPath);
|
||||||
if (is_dir("{$repoPath}/{$candidate}")) {
|
if ($resolved !== null) {
|
||||||
return "{$repoPath}/{$candidate}";
|
SourceResolver::warnIfLegacy($repoPath);
|
||||||
}
|
return $resolved;
|
||||||
}
|
}
|
||||||
|
|
||||||
// Last resort: repo root itself
|
// Last resort: repo root itself
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/dev_branch_reset.php
|
* PATH: /cli/dev_branch_reset.php
|
||||||
* BRIEF: Delete and recreate dev branch from main via Gitea API
|
* BRIEF: Delete and recreate dev branch from main via Gitea API
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/grafana_dashboard.php
|
* PATH: /cli/grafana_dashboard.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Manage Grafana dashboards via API
|
* BRIEF: Manage Grafana dashboards via API
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
+8
-13
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/joomla_build.php
|
* PATH: /cli/joomla_build.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Build a Joomla extension ZIP from manifest — all types supported
|
* BRIEF: Build a Joomla extension ZIP from manifest — all types supported
|
||||||
* NOTE: Called by pre-release and auto-release workflows.
|
* NOTE: Called by pre-release and auto-release workflows.
|
||||||
*/
|
*/
|
||||||
@@ -19,7 +19,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class JoomlaBuildCli extends CliFramework
|
class JoomlaBuildCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -49,17 +49,12 @@ class JoomlaBuildCli extends CliFramework
|
|||||||
$path = realpath($path) ?: $path;
|
$path = realpath($path) ?: $path;
|
||||||
|
|
||||||
// ── Find source directory ──────────────────────────────────────────────
|
// ── Find source directory ──────────────────────────────────────────────
|
||||||
$srcDir = null;
|
$srcDir = SourceResolver::resolveAbsolute($path);
|
||||||
foreach (['src', 'htdocs'] as $d) {
|
|
||||||
if (is_dir("{$path}/{$d}")) {
|
|
||||||
$srcDir = "{$path}/{$d}";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if ($srcDir === null) {
|
if ($srcDir === null) {
|
||||||
$this->log('ERROR', "::error::No src/ or htdocs/ directory in {$path}");
|
$this->log('ERROR', "::error::No source/ or src/ directory in {$path}");
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
SourceResolver::warnIfLegacy($path);
|
||||||
|
|
||||||
// ── Find manifest ──────────────────────────────────────────────────────
|
// ── Find manifest ──────────────────────────────────────────────────────
|
||||||
$manifest = $this->findManifest($srcDir);
|
$manifest = $this->findManifest($srcDir);
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/joomla_compat_check.php
|
* PATH: /cli/joomla_compat_check.php
|
||||||
* BRIEF: Check if extension targetplatform regex matches the latest Joomla version
|
* BRIEF: Check if extension targetplatform regex matches the latest Joomla version
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -0,0 +1,472 @@
|
|||||||
|
#!/usr/bin/env php
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/* Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*
|
||||||
|
* FILE INFORMATION
|
||||||
|
* DEFGROUP: MokoCLI.CLI
|
||||||
|
* INGROUP: MokoCLI
|
||||||
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
|
* PATH: /cli/joomla_metadata_validate.php
|
||||||
|
* VERSION: 09.25.05
|
||||||
|
* BRIEF: Validate MokoGitea repo metadata against Joomla extension manifest XML
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
|
use MokoEnterprise\CliFramework;
|
||||||
|
|
||||||
|
class JoomlaMetadataValidateCli extends CliFramework
|
||||||
|
{
|
||||||
|
/** Joomla element prefix map — must match MokoGitea's cleanJoomlaElement() */
|
||||||
|
private const JOOMLA_PREFIX = [
|
||||||
|
'package' => 'pkg_',
|
||||||
|
'component' => 'com_',
|
||||||
|
'module' => 'mod_',
|
||||||
|
'template' => 'tpl_',
|
||||||
|
'library' => 'lib_',
|
||||||
|
'file' => 'file_',
|
||||||
|
];
|
||||||
|
|
||||||
|
protected function configure(): void
|
||||||
|
{
|
||||||
|
$this->setDescription('Validate MokoGitea repo metadata against Joomla extension manifest XML');
|
||||||
|
$this->addArgument('--path', 'Repo root path (default: current directory)', '.');
|
||||||
|
$this->addArgument('--token', 'Gitea API token (or GITEA_TOKEN env)', '');
|
||||||
|
$this->addArgument('--org', 'Gitea org', 'MokoConsulting');
|
||||||
|
$this->addArgument('--repo', 'Repo name (auto-detected from git if empty)', '');
|
||||||
|
$this->addArgument('--api-base', 'Gitea API base URL', 'https://git.mokoconsulting.tech/api/v1');
|
||||||
|
$this->addArgument('--ci', 'CI mode: exit 1 on any error', false);
|
||||||
|
$this->addArgument('--json', 'Output as JSON', false);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function run(): int
|
||||||
|
{
|
||||||
|
$path = realpath($this->getArgument('--path')) ?: $this->getArgument('--path');
|
||||||
|
$token = $this->getArgument('--token') ?: getenv('GITEA_TOKEN') ?: '';
|
||||||
|
$org = $this->getArgument('--org');
|
||||||
|
$repoName = $this->getArgument('--repo');
|
||||||
|
$apiBase = rtrim($this->getArgument('--api-base'), '/');
|
||||||
|
$ciMode = (bool) $this->getArgument('--ci');
|
||||||
|
$jsonMode = (bool) $this->getArgument('--json');
|
||||||
|
|
||||||
|
if (!is_dir($path)) {
|
||||||
|
$this->log('ERROR', "Path does not exist: {$path}");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($repoName === '') {
|
||||||
|
$repoName = $this->detectRepoName($path);
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Step 1: Find the Joomla extension manifest XML ──────────
|
||||||
|
$joomlaXml = $this->findJoomlaManifest($path);
|
||||||
|
|
||||||
|
if ($joomlaXml === null) {
|
||||||
|
$this->log('ERROR', 'No Joomla extension manifest XML found');
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->log('INFO', "Joomla manifest: {$joomlaXml['path']}");
|
||||||
|
|
||||||
|
// ── Step 2: Load MokoGitea metadata ─────────────────────────
|
||||||
|
$metadata = $this->loadMetadata($path, $org, $repoName, $token, $apiBase);
|
||||||
|
|
||||||
|
if ($metadata === null) {
|
||||||
|
$this->log('ERROR', 'Could not load MokoGitea metadata');
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── Step 3: Compare ─────────────────────────────────────────
|
||||||
|
$results = $this->compare($metadata, $joomlaXml, $path);
|
||||||
|
|
||||||
|
// ── Step 4: Output ──────────────────────────────────────────
|
||||||
|
if ($jsonMode) {
|
||||||
|
echo json_encode([
|
||||||
|
'repo' => $repoName,
|
||||||
|
'results' => $results,
|
||||||
|
], JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES) . "\n";
|
||||||
|
} else {
|
||||||
|
$this->printResults($repoName, $results);
|
||||||
|
}
|
||||||
|
|
||||||
|
$errors = count(array_filter($results, fn($r) => $r['status'] === 'error'));
|
||||||
|
|
||||||
|
return ($ciMode && $errors > 0) ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
// =================================================================
|
||||||
|
// Find Joomla manifest XML
|
||||||
|
// =================================================================
|
||||||
|
|
||||||
|
private function findJoomlaManifest(string $root): ?array
|
||||||
|
{
|
||||||
|
// Search common locations for a Joomla extension manifest
|
||||||
|
$candidates = [];
|
||||||
|
|
||||||
|
// Package manifest: source/pkg_*.xml
|
||||||
|
foreach (glob("{$root}/source/pkg_*.xml") as $file) {
|
||||||
|
$candidates[] = $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Component manifest: source/packages/com_*/[name].xml
|
||||||
|
foreach (glob("{$root}/source/packages/com_*/*.xml") as $file) {
|
||||||
|
$basename = basename($file);
|
||||||
|
// Skip access.xml, config.xml, etc.
|
||||||
|
if (in_array($basename, ['access.xml', 'config.xml'], true)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
$candidates[] = $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Direct source/*.xml
|
||||||
|
foreach (glob("{$root}/source/*.xml") as $file) {
|
||||||
|
if (basename($file) !== 'pkg_mokosuitebackup.xml') {
|
||||||
|
// Already caught above
|
||||||
|
}
|
||||||
|
$candidates[] = $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
// src/ fallback
|
||||||
|
foreach (glob("{$root}/src/pkg_*.xml") as $file) {
|
||||||
|
$candidates[] = $file;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find the first one that has <extension type="...">
|
||||||
|
foreach (array_unique($candidates) as $file) {
|
||||||
|
$content = file_get_contents($file);
|
||||||
|
if ($content === false) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (preg_match('/<extension\s[^>]*type=["\']([^"\']+)["\']/', $content, $typeMatch)) {
|
||||||
|
$xml = @simplexml_load_string($content);
|
||||||
|
if ($xml === false) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$type = strtolower($typeMatch[1]);
|
||||||
|
$relPath = str_replace($root . '/', '', $file);
|
||||||
|
$relPath = str_replace($root . '\\', '', $relPath);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'path' => $relPath,
|
||||||
|
'type' => $type,
|
||||||
|
'xml' => $xml,
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// =================================================================
|
||||||
|
// Load metadata (from API)
|
||||||
|
// =================================================================
|
||||||
|
|
||||||
|
private function loadMetadata(string $root, string $org, string $repoName, string $token, string $apiBase): ?array
|
||||||
|
{
|
||||||
|
if ($token !== '') {
|
||||||
|
$url = "{$apiBase}/repos/{$org}/{$repoName}/metadata";
|
||||||
|
$ctx = stream_context_create([
|
||||||
|
'http' => [
|
||||||
|
'header' => "Authorization: token {$token}\r\nAccept: application/json\r\n",
|
||||||
|
'timeout' => 10,
|
||||||
|
],
|
||||||
|
]);
|
||||||
|
|
||||||
|
$body = @file_get_contents($url, false, $ctx);
|
||||||
|
|
||||||
|
if ($body !== false) {
|
||||||
|
$data = json_decode($body, true);
|
||||||
|
if (is_array($data)) {
|
||||||
|
$data['source'] = 'api';
|
||||||
|
return $data;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// =================================================================
|
||||||
|
// Compare metadata against Joomla manifest
|
||||||
|
// =================================================================
|
||||||
|
|
||||||
|
private function compare(array $metadata, array $joomlaXml, string $root): array
|
||||||
|
{
|
||||||
|
$results = [];
|
||||||
|
$xml = $joomlaXml['xml'];
|
||||||
|
$type = $joomlaXml['type'];
|
||||||
|
|
||||||
|
// 1. Extension type
|
||||||
|
$metaType = $this->normalizeExtensionType($metadata['extension_type'] ?? '');
|
||||||
|
$results[] = [
|
||||||
|
'field' => 'extension_type',
|
||||||
|
'metadata' => $metaType,
|
||||||
|
'joomla' => $type,
|
||||||
|
'status' => ($metaType === $type) ? 'ok' : 'error',
|
||||||
|
'message' => ($metaType === $type)
|
||||||
|
? "matches <extension type=\"{$type}\">"
|
||||||
|
: "metadata has \"{$metaType}\" but Joomla manifest has \"{$type}\"",
|
||||||
|
];
|
||||||
|
|
||||||
|
// 2. Element name
|
||||||
|
$metaName = strtolower($metadata['name'] ?? '');
|
||||||
|
$metaElement = $this->deriveElement($metaType, $metaName);
|
||||||
|
$joomlaElement = $this->extractJoomlaElement($xml, $type);
|
||||||
|
|
||||||
|
$elementMatch = ($metaElement === $joomlaElement);
|
||||||
|
$results[] = [
|
||||||
|
'field' => 'element',
|
||||||
|
'metadata' => $metaElement,
|
||||||
|
'joomla' => $joomlaElement,
|
||||||
|
'status' => $elementMatch ? 'ok' : 'error',
|
||||||
|
'message' => $elementMatch
|
||||||
|
? "derived correctly"
|
||||||
|
: "metadata derives \"{$metaElement}\" but Joomla uses \"{$joomlaElement}\"",
|
||||||
|
];
|
||||||
|
|
||||||
|
// 3. Version
|
||||||
|
$metaVersion = $metadata['version'] ?? '';
|
||||||
|
$joomlaVersion = (string) ($xml->version ?? '');
|
||||||
|
|
||||||
|
if ($metaVersion !== '' && $joomlaVersion !== '') {
|
||||||
|
// Strip dev/rc suffixes for comparison (CI bumps these)
|
||||||
|
$metaBase = preg_replace('/-(dev|rc|alpha|beta)\d*$/', '', $metaVersion);
|
||||||
|
$joomlaBase = preg_replace('/-(dev|rc|alpha|beta)\d*$/', '', $joomlaVersion);
|
||||||
|
$versionMatch = ($metaBase === $joomlaBase);
|
||||||
|
|
||||||
|
$results[] = [
|
||||||
|
'field' => 'version',
|
||||||
|
'metadata' => $metaVersion,
|
||||||
|
'joomla' => $joomlaVersion,
|
||||||
|
'status' => $versionMatch ? 'ok' : 'warn',
|
||||||
|
'message' => $versionMatch
|
||||||
|
? 'matches (base version)'
|
||||||
|
: "metadata has \"{$metaVersion}\" but Joomla has \"{$joomlaVersion}\"",
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 4. PHP minimum (from composer.json)
|
||||||
|
$composerPhp = $this->readComposerPhpRequirement($root);
|
||||||
|
$metaPhp = $metadata['php_minimum'] ?? '';
|
||||||
|
|
||||||
|
if ($composerPhp !== '' && $metaPhp !== '') {
|
||||||
|
$phpMatch = ($metaPhp === $composerPhp);
|
||||||
|
$results[] = [
|
||||||
|
'field' => 'php_minimum',
|
||||||
|
'metadata' => $metaPhp,
|
||||||
|
'joomla' => $composerPhp . ' (composer.json)',
|
||||||
|
'status' => $phpMatch ? 'ok' : 'warn',
|
||||||
|
'message' => $phpMatch
|
||||||
|
? 'matches composer.json'
|
||||||
|
: "metadata has \"{$metaPhp}\" but composer.json requires \"{$composerPhp}\"",
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 5. Description
|
||||||
|
$metaDesc = $metadata['description'] ?? '';
|
||||||
|
$joomlaDesc = (string) ($xml->description ?? '');
|
||||||
|
|
||||||
|
// Joomla descriptions are often language keys, skip those
|
||||||
|
if ($metaDesc !== '' && $joomlaDesc !== '' && !str_starts_with($joomlaDesc, 'COM_') && !str_starts_with($joomlaDesc, 'PKG_')) {
|
||||||
|
$descMatch = ($metaDesc === $joomlaDesc);
|
||||||
|
$results[] = [
|
||||||
|
'field' => 'description',
|
||||||
|
'metadata' => substr($metaDesc, 0, 60) . (strlen($metaDesc) > 60 ? '...' : ''),
|
||||||
|
'joomla' => substr($joomlaDesc, 0, 60) . (strlen($joomlaDesc) > 60 ? '...' : ''),
|
||||||
|
'status' => $descMatch ? 'ok' : 'info',
|
||||||
|
'message' => $descMatch ? 'matches' : 'descriptions differ (informational)',
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
return $results;
|
||||||
|
}
|
||||||
|
|
||||||
|
// =================================================================
|
||||||
|
// Helpers
|
||||||
|
// =================================================================
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalize extension_type — map MokoGitea types to Joomla types.
|
||||||
|
*/
|
||||||
|
private function normalizeExtensionType(string $type): string
|
||||||
|
{
|
||||||
|
return match (strtolower($type)) {
|
||||||
|
'joomla-extension' => 'package', // legacy mapping
|
||||||
|
default => strtolower($type),
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Derive the Joomla element name from type + name.
|
||||||
|
* Replicates MokoGitea's cleanJoomlaElement() + prefix logic.
|
||||||
|
*/
|
||||||
|
private function deriveElement(string $type, string $name): string
|
||||||
|
{
|
||||||
|
// Clean: lowercase, strip non-alphanumeric except . _ -
|
||||||
|
$clean = strtolower($name);
|
||||||
|
$clean = preg_replace('/[^a-z0-9._-]/', '', $clean);
|
||||||
|
|
||||||
|
$prefix = self::JOOMLA_PREFIX[$type] ?? '';
|
||||||
|
|
||||||
|
return $prefix . $clean;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Extract the element name from a Joomla manifest XML.
|
||||||
|
* Follows the same logic as Joomla's InstallerAdapter::getElement().
|
||||||
|
*/
|
||||||
|
private function extractJoomlaElement(\SimpleXMLElement $xml, string $type): string
|
||||||
|
{
|
||||||
|
switch ($type) {
|
||||||
|
case 'package':
|
||||||
|
$packagename = (string) ($xml->packagename ?? '');
|
||||||
|
if ($packagename !== '') {
|
||||||
|
return 'pkg_' . strtolower(preg_replace('/[^a-zA-Z0-9._-]/', '', $packagename));
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'component':
|
||||||
|
$element = (string) ($xml->element ?? '');
|
||||||
|
if ($element !== '') {
|
||||||
|
$element = strtolower($element);
|
||||||
|
return str_starts_with($element, 'com_') ? $element : 'com_' . $element;
|
||||||
|
}
|
||||||
|
$name = (string) ($xml->name ?? '');
|
||||||
|
$name = strtolower(preg_replace('/[^a-zA-Z0-9._-]/', '', $name));
|
||||||
|
return str_starts_with($name, 'com_') ? $name : 'com_' . $name;
|
||||||
|
|
||||||
|
case 'module':
|
||||||
|
$element = (string) ($xml->element ?? '');
|
||||||
|
if ($element !== '') {
|
||||||
|
return strtolower($element);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'plugin':
|
||||||
|
// Plugins derive element from the file attribute
|
||||||
|
if (isset($xml->files)) {
|
||||||
|
foreach ($xml->files->children() as $file) {
|
||||||
|
$plugin = (string) ($file->attributes()->plugin ?? '');
|
||||||
|
if ($plugin !== '') {
|
||||||
|
return strtolower($plugin);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case 'library':
|
||||||
|
$libname = (string) ($xml->libraryname ?? '');
|
||||||
|
if ($libname !== '') {
|
||||||
|
return strtolower($libname);
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback: use <name> tag
|
||||||
|
$name = (string) ($xml->name ?? '');
|
||||||
|
return strtolower(preg_replace('/[^a-zA-Z0-9._-]/', '', $name));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read PHP version requirement from composer.json.
|
||||||
|
*/
|
||||||
|
private function readComposerPhpRequirement(string $root): string
|
||||||
|
{
|
||||||
|
$composerFile = "{$root}/composer.json";
|
||||||
|
|
||||||
|
if (!is_file($composerFile)) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$data = json_decode(file_get_contents($composerFile), true);
|
||||||
|
|
||||||
|
if (!is_array($data)) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
$phpReq = $data['require']['php'] ?? '';
|
||||||
|
|
||||||
|
// Extract version number from constraint like ">=8.1"
|
||||||
|
if (preg_match('/(\d+\.\d+)/', $phpReq, $m)) {
|
||||||
|
return $m[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
private function detectRepoName(string $root): string
|
||||||
|
{
|
||||||
|
$gitConfig = "{$root}/.git/config";
|
||||||
|
|
||||||
|
if (!file_exists($gitConfig)) {
|
||||||
|
return basename($root);
|
||||||
|
}
|
||||||
|
|
||||||
|
$content = file_get_contents($gitConfig);
|
||||||
|
|
||||||
|
if (preg_match('/url\s*=\s*.*\/([^\/\s]+?)(?:\.git)?\s*$/m', $content, $m)) {
|
||||||
|
return $m[1];
|
||||||
|
}
|
||||||
|
|
||||||
|
return basename($root);
|
||||||
|
}
|
||||||
|
|
||||||
|
// =================================================================
|
||||||
|
// Output
|
||||||
|
// =================================================================
|
||||||
|
|
||||||
|
private function printResults(string $repoName, array $results): void
|
||||||
|
{
|
||||||
|
$errors = count(array_filter($results, fn($r) => $r['status'] === 'error'));
|
||||||
|
$warns = count(array_filter($results, fn($r) => $r['status'] === 'warn'));
|
||||||
|
$oks = count(array_filter($results, fn($r) => $r['status'] === 'ok'));
|
||||||
|
|
||||||
|
$this->log('INFO', "Validating {$repoName} Joomla metadata...\n");
|
||||||
|
|
||||||
|
foreach ($results as $r) {
|
||||||
|
$icon = match ($r['status']) {
|
||||||
|
'ok' => "\xE2\x9C\x93", // ✓
|
||||||
|
'error' => "\xE2\x9C\x97", // ✗
|
||||||
|
'warn' => "\xE2\x9A\xA0", // ⚠
|
||||||
|
default => "\xE2\x84\xB9", // ℹ
|
||||||
|
};
|
||||||
|
|
||||||
|
$line = sprintf(
|
||||||
|
" %s %-16s %s",
|
||||||
|
$icon,
|
||||||
|
$r['field'],
|
||||||
|
$r['message']
|
||||||
|
);
|
||||||
|
|
||||||
|
$this->log(
|
||||||
|
match ($r['status']) {
|
||||||
|
'error' => 'ERROR',
|
||||||
|
'warn' => 'WARN',
|
||||||
|
'ok' => 'OK',
|
||||||
|
default => 'INFO',
|
||||||
|
},
|
||||||
|
$line
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "\n";
|
||||||
|
|
||||||
|
if ($errors > 0) {
|
||||||
|
$this->log('ERROR', "{$errors} error(s) — update delivery will fail");
|
||||||
|
} elseif ($warns > 0) {
|
||||||
|
$this->log('WARN', "All critical checks passed, {$warns} warning(s)");
|
||||||
|
} else {
|
||||||
|
$this->log('OK', "All {$oks} checks passed");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$app = new JoomlaMetadataValidateCli();
|
||||||
|
exit($app->execute());
|
||||||
@@ -8,9 +8,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/joomla_release.php
|
* PATH: /cli/joomla_release.php
|
||||||
* BRIEF: Joomla release pipeline — build ZIP+tar.gz, upload to GitHub Release, update updates.xml
|
* BRIEF: Joomla release pipeline — build ZIP+tar.gz, upload to GitHub Release, update updates.xml
|
||||||
*
|
*
|
||||||
@@ -25,7 +25,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../vendor/autoload.php';
|
require_once __DIR__ . '/../vendor/autoload.php';
|
||||||
|
|
||||||
use MokoEnterprise\{ApiClient, AuditLogger, CliFramework, Config, PlatformAdapterFactory};
|
use MokoEnterprise\{ApiClient, AuditLogger, CliFramework, Config, PlatformAdapterFactory, SourceResolver};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Joomla Release Manager
|
* Joomla Release Manager
|
||||||
@@ -121,11 +121,12 @@ class JoomlaRelease extends CliFramework
|
|||||||
$this->log('INFO', "Version: {$displayVersion} | Release tag: {$releaseTag}");
|
$this->log('INFO', "Version: {$displayVersion} | Release tag: {$releaseTag}");
|
||||||
|
|
||||||
// ── Step 3: Build packages ────────────────────────────────────
|
// ── Step 3: Build packages ────────────────────────────────────
|
||||||
$srcDir = is_dir("{$path}/src") ? "{$path}/src" : (is_dir("{$path}/htdocs") ? "{$path}/htdocs" : null);
|
$srcDir = SourceResolver::resolveAbsolute($path);
|
||||||
if ($srcDir === null) {
|
if ($srcDir === null) {
|
||||||
$this->log('ERROR', 'No src/ or htdocs/ directory');
|
$this->log('ERROR', 'No source/ or src/ directory');
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
SourceResolver::warnIfLegacy($path);
|
||||||
|
|
||||||
$prefix = $this->typePrefix($meta);
|
$prefix = $this->typePrefix($meta);
|
||||||
$zipName = "{$prefix}{$meta['element']}-{$displayVersion}.zip";
|
$zipName = "{$prefix}{$meta['element']}-{$displayVersion}.zip";
|
||||||
@@ -406,7 +407,7 @@ class JoomlaRelease extends CliFramework
|
|||||||
$this->api->post("/repos/{$repo}/releases", [
|
$this->api->post("/repos/{$repo}/releases", [
|
||||||
'tag_name' => $tag,
|
'tag_name' => $tag,
|
||||||
'name' => $releaseName,
|
'name' => $releaseName,
|
||||||
'body' => "## {$version}\n\nCreated by moko-platform release pipeline.",
|
'body' => "## {$version}\n\nCreated by MokoCLI release pipeline.",
|
||||||
'prerelease' => ($stability !== 'stable'),
|
'prerelease' => ($stability !== 'stable'),
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/license_manage.php
|
* PATH: /cli/license_manage.php
|
||||||
* BRIEF: Manage license packages and keys via MokoGitea licensing API
|
* BRIEF: Manage license packages and keys via MokoGitea licensing API
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/manifest_element.php
|
* PATH: /cli/manifest_element.php
|
||||||
* BRIEF: Extract element name, type, type prefix, and ZIP name from manifest
|
* BRIEF: Extract element name, type, type prefix, and ZIP name from manifest
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class ManifestElementCli extends CliFramework
|
class ManifestElementCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -48,7 +48,7 @@ class ManifestElementCli extends CliFramework
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
$extManifest = null;
|
$extManifest = null;
|
||||||
$manifestFiles = array_merge(glob("{$root}/src/pkg_*.xml") ?: [], glob("{$root}/src/*.xml") ?: [], glob("{$root}/*.xml") ?: []);
|
$manifestFiles = array_merge(SourceResolver::globSource($root, 'pkg_*.xml'), SourceResolver::globSource($root, '*.xml'), glob("{$root}/*.xml") ?: []);
|
||||||
foreach ($manifestFiles as $file) {
|
foreach ($manifestFiles as $file) {
|
||||||
$c = file_get_contents($file);
|
$c = file_get_contents($file);
|
||||||
if (strpos($c, '<extension') !== false) {
|
if (strpos($c, '<extension') !== false) {
|
||||||
@@ -58,8 +58,7 @@ class ManifestElementCli extends CliFramework
|
|||||||
}
|
}
|
||||||
$modFile = null;
|
$modFile = null;
|
||||||
$modFiles = array_merge(
|
$modFiles = array_merge(
|
||||||
glob("{$root}/src/core/modules/mod*.class.php") ?: [],
|
SourceResolver::globSource($root, 'core/modules/mod*.class.php'),
|
||||||
glob("{$root}/htdocs/core/modules/mod*.class.php") ?: [],
|
|
||||||
glob("{$root}/core/modules/mod*.class.php") ?: []
|
glob("{$root}/core/modules/mod*.class.php") ?: []
|
||||||
);
|
);
|
||||||
foreach ($modFiles as $file) {
|
foreach ($modFiles as $file) {
|
||||||
|
|||||||
@@ -0,0 +1,280 @@
|
|||||||
|
#!/usr/bin/env php
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/* Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*
|
||||||
|
* FILE INFORMATION
|
||||||
|
* DEFGROUP: MokoCLI.CLI
|
||||||
|
* INGROUP: MokoCLI
|
||||||
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
|
* PATH: /cli/manifest_licensing.php
|
||||||
|
* VERSION: 09.25.05
|
||||||
|
* BRIEF: Ensure licensing tags (updateservers, dlid) in Joomla extension manifests
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Reads the <licensing> block from .mokogitea/manifest.xml and ensures that the
|
||||||
|
* Joomla extension manifest contains the correct <updateservers> and <dlid> tags.
|
||||||
|
*
|
||||||
|
* manifest.xml licensing block example:
|
||||||
|
*
|
||||||
|
* <licensing>
|
||||||
|
* <enabled>true</enabled>
|
||||||
|
* <dlid>true</dlid>
|
||||||
|
* <update-server>https://git.mokoconsulting.tech/{org}/{repo}/updates.xml</update-server>
|
||||||
|
* <update-server-name>MyExtension Updates</update-server-name>
|
||||||
|
* </licensing>
|
||||||
|
*
|
||||||
|
* Supports {org} and {repo} placeholders in update-server URL, resolved from
|
||||||
|
* the manifest's <identity> block or git remote.
|
||||||
|
*/
|
||||||
|
class ManifestLicensingCli extends CliFramework
|
||||||
|
{
|
||||||
|
protected function configure(): void
|
||||||
|
{
|
||||||
|
$this->setDescription('Ensure licensing tags (updateservers, dlid) in Joomla extension manifests');
|
||||||
|
$this->addArgument('--path', 'Repository root path', '.');
|
||||||
|
$this->addArgument('--fix', 'Apply fixes (default: dry-run check only)', false);
|
||||||
|
$this->addArgument('--github-output', 'Write results to $GITHUB_OUTPUT', false);
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function run(): int
|
||||||
|
{
|
||||||
|
$root = realpath($this->getArgument('--path')) ?: $this->getArgument('--path');
|
||||||
|
$fix = (bool) $this->getArgument('--fix');
|
||||||
|
$ghOutput = (bool) $this->getArgument('--github-output');
|
||||||
|
|
||||||
|
// ── 1. Read manifest.xml ──────────────────────────────────────────
|
||||||
|
$manifestFile = "{$root}/.mokogitea/manifest.xml";
|
||||||
|
|
||||||
|
if (!file_exists($manifestFile)) {
|
||||||
|
$this->log('WARN', "No manifest.xml found at {$manifestFile}");
|
||||||
|
$this->outputResult($ghOutput, 'skipped', 'No manifest.xml');
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$xml = @simplexml_load_file($manifestFile);
|
||||||
|
|
||||||
|
if ($xml === false) {
|
||||||
|
$this->log('ERROR', "Failed to parse {$manifestFile}");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── 2. Check if licensing is enabled ──────────────────────────────
|
||||||
|
if (!isset($xml->licensing) || (string) ($xml->licensing->enabled ?? '') !== 'true') {
|
||||||
|
$this->log('INFO', 'Licensing not enabled in manifest.xml — skipping');
|
||||||
|
$this->outputResult($ghOutput, 'skipped', 'Licensing not enabled');
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$licensingNode = $xml->licensing;
|
||||||
|
$dlidEnabled = ((string) ($licensingNode->dlid ?? 'true')) === 'true';
|
||||||
|
$updateServerUrl = (string) ($licensingNode->{'update-server'} ?? '');
|
||||||
|
$updateServerName = (string) ($licensingNode->{'update-server-name'} ?? '');
|
||||||
|
|
||||||
|
// ── 3. Resolve placeholders ───────────────────────────────────────
|
||||||
|
$org = (string) ($xml->identity->org ?? '');
|
||||||
|
$repo = (string) ($xml->identity->name ?? '');
|
||||||
|
|
||||||
|
// Fallback to git remote if manifest doesn't have org/name
|
||||||
|
if (empty($org) || empty($repo)) {
|
||||||
|
$remote = trim((string) @shell_exec("cd " . escapeshellarg($root) . " && git remote get-url origin 2>/dev/null"));
|
||||||
|
|
||||||
|
if (preg_match('#[/:]([^/]+)/([^/.]+?)(?:\.git)?$#', $remote, $m)) {
|
||||||
|
if (empty($org)) {
|
||||||
|
$org = $m[1];
|
||||||
|
}
|
||||||
|
if (empty($repo)) {
|
||||||
|
$repo = $m[2];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Default update server URL if not specified
|
||||||
|
if (empty($updateServerUrl) && !empty($org) && !empty($repo)) {
|
||||||
|
$updateServerUrl = "https://git.mokoconsulting.tech/{$org}/{$repo}/updates.xml";
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resolve {org} and {repo} placeholders
|
||||||
|
$updateServerUrl = str_replace(['{org}', '{repo}'], [$org, $repo], $updateServerUrl);
|
||||||
|
|
||||||
|
// Default server name from display-name or repo name
|
||||||
|
if (empty($updateServerName)) {
|
||||||
|
$displayName = (string) ($xml->identity->{'display-name'} ?? $repo);
|
||||||
|
$updateServerName = $displayName . ' Updates';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (empty($updateServerUrl)) {
|
||||||
|
$this->log('ERROR', 'Cannot determine update server URL — set <update-server> in manifest.xml or ensure org/repo are available');
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->log('INFO', "Licensing enabled — org={$org}, repo={$repo}");
|
||||||
|
$this->log('INFO', "Update server: {$updateServerUrl}");
|
||||||
|
$this->log('INFO', "DLID required: " . ($dlidEnabled ? 'yes' : 'no'));
|
||||||
|
|
||||||
|
// ── 4. Find Joomla extension manifests ────────────────────────────
|
||||||
|
$xmlFiles = array_merge(
|
||||||
|
SourceResolver::globSource($root, '*.xml'),
|
||||||
|
SourceResolver::globSource($root, 'packages/*/*.xml'),
|
||||||
|
glob("{$root}/*.xml") ?: []
|
||||||
|
);
|
||||||
|
|
||||||
|
$packageManifest = null;
|
||||||
|
|
||||||
|
foreach ($xmlFiles as $file) {
|
||||||
|
$content = file_get_contents($file);
|
||||||
|
|
||||||
|
if (!str_contains($content, '<extension')) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Find the package manifest (type="package") or the main extension manifest
|
||||||
|
if (str_contains($content, 'type="package"')) {
|
||||||
|
$packageManifest = $file;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback: first extension manifest found
|
||||||
|
if ($packageManifest === null) {
|
||||||
|
$packageManifest = $file;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($packageManifest === null) {
|
||||||
|
$this->log('WARN', 'No Joomla extension manifest found');
|
||||||
|
$this->outputResult($ghOutput, 'skipped', 'No extension manifest');
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$relPath = str_replace($root . '/', '', str_replace('\\', '/', $packageManifest));
|
||||||
|
$this->log('INFO', "Package manifest: {$relPath}");
|
||||||
|
|
||||||
|
// ── 5. Check and fix the manifest ─────────────────────────────────
|
||||||
|
$content = file_get_contents($packageManifest);
|
||||||
|
$original = $content;
|
||||||
|
$changes = [];
|
||||||
|
|
||||||
|
// --- 5a. Ensure <updateservers> block with correct URL ---
|
||||||
|
if (preg_match('#<updateservers>\s*</updateservers>#s', $content)) {
|
||||||
|
// Empty updateservers block — inject the server
|
||||||
|
$replacement = "<updateservers>\n"
|
||||||
|
. " <server type=\"extension\" name=\"{$updateServerName}\">{$updateServerUrl}</server>\n"
|
||||||
|
. " </updateservers>";
|
||||||
|
$content = preg_replace('#<updateservers>\s*</updateservers>#s', $replacement, $content);
|
||||||
|
$changes[] = 'Added update server URL to empty <updateservers>';
|
||||||
|
} elseif (!str_contains($content, '<updateservers>')) {
|
||||||
|
// No updateservers at all — add before </extension>
|
||||||
|
$serverBlock = "\n <updateservers>\n"
|
||||||
|
. " <server type=\"extension\" name=\"{$updateServerName}\">{$updateServerUrl}</server>\n"
|
||||||
|
. " </updateservers>\n";
|
||||||
|
$content = str_replace('</extension>', $serverBlock . '</extension>', $content);
|
||||||
|
$changes[] = 'Added <updateservers> block';
|
||||||
|
} else {
|
||||||
|
// updateservers exists — verify URL is correct
|
||||||
|
if (preg_match('#<server[^>]*>([^<]+)</server>#', $content, $m)) {
|
||||||
|
if ($m[1] !== $updateServerUrl) {
|
||||||
|
$content = preg_replace(
|
||||||
|
'#(<server[^>]*>)[^<]+(</server>)#',
|
||||||
|
"\${1}{$updateServerUrl}\${2}",
|
||||||
|
$content
|
||||||
|
);
|
||||||
|
$changes[] = "Updated server URL: {$m[1]} → {$updateServerUrl}";
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- 5b. Ensure <dlid> tag if required ---
|
||||||
|
if ($dlidEnabled) {
|
||||||
|
if (!str_contains($content, '<dlid')) {
|
||||||
|
// Add before <updateservers> if present, otherwise before </extension>
|
||||||
|
$dlidTag = ' <dlid prefix="dlid=" suffix=""/>' . "\n";
|
||||||
|
|
||||||
|
if (str_contains($content, '<updateservers>')) {
|
||||||
|
$content = str_replace('<updateservers>', $dlidTag . "\n <updateservers>", $content);
|
||||||
|
} else {
|
||||||
|
$content = str_replace('</extension>', $dlidTag . '</extension>', $content);
|
||||||
|
}
|
||||||
|
|
||||||
|
$changes[] = 'Added <dlid> tag';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// --- 5c. Ensure <blockChildUninstall> for packages ---
|
||||||
|
if (str_contains($content, 'type="package"') && !str_contains($content, '<blockChildUninstall>')) {
|
||||||
|
$blockTag = ' <blockChildUninstall>true</blockChildUninstall>' . "\n";
|
||||||
|
|
||||||
|
if (str_contains($content, '<dlid')) {
|
||||||
|
// Add after <dlid>
|
||||||
|
$content = preg_replace(
|
||||||
|
'#(<dlid[^/]*/>\s*\n)#',
|
||||||
|
"\${1}{$blockTag}",
|
||||||
|
$content
|
||||||
|
);
|
||||||
|
} elseif (str_contains($content, '<updateservers>')) {
|
||||||
|
$content = str_replace('<updateservers>', $blockTag . "\n <updateservers>", $content);
|
||||||
|
} else {
|
||||||
|
$content = str_replace('</extension>', $blockTag . '</extension>', $content);
|
||||||
|
}
|
||||||
|
|
||||||
|
$changes[] = 'Added <blockChildUninstall>true</blockChildUninstall>';
|
||||||
|
}
|
||||||
|
|
||||||
|
// ── 6. Report and apply ───────────────────────────────────────────
|
||||||
|
if (empty($changes)) {
|
||||||
|
$this->log('INFO', 'All licensing tags are correct — no changes needed');
|
||||||
|
$this->outputResult($ghOutput, 'ok', 'No changes needed');
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($changes as $change) {
|
||||||
|
$this->log($fix ? 'INFO' : 'WARN', ($fix ? 'Fixed: ' : 'Needs fix: ') . $change);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($fix) {
|
||||||
|
file_put_contents($packageManifest, $content);
|
||||||
|
$this->log('INFO', "Wrote {$relPath} with " . count($changes) . " change(s)");
|
||||||
|
$this->outputResult($ghOutput, 'fixed', implode('; ', $changes));
|
||||||
|
} else {
|
||||||
|
$this->log('WARN', 'Run with --fix to apply changes');
|
||||||
|
$this->outputResult($ghOutput, 'needs-fix', implode('; ', $changes));
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Write result to $GITHUB_OUTPUT if requested.
|
||||||
|
*/
|
||||||
|
private function outputResult(bool $ghOutput, string $status, string $detail): void
|
||||||
|
{
|
||||||
|
if (!$ghOutput) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$outputFile = getenv('GITHUB_OUTPUT');
|
||||||
|
|
||||||
|
if ($outputFile === false || $outputFile === '') {
|
||||||
|
echo "licensing_status={$status}\n";
|
||||||
|
echo "licensing_detail={$detail}\n";
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$fh = fopen($outputFile, 'a');
|
||||||
|
fwrite($fh, "licensing_status={$status}\n");
|
||||||
|
fwrite($fh, "licensing_detail={$detail}\n");
|
||||||
|
fclose($fh);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$app = new ManifestLicensingCli();
|
||||||
|
exit($app->execute());
|
||||||
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/manifest_read.php
|
* PATH: /cli/manifest_read.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Parse .manifest.xml and output requested field(s) for CI consumption
|
* BRIEF: Parse .manifest.xml and output requested field(s) for CI consumption
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@@ -59,7 +59,7 @@ class ManifestReadCli extends CliFramework
|
|||||||
$candidates = [
|
$candidates = [
|
||||||
"{$root}/.mokogitea/manifest.xml",
|
"{$root}/.mokogitea/manifest.xml",
|
||||||
"{$root}/.mokogitea/.manifest.xml", // legacy (dot-prefixed)
|
"{$root}/.mokogitea/.manifest.xml", // legacy (dot-prefixed)
|
||||||
"{$root}/.mokogitea/.moko-platform", // legacy v4
|
"{$root}/.mokogitea/.MokoCLI", // legacy v4
|
||||||
];
|
];
|
||||||
|
|
||||||
foreach ($candidates as $candidate) {
|
foreach ($candidates as $candidate) {
|
||||||
|
|||||||
+7
-12
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/package_build.php
|
* PATH: /cli/package_build.php
|
||||||
* BRIEF: Build ZIP and tar.gz install packages for Joomla/Dolibarr/generic projects
|
* BRIEF: Build ZIP and tar.gz install packages for Joomla/Dolibarr/generic projects
|
||||||
*
|
*
|
||||||
@@ -19,7 +19,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class PackageBuildCli extends CliFramework
|
class PackageBuildCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -56,18 +56,13 @@ class PackageBuildCli extends CliFramework
|
|||||||
}
|
}
|
||||||
|
|
||||||
// -- Determine source directory -----------------------------------------------
|
// -- Determine source directory -----------------------------------------------
|
||||||
$sourceDir = null;
|
$sourceDir = SourceResolver::resolveAbsolute($root);
|
||||||
foreach (['src', 'htdocs'] as $candidate) {
|
|
||||||
if (is_dir("{$root}/{$candidate}")) {
|
|
||||||
$sourceDir = "{$root}/{$candidate}";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($sourceDir === null) {
|
if ($sourceDir === null) {
|
||||||
$this->log('ERROR', "No src/ or htdocs/ directory found in {$root}");
|
$this->log('ERROR', "No source/ or src/ directory found in {$root}");
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
SourceResolver::warnIfLegacy($root);
|
||||||
|
|
||||||
// -- Determine element and type prefix from manifest --------------------------
|
// -- Determine element and type prefix from manifest --------------------------
|
||||||
$extElement = $elementOverride;
|
$extElement = $elementOverride;
|
||||||
|
|||||||
+162
-19
@@ -6,11 +6,12 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/platform_detect.php
|
* PATH: /cli/platform_detect.php
|
||||||
* BRIEF: Detect platform from manifest.xml file — outputs platform string
|
* VERSION: 09.25.05
|
||||||
|
* BRIEF: Auto-detect repository platform type and optionally update manifest
|
||||||
*/
|
*/
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
@@ -23,8 +24,14 @@ class PlatformDetectCli extends CliFramework
|
|||||||
{
|
{
|
||||||
protected function configure(): void
|
protected function configure(): void
|
||||||
{
|
{
|
||||||
$this->setDescription('Detect platform from manifest.xml file');
|
$this->setDescription('Auto-detect repository platform type and optionally update manifest');
|
||||||
$this->addArgument('--path', 'Repository root path', '.');
|
$this->addArgument('--path', 'Local repo path to scan (default: .)', '.');
|
||||||
|
$this->addArgument('--token', 'Gitea API token for updating manifest', '');
|
||||||
|
$this->addArgument('--gitea-url', 'Gitea URL (default: https://git.mokoconsulting.tech)', 'https://git.mokoconsulting.tech');
|
||||||
|
$this->addArgument('--owner', 'Repo owner for API update', '');
|
||||||
|
$this->addArgument('--repo', 'Repo name for API update', '');
|
||||||
|
$this->addArgument('--update', 'Update manifest.platform via API (flag)', 'false');
|
||||||
|
$this->addArgument('--github-output', 'Append platform=xxx to $GITHUB_OUTPUT (flag)', 'false');
|
||||||
}
|
}
|
||||||
|
|
||||||
protected function run(): int
|
protected function run(): int
|
||||||
@@ -32,25 +39,161 @@ class PlatformDetectCli extends CliFramework
|
|||||||
$path = $this->getArgument('--path');
|
$path = $this->getArgument('--path');
|
||||||
$root = realpath($path) ?: $path;
|
$root = realpath($path) ?: $path;
|
||||||
|
|
||||||
// Check .mokogitea/manifest.xml first, fallback to root
|
$token = $this->getArgument('--token');
|
||||||
$file = "{$root}/.mokogitea/manifest.xml";
|
$giteaUrl = rtrim($this->getArgument('--gitea-url'), '/');
|
||||||
if (!file_exists($file)) {
|
$owner = $this->getArgument('--owner');
|
||||||
$file = "{$root}/.mokostandards";
|
$repo = $this->getArgument('--repo');
|
||||||
}
|
$doUpdate = $this->isFlagSet('--update');
|
||||||
if (!file_exists($file)) {
|
$githubOutput = $this->isFlagSet('--github-output');
|
||||||
echo "unknown\n";
|
|
||||||
return 0;
|
$platform = $this->detectPlatform($root);
|
||||||
|
|
||||||
|
$this->log('INFO', "Detected platform: {$platform}");
|
||||||
|
echo $platform . "\n";
|
||||||
|
|
||||||
|
// Append to $GITHUB_OUTPUT if requested
|
||||||
|
if ($githubOutput) {
|
||||||
|
$outputFile = getenv('GITHUB_OUTPUT');
|
||||||
|
|
||||||
|
if ($outputFile !== false && $outputFile !== '') {
|
||||||
|
file_put_contents($outputFile, "platform={$platform}\n", FILE_APPEND);
|
||||||
|
$this->log('INFO', "Appended platform={$platform} to \$GITHUB_OUTPUT");
|
||||||
|
} else {
|
||||||
|
$this->log('WARN', '$GITHUB_OUTPUT is not set; skipping output append.');
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$content = file_get_contents($file);
|
// Update manifest via API if requested
|
||||||
if (preg_match('/^platform:\s*(.+)/m', $content, $m)) {
|
if ($doUpdate) {
|
||||||
echo trim($m[1], " \t\n\r\"'") . "\n";
|
if ($token === '' || $owner === '' || $repo === '') {
|
||||||
} else {
|
$this->log('ERROR', '--update requires --token, --owner, and --repo.');
|
||||||
echo "unknown\n";
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->dryRun) {
|
||||||
|
$this->log('INFO', "[DRY RUN] Would update manifest.platform to \"{$platform}\" "
|
||||||
|
. "for {$owner}/{$repo}.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->log('INFO', "Updating manifest.platform for {$owner}/{$repo} to \"{$platform}\"...");
|
||||||
|
|
||||||
|
$response = $this->apiRequest(
|
||||||
|
$giteaUrl,
|
||||||
|
$token,
|
||||||
|
'PATCH',
|
||||||
|
"/api/v1/repos/{$owner}/{$repo}/manifest",
|
||||||
|
json_encode(['platform' => $platform])
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($response['code'] >= 200 && $response['code'] < 300) {
|
||||||
|
$this->log('INFO', "Manifest updated successfully (HTTP {$response['code']}).");
|
||||||
|
} else {
|
||||||
|
$this->log('ERROR', "Failed to update manifest (HTTP {$response['code']}): "
|
||||||
|
. $response['body']);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
private function detectPlatform(string $root): string
|
||||||
|
{
|
||||||
|
// 1. Joomla — has pkg_*.xml or Joomla-style extension manifest
|
||||||
|
$joomlaIndicators = array_merge(
|
||||||
|
glob("{$root}/source/pkg_*.xml") ?: [],
|
||||||
|
glob("{$root}/pkg_*.xml") ?: [],
|
||||||
|
glob("{$root}/source/packages/*/services/provider.php") ?: [],
|
||||||
|
glob("{$root}/**/templateDetails.xml") ?: [],
|
||||||
|
);
|
||||||
|
if (!empty($joomlaIndicators)) {
|
||||||
|
return 'joomla';
|
||||||
|
}
|
||||||
|
|
||||||
|
// 2. Dolibarr — has mod*.class.php or dolibarr module descriptor
|
||||||
|
$doliIndicators = array_merge(
|
||||||
|
glob("{$root}/core/modules/mod*.class.php") ?: [],
|
||||||
|
glob("{$root}/class/*.class.php") ?: [],
|
||||||
|
);
|
||||||
|
if (!empty($doliIndicators) && file_exists("{$root}/langs")) {
|
||||||
|
return 'dolibarr';
|
||||||
|
}
|
||||||
|
|
||||||
|
// 3. Go — has go.mod
|
||||||
|
if (file_exists("{$root}/go.mod")) {
|
||||||
|
return 'go';
|
||||||
|
}
|
||||||
|
|
||||||
|
// 4. MCP — has package.json with mcp-related content or dist/index.js pattern
|
||||||
|
if (file_exists("{$root}/package.json")) {
|
||||||
|
$pkg = json_decode(file_get_contents("{$root}/package.json"), true);
|
||||||
|
$name = $pkg['name'] ?? '';
|
||||||
|
if (str_contains($name, 'mcp') || isset($pkg['dependencies']['@modelcontextprotocol/sdk'])) {
|
||||||
|
return 'mcp';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 5. Platform — is MokoCLI itself or org-config
|
||||||
|
$repoName = basename($root);
|
||||||
|
if (in_array($repoName, ['mokocli', 'mokoplatform', 'mokogitea-org-config'])) {
|
||||||
|
return 'platform';
|
||||||
|
}
|
||||||
|
|
||||||
|
// 6. Default
|
||||||
|
return 'generic';
|
||||||
|
}
|
||||||
|
|
||||||
|
private function isFlagSet(string $flag): bool
|
||||||
|
{
|
||||||
|
$value = $this->getArgument($flag);
|
||||||
|
|
||||||
|
return $value === 'true' || $value === '1' || $value === 'yes';
|
||||||
|
}
|
||||||
|
|
||||||
|
private function apiRequest(
|
||||||
|
string $giteaUrl,
|
||||||
|
string $token,
|
||||||
|
string $method,
|
||||||
|
string $endpoint,
|
||||||
|
?string $body = null
|
||||||
|
): array {
|
||||||
|
$url = $giteaUrl . $endpoint;
|
||||||
|
|
||||||
|
$ch = curl_init();
|
||||||
|
curl_setopt($ch, CURLOPT_URL, $url);
|
||||||
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||||
|
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method);
|
||||||
|
curl_setopt($ch, CURLOPT_HTTPHEADER, [
|
||||||
|
'Content-Type: application/json',
|
||||||
|
'Accept: application/json',
|
||||||
|
"Authorization: token {$token}",
|
||||||
|
]);
|
||||||
|
|
||||||
|
if ($body !== null) {
|
||||||
|
curl_setopt($ch, CURLOPT_POSTFIELDS, $body);
|
||||||
|
}
|
||||||
|
|
||||||
|
$responseBody = curl_exec($ch);
|
||||||
|
$httpCode = (int) curl_getinfo(
|
||||||
|
$ch,
|
||||||
|
CURLINFO_HTTP_CODE
|
||||||
|
);
|
||||||
|
|
||||||
|
if (curl_errno($ch)) {
|
||||||
|
$error = curl_error($ch);
|
||||||
|
curl_close($ch);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'code' => 0,
|
||||||
|
'body' => "cURL error: {$error}",
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
curl_close($ch);
|
||||||
|
|
||||||
|
return ['code' => $httpCode, 'body' => $responseBody];
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
$app = new PlatformDetectCli();
|
$app = new PlatformDetectCli();
|
||||||
|
|||||||
+5
-5
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release.php
|
* PATH: /cli/release.php
|
||||||
* BRIEF: Automate the moko-platform version branch release flow
|
* BRIEF: Automate the MokoCLI version branch release flow
|
||||||
*/
|
*/
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
@@ -23,7 +23,7 @@ class ReleaseCli extends CliFramework
|
|||||||
{
|
{
|
||||||
protected function configure(): void
|
protected function configure(): void
|
||||||
{
|
{
|
||||||
$this->setDescription('Automate the moko-platform version branch release flow');
|
$this->setDescription('Automate the MokoCLI version branch release flow');
|
||||||
$this->addArgument('--bump', 'Bump type: patch, minor, or major', '');
|
$this->addArgument('--bump', 'Bump type: patch, minor, or major', '');
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_body_update.php
|
* PATH: /cli/release_body_update.php
|
||||||
* BRIEF: Update Gitea release body with changelog extract and checksums
|
* BRIEF: Update Gitea release body with changelog extract and checksums
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_cascade.php
|
* PATH: /cli/release_cascade.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: DEPRECATED — cascade behavior removed. Each release stream is independent.
|
* BRIEF: DEPRECATED — cascade behavior removed. Each release stream is independent.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_create.php
|
* PATH: /cli/release_create.php
|
||||||
* BRIEF: Create or overwrite a Gitea release with proper naming
|
* BRIEF: Create or overwrite a Gitea release with proper naming
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class ReleaseCreateCli extends CliFramework
|
class ReleaseCreateCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -97,8 +97,8 @@ class ReleaseCreateCli extends CliFramework
|
|||||||
// Find extension manifest (Joomla XML)
|
// Find extension manifest (Joomla XML)
|
||||||
$extManifest = null;
|
$extManifest = null;
|
||||||
$manifestFiles = array_merge(
|
$manifestFiles = array_merge(
|
||||||
glob("{$root}/src/pkg_*.xml") ?: [],
|
SourceResolver::globSource($root, 'pkg_*.xml'),
|
||||||
glob("{$root}/src/*.xml") ?: [],
|
SourceResolver::globSource($root, '*.xml'),
|
||||||
glob("{$root}/*.xml") ?: []
|
glob("{$root}/*.xml") ?: []
|
||||||
);
|
);
|
||||||
foreach ($manifestFiles as $file) {
|
foreach ($manifestFiles as $file) {
|
||||||
@@ -112,8 +112,7 @@ class ReleaseCreateCli extends CliFramework
|
|||||||
// Find Dolibarr module file
|
// Find Dolibarr module file
|
||||||
$modFile = null;
|
$modFile = null;
|
||||||
$modFiles = array_merge(
|
$modFiles = array_merge(
|
||||||
glob("{$root}/src/core/modules/mod*.class.php") ?: [],
|
SourceResolver::globSource($root, 'core/modules/mod*.class.php'),
|
||||||
glob("{$root}/htdocs/core/modules/mod*.class.php") ?: [],
|
|
||||||
glob("{$root}/core/modules/mod*.class.php") ?: []
|
glob("{$root}/core/modules/mod*.class.php") ?: []
|
||||||
);
|
);
|
||||||
foreach ($modFiles as $file) {
|
foreach ($modFiles as $file) {
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_manage.php
|
* PATH: /cli/release_manage.php
|
||||||
* BRIEF: Create/update Gitea releases, upload assets, update release body
|
* BRIEF: Create/update Gitea releases, upload assets, update release body
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_mirror.php
|
* PATH: /cli/release_mirror.php
|
||||||
* BRIEF: Mirror a Gitea release (with assets) to a GitHub repository
|
* BRIEF: Mirror a Gitea release (with assets) to a GitHub repository
|
||||||
*/
|
*/
|
||||||
@@ -201,7 +201,7 @@ class ReleaseMirrorCli extends CliFramework
|
|||||||
CURLOPT_HTTPHEADER => [
|
CURLOPT_HTTPHEADER => [
|
||||||
"Authorization: token {$token}",
|
"Authorization: token {$token}",
|
||||||
'Accept: application/vnd.github+json',
|
'Accept: application/vnd.github+json',
|
||||||
'User-Agent: moko-platform',
|
'User-Agent: MokoCLI',
|
||||||
'Content-Type: application/json',
|
'Content-Type: application/json',
|
||||||
],
|
],
|
||||||
CURLOPT_TIMEOUT => 30,
|
CURLOPT_TIMEOUT => 30,
|
||||||
@@ -229,7 +229,7 @@ class ReleaseMirrorCli extends CliFramework
|
|||||||
CURLOPT_HTTPHEADER => [
|
CURLOPT_HTTPHEADER => [
|
||||||
"Authorization: token {$token}",
|
"Authorization: token {$token}",
|
||||||
'Accept: application/vnd.github+json',
|
'Accept: application/vnd.github+json',
|
||||||
'User-Agent: moko-platform',
|
'User-Agent: MokoCLI',
|
||||||
'Content-Type: application/octet-stream',
|
'Content-Type: application/octet-stream',
|
||||||
],
|
],
|
||||||
CURLOPT_POSTFIELDS => file_get_contents($filePath),
|
CURLOPT_POSTFIELDS => file_get_contents($filePath),
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_notes.php
|
* PATH: /cli/release_notes.php
|
||||||
* BRIEF: Extract release notes from CHANGELOG.md for a given version
|
* BRIEF: Extract release notes from CHANGELOG.md for a given version
|
||||||
*/
|
*/
|
||||||
|
|||||||
+18
-18
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_package.php
|
* PATH: /cli/release_package.php
|
||||||
* BRIEF: Build packages (ZIP + tar.gz) with SHA-256 and upload to Gitea release
|
* BRIEF: Build packages (ZIP + tar.gz) with SHA-256 and upload to Gitea release
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class ReleasePackageCli extends CliFramework
|
class ReleasePackageCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -99,9 +99,10 @@ class ReleasePackageCli extends CliFramework
|
|||||||
$extFolder = '';
|
$extFolder = '';
|
||||||
$typePrefix = '';
|
$typePrefix = '';
|
||||||
|
|
||||||
|
SourceResolver::warnIfLegacy($root);
|
||||||
$manifestFiles = array_merge(
|
$manifestFiles = array_merge(
|
||||||
glob("{$root}/src/pkg_*.xml") ?: [],
|
SourceResolver::globSource($root, 'pkg_*.xml'),
|
||||||
glob("{$root}/src/*.xml") ?: [],
|
SourceResolver::globSource($root, '*.xml'),
|
||||||
glob("{$root}/*.xml") ?: []
|
glob("{$root}/*.xml") ?: []
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -200,14 +201,12 @@ class ReleasePackageCli extends CliFramework
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($sourceDir === null && is_dir("{$root}/src")) {
|
if ($sourceDir === null) {
|
||||||
$sourceDir = "{$root}/src";
|
$sourceDir = SourceResolver::resolveAbsolute($root);
|
||||||
} elseif ($sourceDir === null && is_dir("{$root}/htdocs")) {
|
|
||||||
$sourceDir = "{$root}/htdocs";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($sourceDir === null) {
|
if ($sourceDir === null) {
|
||||||
echo "No src/ or htdocs/ directory found — skipping package build\n";
|
echo "No source/ or src/ directory found — skipping package build\n";
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -231,19 +230,20 @@ class ReleasePackageCli extends CliFramework
|
|||||||
$subZipPath = "{$outputDir}/{$subName}.zip";
|
$subZipPath = "{$outputDir}/{$subName}.zip";
|
||||||
|
|
||||||
// If sub-package is a full repo checkout (e.g. git submodule),
|
// If sub-package is a full repo checkout (e.g. git submodule),
|
||||||
// look for a src/ subdirectory containing a Joomla manifest XML
|
// look for a source/ or src/ subdirectory containing a Joomla manifest XML
|
||||||
// and zip that instead of the repo root.
|
// and zip that instead of the repo root.
|
||||||
$subSourceDir = $pkgDir;
|
$subSourceDir = $pkgDir;
|
||||||
$srcCandidate = "{$pkgDir}/src";
|
$subSrcAbs = SourceResolver::resolveAbsolute($pkgDir);
|
||||||
if (is_dir($srcCandidate)) {
|
if ($subSrcAbs !== null) {
|
||||||
$srcManifests = array_merge(
|
$srcManifests = array_merge(
|
||||||
glob("{$srcCandidate}/*.xml") ?: [],
|
glob("{$subSrcAbs}/*.xml") ?: [],
|
||||||
glob("{$srcCandidate}/pkg_*.xml") ?: []
|
glob("{$subSrcAbs}/pkg_*.xml") ?: []
|
||||||
);
|
);
|
||||||
foreach ($srcManifests as $mf) {
|
foreach ($srcManifests as $mf) {
|
||||||
if (strpos(file_get_contents($mf) ?: '', '<extension') !== false) {
|
if (strpos(file_get_contents($mf) ?: '', '<extension') !== false) {
|
||||||
$subSourceDir = $srcCandidate;
|
$subSourceDir = $subSrcAbs;
|
||||||
echo " Sub-package {$subName}: using src/ entry-point\n";
|
$subSrcName = SourceResolver::resolve($pkgDir);
|
||||||
|
echo " Sub-package {$subName}: using {$subSrcName}/ entry-point\n";
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_promote.php
|
* PATH: /cli/release_promote.php
|
||||||
* BRIEF: Promote a Gitea release from one channel to another (rename release, tag, assets)
|
* BRIEF: Promote a Gitea release from one channel to another (rename release, tag, assets)
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class ReleasePromoteCli extends CliFramework
|
class ReleasePromoteCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -109,8 +109,8 @@ class ReleasePromoteCli extends CliFramework
|
|||||||
if ($to === 'stable') {
|
if ($to === 'stable') {
|
||||||
$root = realpath($path) ?: $path;
|
$root = realpath($path) ?: $path;
|
||||||
$manifestFiles = array_merge(
|
$manifestFiles = array_merge(
|
||||||
glob("{$root}/src/pkg_*.xml") ?: [],
|
SourceResolver::globSource($root, 'pkg_*.xml'),
|
||||||
glob("{$root}/src/*.xml") ?: [],
|
SourceResolver::globSource($root, '*.xml'),
|
||||||
glob("{$root}/*.xml") ?: []
|
glob("{$root}/*.xml") ?: []
|
||||||
);
|
);
|
||||||
foreach ($manifestFiles as $xmlFile) {
|
foreach ($manifestFiles as $xmlFile) {
|
||||||
|
|||||||
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_publish.php
|
* PATH: /cli/release_publish.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Publish a release and create copies for all lesser stability streams.
|
* BRIEF: Publish a release and create copies for all lesser stability streams.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_validate.php
|
* PATH: /cli/release_validate.php
|
||||||
* BRIEF: Pre-release validation -- version consistency, required files, manifest checks
|
* BRIEF: Pre-release validation -- version consistency, required files, manifest checks
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class ReleaseValidateCli extends CliFramework
|
class ReleaseValidateCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -66,8 +66,10 @@ class ReleaseValidateCli extends CliFramework
|
|||||||
$platform = 'generic';
|
$platform = 'generic';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$hasSource = is_dir("{$root}/src") || is_dir("{$root}/htdocs");
|
$hasSource = SourceResolver::resolveAbsolute($root) !== null;
|
||||||
$this->addVResult('Source directory', $hasSource ? 'PASS' : 'WARN', $hasSource ? 'src/ or htdocs/ found' : 'No src/ or htdocs/ directory');
|
SourceResolver::warnIfLegacy($root);
|
||||||
|
$srcDirName = SourceResolver::resolve($root);
|
||||||
|
$this->addVResult('Source directory', $hasSource ? 'PASS' : 'WARN', $hasSource ? "{$srcDirName}/ found" : 'No source/ or src/ directory');
|
||||||
if (!file_exists("{$root}/README.md")) {
|
if (!file_exists("{$root}/README.md")) {
|
||||||
$this->addVResult('README.md', 'FAIL', 'Not found');
|
$this->addVResult('README.md', 'FAIL', 'Not found');
|
||||||
} else {
|
} else {
|
||||||
@@ -109,7 +111,8 @@ class ReleaseValidateCli extends CliFramework
|
|||||||
$this->addVResult('LICENSE', $licenseFound ? 'PASS' : 'FAIL', $licenseFound ? 'Found' : 'Not found');
|
$this->addVResult('LICENSE', $licenseFound ? 'PASS' : 'FAIL', $licenseFound ? 'Found' : 'Not found');
|
||||||
if ($platform === 'joomla') {
|
if ($platform === 'joomla') {
|
||||||
$manifest = null;
|
$manifest = null;
|
||||||
foreach (["{$root}/src", $root] as $dir) {
|
$srcAbs = SourceResolver::resolveAbsolute($root);
|
||||||
|
foreach (array_filter([$srcAbs, $root]) as $dir) {
|
||||||
if (!is_dir($dir)) {
|
if (!is_dir($dir)) {
|
||||||
continue;
|
continue;
|
||||||
} foreach (glob("{$dir}/*.xml") as $xmlFile) {
|
} foreach (glob("{$dir}/*.xml") as $xmlFile) {
|
||||||
@@ -156,7 +159,7 @@ class ReleaseValidateCli extends CliFramework
|
|||||||
}
|
}
|
||||||
} elseif ($platform === 'dolibarr') {
|
} elseif ($platform === 'dolibarr') {
|
||||||
$modFile = null;
|
$modFile = null;
|
||||||
foreach (['src', 'htdocs'] as $sd) {
|
foreach (SourceResolver::getCandidates() as $sd) {
|
||||||
$matches = glob("{$root}/{$sd}/mod*.class.php");
|
$matches = glob("{$root}/{$sd}/mod*.class.php");
|
||||||
if (!empty($matches)) {
|
if (!empty($matches)) {
|
||||||
$modFile = $matches[0];
|
$modFile = $matches[0];
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/release_verify.php
|
* PATH: /cli/release_verify.php
|
||||||
* BRIEF: Verify a built release artifact — version, SHA256, disallowed files
|
* BRIEF: Verify a built release artifact — version, SHA256, disallowed files
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/scaffold_client.php
|
* PATH: /cli/scaffold_client.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Scaffold a new client-waas repo from Template-Client-WaaS with pre-configured settings
|
* BRIEF: Scaffold a new client-waas repo from Template-Client-WaaS with pre-configured settings
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
@@ -8,9 +8,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/sync_rulesets.php
|
* PATH: /cli/sync_rulesets.php
|
||||||
* BRIEF: Apply branch protection rules to all repos via platform adapter
|
* BRIEF: Apply branch protection rules to all repos via platform adapter
|
||||||
*/
|
*/
|
||||||
@@ -46,7 +46,7 @@ class SyncRulesetsCli extends CliFramework
|
|||||||
);
|
);
|
||||||
|
|
||||||
$platformName = $adapter->getPlatformName();
|
$platformName = $adapter->getPlatformName();
|
||||||
$ALWAYS_EXCLUDE = ['moko-platform', '.github-private'];
|
$ALWAYS_EXCLUDE = ['MokoCLI', '.github-private'];
|
||||||
|
|
||||||
// -- Protection rules (platform-agnostic format) --
|
// -- Protection rules (platform-agnostic format) --
|
||||||
$PROTECTIONS = [
|
$PROTECTIONS = [
|
||||||
|
|||||||
+7
-12
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/theme_lint.php
|
* PATH: /cli/theme_lint.php
|
||||||
* BRIEF: Lint theme files -- CSS syntax, image sizes, hardcoded URLs
|
* BRIEF: Lint theme files -- CSS syntax, image sizes, hardcoded URLs
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class ThemeLintCli extends CliFramework
|
class ThemeLintCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -41,17 +41,12 @@ class ThemeLintCli extends CliFramework
|
|||||||
$errors = 0;
|
$errors = 0;
|
||||||
$warnings = 0;
|
$warnings = 0;
|
||||||
|
|
||||||
$srcDir = null;
|
$srcDir = SourceResolver::resolveAbsolute($root);
|
||||||
foreach (['src', 'htdocs'] as $d) {
|
|
||||||
if (is_dir("{$root}/{$d}")) {
|
|
||||||
$srcDir = "{$root}/{$d}";
|
|
||||||
break;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if ($srcDir === null) {
|
if ($srcDir === null) {
|
||||||
$this->log('ERROR', "No src/ or htdocs/ directory in {$root}");
|
$this->log('ERROR', "No source/ or src/ directory in {$root}");
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
|
SourceResolver::warnIfLegacy($root);
|
||||||
|
|
||||||
echo "Theme Lint: {$srcDir}\n\n";
|
echo "Theme Lint: {$srcDir}\n\n";
|
||||||
|
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/updates_xml_build.php
|
* PATH: /cli/updates_xml_build.php
|
||||||
* BRIEF: Generate Joomla updates.xml from extension manifest metadata
|
* BRIEF: Generate Joomla updates.xml from extension manifest metadata
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class UpdatesXmlBuildCli extends CliFramework
|
class UpdatesXmlBuildCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -109,7 +109,7 @@ class UpdatesXmlBuildCli extends CliFramework
|
|||||||
// -- Locate Joomla manifest ---------------------------------------------------
|
// -- Locate Joomla manifest ---------------------------------------------------
|
||||||
$manifest = null;
|
$manifest = null;
|
||||||
|
|
||||||
$candidates = glob("{$root}/src/pkg_*.xml") ?: [];
|
$candidates = SourceResolver::globSource($root, 'pkg_*.xml');
|
||||||
foreach ($candidates as $f) {
|
foreach ($candidates as $f) {
|
||||||
if (strpos(file_get_contents($f), '<extension') !== false) {
|
if (strpos(file_get_contents($f), '<extension') !== false) {
|
||||||
$manifest = $f;
|
$manifest = $f;
|
||||||
|
|||||||
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/updates_xml_sync.php
|
* PATH: /cli/updates_xml_sync.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Sync updates.xml to target branches via Gitea API
|
* BRIEF: Sync updates.xml to target branches via Gitea API
|
||||||
* NOTE: Called by pre-release and auto-release workflows after updates.xml
|
* NOTE: Called by pre-release and auto-release workflows after updates.xml
|
||||||
* is modified on the current branch. Pushes the file to other branches
|
* is modified on the current branch. Pushes the file to other branches
|
||||||
|
|||||||
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/version_auto_bump.php
|
* PATH: /cli/version_auto_bump.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Auto patch-bump, set stability suffix, and commit — single CLI replacing inline workflow bash
|
* BRIEF: Auto patch-bump, set stability suffix, and commit — single CLI replacing inline workflow bash
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
+110
-34
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/version_bump.php
|
* PATH: /cli/version_bump.php
|
||||||
* BRIEF: Auto-increment version -- manifest.xml is canonical, cascades to all XML and MD files
|
* BRIEF: Auto-increment version -- manifest.xml is canonical, cascades to all XML and MD files
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class VersionBumpCli extends CliFramework
|
class VersionBumpCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -42,6 +42,7 @@ class VersionBumpCli extends CliFramework
|
|||||||
$root = realpath($path) ?: $path;
|
$root = realpath($path) ?: $path;
|
||||||
$mokoVersion = null;
|
$mokoVersion = null;
|
||||||
$existingSuffix = '';
|
$existingSuffix = '';
|
||||||
|
$versionPrefix = '';
|
||||||
$mokoManifest = "{$root}/.mokogitea/manifest.xml";
|
$mokoManifest = "{$root}/.mokogitea/manifest.xml";
|
||||||
$mokoContent = '';
|
$mokoContent = '';
|
||||||
if (file_exists($mokoManifest)) {
|
if (file_exists($mokoManifest)) {
|
||||||
@@ -50,29 +51,58 @@ class VersionBumpCli extends CliFramework
|
|||||||
$mokoVersion = $m[1];
|
$mokoVersion = $m[1];
|
||||||
$existingSuffix = $m[2] ?? '';
|
$existingSuffix = $m[2] ?? '';
|
||||||
}
|
}
|
||||||
|
// Read version_prefix from manifest.xml (supports nested and flat structure)
|
||||||
|
$xml = @simplexml_load_file($mokoManifest);
|
||||||
|
if ($xml !== false) {
|
||||||
|
$prefix = (string)($xml->identity->version_prefix ?? '');
|
||||||
|
if ($prefix === '') {
|
||||||
|
$prefix = (string)($xml->version_prefix ?? '');
|
||||||
|
}
|
||||||
|
$versionPrefix = $prefix;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
$readmeVersion = null;
|
$readmeVersion = null;
|
||||||
$readme = "{$root}/README.md";
|
$readme = "{$root}/README.md";
|
||||||
$readmeContent = '';
|
$readmeContent = '';
|
||||||
if (file_exists($readme)) {
|
if (file_exists($readme)) {
|
||||||
$readmeContent = file_get_contents($readme);
|
$readmeContent = file_get_contents($readme);
|
||||||
if (preg_match('/VERSION:\s*(\d{2}\.\d{2}\.\d{2})/m', $readmeContent, $m)) {
|
if (!empty($versionPrefix)) {
|
||||||
|
// Prefix-aware README scan
|
||||||
|
$prefixPattern = preg_quote($versionPrefix, '/');
|
||||||
|
if (preg_match('/' . $prefixPattern . '(\d{2}\.\d{2}\.\d{2})/m', $readmeContent, $m)) {
|
||||||
|
$readmeVersion = $m[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($readmeVersion === null && preg_match('/VERSION:\s*(\d{2}\.\d{2}\.\d{2})/m', $readmeContent, $m)) {
|
||||||
$readmeVersion = $m[1];
|
$readmeVersion = $m[1];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$manifestVersion = null;
|
$manifestVersion = null;
|
||||||
|
SourceResolver::warnIfLegacy($root);
|
||||||
$manifestFiles = array_merge(
|
$manifestFiles = array_merge(
|
||||||
glob("{$root}/src/pkg_*.xml") ?: [],
|
SourceResolver::globSource($root, 'pkg_*.xml'),
|
||||||
glob("{$root}/src/*.xml") ?: [],
|
SourceResolver::globSource($root, '*.xml'),
|
||||||
glob("{$root}/src/packages/*/mokowaas.xml") ?: [],
|
SourceResolver::globSource($root, 'packages/*/mokowaas.xml'),
|
||||||
glob("{$root}/src/packages/*/*.xml") ?: [],
|
SourceResolver::globSource($root, 'packages/*/*.xml'),
|
||||||
glob("{$root}/*.xml") ?: []
|
glob("{$root}/*.xml") ?: []
|
||||||
);
|
);
|
||||||
foreach ($manifestFiles as $xmlFile) {
|
foreach ($manifestFiles as $xmlFile) {
|
||||||
$xmlContent = file_get_contents($xmlFile);
|
$xmlContent = file_get_contents($xmlFile);
|
||||||
if (strpos($xmlContent, '<extension') === false && strpos($xmlContent, '<version>') === false) {
|
if (strpos($xmlContent, '<extension') === false && strpos($xmlContent, '<version>') === false) {
|
||||||
continue;
|
continue;
|
||||||
} if (preg_match('#<version>(\d{2}\.\d{2}\.\d{2})((?:-(?:dev|alpha|beta|rc))+)?</version>#', $xmlContent, $xm)) {
|
}
|
||||||
|
if (!empty($versionPrefix)) {
|
||||||
|
// Prefix-aware: look for <version>prefix + XX.YY.ZZ</version>
|
||||||
|
$prefixPattern = preg_quote($versionPrefix, '#');
|
||||||
|
if (preg_match('#<version>' . $prefixPattern . '(\d{2}\.\d{2}\.\d{2})</version>#', $xmlContent, $xm)) {
|
||||||
|
$candidate = $xm[1];
|
||||||
|
if ($manifestVersion === null || version_compare($candidate, $manifestVersion, '>')) {
|
||||||
|
$manifestVersion = $candidate;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (preg_match('#<version>(\d{2}\.\d{2}\.\d{2})((?:-(?:dev|alpha|beta|rc))+)?</version>#', $xmlContent, $xm)) {
|
||||||
$candidate = $xm[1];
|
$candidate = $xm[1];
|
||||||
if ($manifestVersion === null || version_compare($candidate, $manifestVersion, '>')) {
|
if ($manifestVersion === null || version_compare($candidate, $manifestVersion, '>')) {
|
||||||
$manifestVersion = $candidate;
|
$manifestVersion = $candidate;
|
||||||
@@ -135,25 +165,43 @@ class VersionBumpCli extends CliFramework
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
if (file_exists($readme) && !empty($readmeContent)) {
|
if (file_exists($readme) && !empty($readmeContent)) {
|
||||||
$updated = preg_replace('/(VERSION:\s*)\d{2}\.\d{2}\.\d{2}(?:(?:-(?:dev|alpha|beta|rc))+)?/m', '${1}' . $newBase, $readmeContent, 1);
|
if (!empty($versionPrefix)) {
|
||||||
|
// Prefix-aware README replacement: preserve prefix, replace only version part
|
||||||
|
$prefixPattern = preg_quote($versionPrefix, '/');
|
||||||
|
$updated = preg_replace('/(' . $prefixPattern . ')\d{2}\.\d{2}\.\d{2}/m', '${1}' . $newBase, $readmeContent, 1);
|
||||||
|
} else {
|
||||||
|
$updated = preg_replace('/(VERSION:\s*)\d{2}\.\d{2}\.\d{2}(?:(?:-(?:dev|alpha|beta|rc))+)?/m', '${1}' . $newBase, $readmeContent, 1);
|
||||||
|
}
|
||||||
if ($updated !== null) {
|
if ($updated !== null) {
|
||||||
file_put_contents($readme, $updated);
|
file_put_contents($readme, $updated);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
$updatedFiles = [];
|
$updatedFiles = [];
|
||||||
foreach (["{$root}/src/pkg_*.xml", "{$root}/src/*.xml", "{$root}/src/packages/*/*.xml", "{$root}/*.xml"] as $pattern) {
|
$srcName = SourceResolver::resolve($root);
|
||||||
|
foreach (["{$root}/{$srcName}/pkg_*.xml", "{$root}/{$srcName}/*.xml", "{$root}/{$srcName}/packages/*/*.xml", "{$root}/*.xml"] as $pattern) {
|
||||||
foreach (glob($pattern) ?: [] as $xmlFile) {
|
foreach (glob($pattern) ?: [] as $xmlFile) {
|
||||||
$content = file_get_contents($xmlFile);
|
$content = file_get_contents($xmlFile);
|
||||||
if (strpos($content, '<extension') === false) {
|
if (strpos($content, '<extension') === false) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
$xmlPattern = '#<version>\d{2}\.\d{2}\.\d{2}'
|
if (!empty($versionPrefix)) {
|
||||||
. '(?:(?:-(?:dev|alpha|beta|rc))+)?</version>#';
|
// Prefix-aware: preserve prefix, replace only the Moko version part
|
||||||
$newContent = preg_replace(
|
$prefixPattern = preg_quote($versionPrefix, '#');
|
||||||
$xmlPattern,
|
$xmlPattern = '#(<version>' . $prefixPattern . ')\d{2}\.\d{2}\.\d{2}</version>#';
|
||||||
"<version>{$newFull}</version>",
|
$newContent = preg_replace(
|
||||||
$content
|
$xmlPattern,
|
||||||
);
|
'${1}' . $newBase . '</version>',
|
||||||
|
$content
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
$xmlPattern = '#<version>\d{2}\.\d{2}\.\d{2}'
|
||||||
|
. '(?:(?:-(?:dev|alpha|beta|rc))+)?</version>#';
|
||||||
|
$newContent = preg_replace(
|
||||||
|
$xmlPattern,
|
||||||
|
"<version>{$newFull}</version>",
|
||||||
|
$content
|
||||||
|
);
|
||||||
|
}
|
||||||
if ($newContent !== null && $newContent !== $content) {
|
if ($newContent !== null && $newContent !== $content) {
|
||||||
file_put_contents($xmlFile, $newContent);
|
file_put_contents($xmlFile, $newContent);
|
||||||
$updatedFiles[] = substr($xmlFile, strlen($root) + 1);
|
$updatedFiles[] = substr($xmlFile, strlen($root) + 1);
|
||||||
@@ -166,13 +214,24 @@ class VersionBumpCli extends CliFramework
|
|||||||
$packageJsonFile = "{$root}/package.json";
|
$packageJsonFile = "{$root}/package.json";
|
||||||
if (file_exists($packageJsonFile)) {
|
if (file_exists($packageJsonFile)) {
|
||||||
$pkgContent = file_get_contents($packageJsonFile);
|
$pkgContent = file_get_contents($packageJsonFile);
|
||||||
$pkgPattern = '/("version"\s*:\s*")\d{2}\.\d{2}\.\d{2}'
|
if (!empty($versionPrefix)) {
|
||||||
. '(?:(?:-(?:dev|alpha|beta|rc))+)?(")/m';
|
// Prefix-aware package.json replacement
|
||||||
$updatedPkg = preg_replace(
|
$prefixPattern = preg_quote($versionPrefix, '/');
|
||||||
$pkgPattern,
|
$pkgPattern = '/("version"\s*:\s*")' . $prefixPattern . '\d{2}\.\d{2}\.\d{2}(")/m';
|
||||||
'${1}' . $newFull . '${2}',
|
$updatedPkg = preg_replace(
|
||||||
$pkgContent
|
$pkgPattern,
|
||||||
);
|
'${1}' . $versionPrefix . $newBase . '${2}',
|
||||||
|
$pkgContent
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
$pkgPattern = '/("version"\s*:\s*")\d{2}\.\d{2}\.\d{2}'
|
||||||
|
. '(?:(?:-(?:dev|alpha|beta|rc))+)?(")/m';
|
||||||
|
$updatedPkg = preg_replace(
|
||||||
|
$pkgPattern,
|
||||||
|
'${1}' . $newFull . '${2}',
|
||||||
|
$pkgContent
|
||||||
|
);
|
||||||
|
}
|
||||||
if ($updatedPkg !== $pkgContent) {
|
if ($updatedPkg !== $pkgContent) {
|
||||||
file_put_contents($packageJsonFile, $updatedPkg);
|
file_put_contents($packageJsonFile, $updatedPkg);
|
||||||
fwrite(STDERR, "Updated package.json\n");
|
fwrite(STDERR, "Updated package.json\n");
|
||||||
@@ -181,13 +240,24 @@ class VersionBumpCli extends CliFramework
|
|||||||
$pyprojectFile = "{$root}/pyproject.toml";
|
$pyprojectFile = "{$root}/pyproject.toml";
|
||||||
if (file_exists($pyprojectFile)) {
|
if (file_exists($pyprojectFile)) {
|
||||||
$pyContent = file_get_contents($pyprojectFile);
|
$pyContent = file_get_contents($pyprojectFile);
|
||||||
$pyPattern = '/^(version\s*=\s*")\d{2}\.\d{2}\.\d{2}'
|
if (!empty($versionPrefix)) {
|
||||||
. '(?:(?:-(?:dev|alpha|beta|rc))+)?(")/m';
|
// Prefix-aware pyproject.toml replacement
|
||||||
$updatedPy = preg_replace(
|
$prefixPattern = preg_quote($versionPrefix, '/');
|
||||||
$pyPattern,
|
$pyPattern = '/^(version\s*=\s*")' . $prefixPattern . '\d{2}\.\d{2}\.\d{2}(")/m';
|
||||||
'${1}' . $newFull . '${2}',
|
$updatedPy = preg_replace(
|
||||||
$pyContent
|
$pyPattern,
|
||||||
);
|
'${1}' . $versionPrefix . $newBase . '${2}',
|
||||||
|
$pyContent
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
$pyPattern = '/^(version\s*=\s*")\d{2}\.\d{2}\.\d{2}'
|
||||||
|
. '(?:(?:-(?:dev|alpha|beta|rc))+)?(")/m';
|
||||||
|
$updatedPy = preg_replace(
|
||||||
|
$pyPattern,
|
||||||
|
'${1}' . $newFull . '${2}',
|
||||||
|
$pyContent
|
||||||
|
);
|
||||||
|
}
|
||||||
if ($updatedPy !== $pyContent) {
|
if ($updatedPy !== $pyContent) {
|
||||||
file_put_contents($pyprojectFile, $updatedPy);
|
file_put_contents($pyprojectFile, $updatedPy);
|
||||||
fwrite(STDERR, "Updated pyproject.toml\n");
|
fwrite(STDERR, "Updated pyproject.toml\n");
|
||||||
@@ -204,7 +274,13 @@ class VersionBumpCli extends CliFramework
|
|||||||
}
|
}
|
||||||
$scanExtensions = ['php', 'yml', 'yaml', 'md', 'txt', 'xml', 'sh', 'toml', 'ini', 'css', 'js'];
|
$scanExtensions = ['php', 'yml', 'yaml', 'md', 'txt', 'xml', 'sh', 'toml', 'ini', 'css', 'js'];
|
||||||
$excludeDirs = ['.git', 'vendor', 'node_modules', 'build', 'dist', '.claude'];
|
$excludeDirs = ['.git', 'vendor', 'node_modules', 'build', 'dist', '.claude'];
|
||||||
$versionPattern = '/(VERSION:\s*)\d{2}\.\d{2}\.\d{2}/m';
|
// Build the generic VERSION: pattern — prefix-aware if configured
|
||||||
|
if (!empty($versionPrefix)) {
|
||||||
|
$prefixPatternGeneric = preg_quote($versionPrefix, '/');
|
||||||
|
$versionPattern = '/(' . $prefixPatternGeneric . ')\d{2}\.\d{2}\.\d{2}/m';
|
||||||
|
} else {
|
||||||
|
$versionPattern = '/(VERSION:\s*)\d{2}\.\d{2}\.\d{2}/m';
|
||||||
|
}
|
||||||
$directory = new RecursiveDirectoryIterator($root, RecursiveDirectoryIterator::SKIP_DOTS);
|
$directory = new RecursiveDirectoryIterator($root, RecursiveDirectoryIterator::SKIP_DOTS);
|
||||||
$filter = new RecursiveCallbackFilterIterator($directory, function ($current, $key, $iterator) use ($excludeDirs) {
|
$filter = new RecursiveCallbackFilterIterator($directory, function ($current, $key, $iterator) use ($excludeDirs) {
|
||||||
if ($current->isDir() && in_array($current->getFilename(), $excludeDirs, true)) {
|
if ($current->isDir() && in_array($current->getFilename(), $excludeDirs, true)) {
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/version_bump_remote.php
|
* PATH: /cli/version_bump_remote.php
|
||||||
* BRIEF: Bump version in manifest XML and CHANGELOG.md on a remote branch via Gitea API
|
* BRIEF: Bump version in manifest XML and CHANGELOG.md on a remote branch via Gitea API
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class VersionBumpRemoteCli extends CliFramework
|
class VersionBumpRemoteCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -104,11 +104,15 @@ class VersionBumpRemoteCli extends CliFramework
|
|||||||
$nextVersion = sprintf('%02d.%02d.%02d', $major, $minor, $patch);
|
$nextVersion = sprintf('%02d.%02d.%02d', $major, $minor, $patch);
|
||||||
echo "{$version} -> {$nextVersion} ({$branch})\n";
|
echo "{$version} -> {$nextVersion} ({$branch})\n";
|
||||||
|
|
||||||
|
// Try both source/ and src/ paths for backwards compatibility with remote repos
|
||||||
$manifestPaths = [];
|
$manifestPaths = [];
|
||||||
if ($manifestFile !== null) {
|
foreach (['source', 'src'] as $srcPrefix) {
|
||||||
$manifestPaths[] = "src/{$manifestFile}";
|
if ($manifestFile !== null) {
|
||||||
|
$manifestPaths[] = "{$srcPrefix}/{$manifestFile}";
|
||||||
|
}
|
||||||
|
$manifestPaths[] = "{$srcPrefix}/templateDetails.xml";
|
||||||
|
$manifestPaths[] = "{$srcPrefix}/manifest.xml";
|
||||||
}
|
}
|
||||||
$manifestPaths = array_merge($manifestPaths, ['src/templateDetails.xml', 'src/manifest.xml']);
|
|
||||||
$manifestUpdated = false;
|
$manifestUpdated = false;
|
||||||
foreach ($manifestPaths as $mPath) {
|
foreach ($manifestPaths as $mPath) {
|
||||||
$result = $this->updateRemoteFile($apiBase, $token, $mPath, $branch, function (string $content) use ($version, $nextVersion): string {
|
$result = $this->updateRemoteFile($apiBase, $token, $mPath, $branch, function (string $content) use ($version, $nextVersion): string {
|
||||||
|
|||||||
@@ -6,11 +6,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/version_check.php
|
* PATH: /cli/version_check.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Validate version consistency across README, manifests, and sub-packages
|
* BRIEF: Validate version consistency across README, manifests, and sub-packages
|
||||||
*/
|
*/
|
||||||
|
|
||||||
@@ -18,7 +18,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class VersionCheckCli extends CliFramework
|
class VersionCheckCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -77,7 +77,8 @@ class VersionCheckCli extends CliFramework
|
|||||||
$versions['pyproject.toml'] = $m[1];
|
$versions['pyproject.toml'] = $m[1];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
foreach (["{$root}/src/pkg_*.xml", "{$root}/src/*.xml", "{$root}/src/packages/*/*.xml", "{$root}/*.xml"] as $glob) {
|
$srcName = SourceResolver::resolve($root);
|
||||||
|
foreach (["{$root}/{$srcName}/pkg_*.xml", "{$root}/{$srcName}/*.xml", "{$root}/{$srcName}/packages/*/*.xml", "{$root}/*.xml"] as $glob) {
|
||||||
foreach (glob($glob) ?: [] as $file) {
|
foreach (glob($glob) ?: [] as $file) {
|
||||||
if (basename($file) === 'updates.xml') {
|
if (basename($file) === 'updates.xml') {
|
||||||
continue;
|
continue;
|
||||||
|
|||||||
+36
-10
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/version_read.php
|
* PATH: /cli/version_read.php
|
||||||
* BRIEF: Read version — manifest.xml is canonical, falls back to README.md and Joomla XML
|
* BRIEF: Read version — manifest.xml is canonical, falls back to README.md and Joomla XML
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class VersionReadCli extends CliFramework
|
class VersionReadCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -34,6 +34,7 @@ class VersionReadCli extends CliFramework
|
|||||||
|
|
||||||
// -- 1. Read from .mokogitea/manifest.xml (canonical source) --
|
// -- 1. Read from .mokogitea/manifest.xml (canonical source) --
|
||||||
$mokoVersion = null;
|
$mokoVersion = null;
|
||||||
|
$versionPrefix = '';
|
||||||
$mokoManifest = "{$root}/.mokogitea/manifest.xml";
|
$mokoManifest = "{$root}/.mokogitea/manifest.xml";
|
||||||
if (file_exists($mokoManifest)) {
|
if (file_exists($mokoManifest)) {
|
||||||
$xml = @simplexml_load_file($mokoManifest);
|
$xml = @simplexml_load_file($mokoManifest);
|
||||||
@@ -42,6 +43,12 @@ class VersionReadCli extends CliFramework
|
|||||||
if (preg_match('/^\d{2}\.\d{2}\.\d{2}((?:-(?:dev|alpha|beta|rc))+)?$/', $v)) {
|
if (preg_match('/^\d{2}\.\d{2}\.\d{2}((?:-(?:dev|alpha|beta|rc))+)?$/', $v)) {
|
||||||
$mokoVersion = $v;
|
$mokoVersion = $v;
|
||||||
}
|
}
|
||||||
|
// Read version_prefix (supports both nested and flat structure)
|
||||||
|
$prefix = (string)($xml->identity->version_prefix ?? '');
|
||||||
|
if ($prefix === '') {
|
||||||
|
$prefix = (string)($xml->version_prefix ?? '');
|
||||||
|
}
|
||||||
|
$versionPrefix = $prefix;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -56,7 +63,14 @@ class VersionReadCli extends CliFramework
|
|||||||
$readme = "{$root}/README.md";
|
$readme = "{$root}/README.md";
|
||||||
if (file_exists($readme)) {
|
if (file_exists($readme)) {
|
||||||
$content = file_get_contents($readme);
|
$content = file_get_contents($readme);
|
||||||
if (preg_match('/VERSION:\s*(\d{2}\.\d{2}\.\d{2})/m', $content, $m)) {
|
if (!empty($versionPrefix)) {
|
||||||
|
// Prefix-aware: search for prefix followed by version
|
||||||
|
$prefixPattern = preg_quote($versionPrefix, '/');
|
||||||
|
if (preg_match('/' . $prefixPattern . '(\d{2}\.\d{2}\.\d{2})/m', $content, $m)) {
|
||||||
|
$readmeVersion = $m[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if ($readmeVersion === null && preg_match('/VERSION:\s*(\d{2}\.\d{2}\.\d{2})/m', $content, $m)) {
|
||||||
$readmeVersion = $m[1];
|
$readmeVersion = $m[1];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -64,9 +78,9 @@ class VersionReadCli extends CliFramework
|
|||||||
// -- 3. Fallback: Joomla manifest XML --
|
// -- 3. Fallback: Joomla manifest XML --
|
||||||
$manifestVersion = null;
|
$manifestVersion = null;
|
||||||
$manifestFiles = array_merge(
|
$manifestFiles = array_merge(
|
||||||
glob("{$root}/src/pkg_*.xml") ?: [],
|
SourceResolver::globSource($root, 'pkg_*.xml'),
|
||||||
glob("{$root}/src/*.xml") ?: [],
|
SourceResolver::globSource($root, '*.xml'),
|
||||||
glob("{$root}/src/packages/*/*.xml") ?: [],
|
SourceResolver::globSource($root, 'packages/*/*.xml'),
|
||||||
glob("{$root}/*.xml") ?: []
|
glob("{$root}/*.xml") ?: []
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -75,10 +89,22 @@ class VersionReadCli extends CliFramework
|
|||||||
if (strpos($xmlContent, '<extension') === false && strpos($xmlContent, '<version>') === false) {
|
if (strpos($xmlContent, '<extension') === false && strpos($xmlContent, '<version>') === false) {
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
if (!empty($versionPrefix)) {
|
||||||
|
// Prefix-aware: look for <version>prefix + XX.YY.ZZ</version>
|
||||||
|
$prefixPattern = preg_quote($versionPrefix, '#');
|
||||||
|
if (preg_match('#<version>' . $prefixPattern . '(\d{2}\.\d{2}\.\d{2})</version>#', $xmlContent, $xm)) {
|
||||||
|
$candidate = $xm[1];
|
||||||
|
$currentBase = $manifestVersion ? preg_replace('/(-(?:dev|alpha|beta|rc))+$/', '', $manifestVersion) : null;
|
||||||
|
if ($currentBase === null || version_compare($candidate, $currentBase, '>')) {
|
||||||
|
$manifestVersion = $candidate;
|
||||||
|
}
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
}
|
||||||
if (preg_match('#<version>(\d{2}\.\d{2}\.\d{2}(?:(?:-(?:dev|alpha|beta|rc))+)?)</version>#', $xmlContent, $xm)) {
|
if (preg_match('#<version>(\d{2}\.\d{2}\.\d{2}(?:(?:-(?:dev|alpha|beta|rc))+)?)</version>#', $xmlContent, $xm)) {
|
||||||
$candidate = $xm[1];
|
$candidate = $xm[1];
|
||||||
$candidateBase = preg_replace('/(-(dev|alpha|beta|rc))+$/', '', $candidate);
|
$candidateBase = preg_replace('/(-(?:dev|alpha|beta|rc))+$/', '', $candidate);
|
||||||
$currentBase = $manifestVersion ? preg_replace('/(-(dev|alpha|beta|rc))+$/', '', $manifestVersion) : null;
|
$currentBase = $manifestVersion ? preg_replace('/(-(?:dev|alpha|beta|rc))+$/', '', $manifestVersion) : null;
|
||||||
if ($currentBase === null || version_compare($candidateBase, $currentBase, '>')) {
|
if ($currentBase === null || version_compare($candidateBase, $currentBase, '>')) {
|
||||||
$manifestVersion = $candidate;
|
$manifestVersion = $candidate;
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/version_reset_dev.php
|
* PATH: /cli/version_reset_dev.php
|
||||||
* BRIEF: Reset platform version to 'development' on a branch via Gitea API
|
* BRIEF: Reset platform version to 'development' on a branch via Gitea API
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -6,9 +6,9 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/version_set_platform.php
|
* PATH: /cli/version_set_platform.php
|
||||||
* BRIEF: Set version in platform-specific files (Dolibarr $this->version, Joomla <version>)
|
* BRIEF: Set version in platform-specific files (Dolibarr $this->version, Joomla <version>)
|
||||||
*/
|
*/
|
||||||
@@ -17,7 +17,7 @@ declare(strict_types=1);
|
|||||||
|
|
||||||
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
use MokoEnterprise\CliFramework;
|
use MokoEnterprise\{CliFramework, SourceResolver};
|
||||||
|
|
||||||
class VersionSetPlatformCli extends CliFramework
|
class VersionSetPlatformCli extends CliFramework
|
||||||
{
|
{
|
||||||
@@ -110,7 +110,8 @@ class VersionSetPlatformCli extends CliFramework
|
|||||||
|
|
||||||
// Dolibarr: $this->version + $this->url_last_version in mod*.class.php
|
// Dolibarr: $this->version + $this->url_last_version in mod*.class.php
|
||||||
if ($platform === 'crm-module') {
|
if ($platform === 'crm-module') {
|
||||||
$pattern = "{$root}/src/core/modules/mod*.class.php";
|
$srcName = SourceResolver::resolve($root);
|
||||||
|
$pattern = "{$root}/{$srcName}/core/modules/mod*.class.php";
|
||||||
foreach (glob($pattern) ?: [] as $file) {
|
foreach (glob($pattern) ?: [] as $file) {
|
||||||
$content = file_get_contents($file);
|
$content = file_get_contents($file);
|
||||||
|
|
||||||
@@ -146,9 +147,10 @@ class VersionSetPlatformCli extends CliFramework
|
|||||||
|
|
||||||
// Joomla: <version> in XML manifests (top-level + sub-packages)
|
// Joomla: <version> in XML manifests (top-level + sub-packages)
|
||||||
if (in_array($platform, ['waas-component', 'joomla'], true)) {
|
if (in_array($platform, ['waas-component', 'joomla'], true)) {
|
||||||
|
$srcName = SourceResolver::resolve($root);
|
||||||
$xmlFiles = array_merge(
|
$xmlFiles = array_merge(
|
||||||
glob("{$root}/src/*.xml") ?: [],
|
glob("{$root}/{$srcName}/*.xml") ?: [],
|
||||||
glob("{$root}/src/packages/*/*.xml") ?: [],
|
glob("{$root}/{$srcName}/packages/*/*.xml") ?: [],
|
||||||
glob("{$root}/*.xml") ?: []
|
glob("{$root}/*.xml") ?: []
|
||||||
);
|
);
|
||||||
if (empty($xmlFiles)) {
|
if (empty($xmlFiles)) {
|
||||||
|
|||||||
+8
-8
@@ -6,12 +6,12 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: moko-platform.CLI
|
* DEFGROUP: MokoCLI.CLI
|
||||||
* INGROUP: moko-platform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /cli/wiki_sync.php
|
* PATH: /cli/wiki_sync.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Sync select wiki pages from moko-platform to all template repos
|
* BRIEF: Sync select wiki pages from MokoCLI to all template repos
|
||||||
*/
|
*/
|
||||||
|
|
||||||
declare(strict_types=1);
|
declare(strict_types=1);
|
||||||
@@ -25,7 +25,7 @@ class WikiSyncCli extends CliFramework
|
|||||||
private string $giteaUrl = 'https://git.mokoconsulting.tech';
|
private string $giteaUrl = 'https://git.mokoconsulting.tech';
|
||||||
private string $token = '';
|
private string $token = '';
|
||||||
private string $org = 'MokoConsulting';
|
private string $org = 'MokoConsulting';
|
||||||
private string $sourceRepo = 'moko-platform';
|
private string $sourceRepo = 'MokoCLI';
|
||||||
private array $targetRepos = [];
|
private array $targetRepos = [];
|
||||||
private array $pages = [];
|
private array $pages = [];
|
||||||
private bool $allTemplates = false;
|
private bool $allTemplates = false;
|
||||||
@@ -38,10 +38,10 @@ class WikiSyncCli extends CliFramework
|
|||||||
|
|
||||||
protected function configure(): void
|
protected function configure(): void
|
||||||
{
|
{
|
||||||
$this->setDescription('Sync wiki pages from moko-platform to template repos');
|
$this->setDescription('Sync wiki pages from MokoCLI to template repos');
|
||||||
$this->addArgument('--token', 'Gitea API token (required)', '');
|
$this->addArgument('--token', 'Gitea API token (required)', '');
|
||||||
$this->addArgument('--org', 'Organization (default: MokoConsulting)', 'MokoConsulting');
|
$this->addArgument('--org', 'Organization (default: MokoConsulting)', 'MokoConsulting');
|
||||||
$this->addArgument('--source', 'Source repo (default: moko-platform)', 'moko-platform');
|
$this->addArgument('--source', 'Source repo (default: MokoCLI)', 'MokoCLI');
|
||||||
$this->addArgument('--target', 'Target repo (can repeat)', '');
|
$this->addArgument('--target', 'Target repo (can repeat)', '');
|
||||||
$this->addArgument('--page', 'Page to sync (can repeat)', '');
|
$this->addArgument('--page', 'Page to sync (can repeat)', '');
|
||||||
$this->addArgument('--all-standards', 'Sync all UPPERCASE standards pages', false);
|
$this->addArgument('--all-standards', 'Sync all UPPERCASE standards pages', false);
|
||||||
|
|||||||
@@ -0,0 +1,646 @@
|
|||||||
|
#!/usr/bin/env php
|
||||||
|
<?php
|
||||||
|
|
||||||
|
/* Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
|
*
|
||||||
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
*
|
||||||
|
* FILE INFORMATION
|
||||||
|
* DEFGROUP: MokoCLI.CLI
|
||||||
|
* INGROUP: MokoCLI
|
||||||
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
|
* PATH: /cli/workflow_sync.php
|
||||||
|
* VERSION: 09.25.05
|
||||||
|
* BRIEF: Sync workflows from Generic → platform templates → live repos based on manifest.platform
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare(strict_types=1);
|
||||||
|
|
||||||
|
require_once __DIR__ . '/../lib/Enterprise/CliFramework.php';
|
||||||
|
|
||||||
|
use MokoEnterprise\CliFramework;
|
||||||
|
|
||||||
|
class WorkflowSyncCli extends CliFramework
|
||||||
|
{
|
||||||
|
private const PLATFORM_TEMPLATES = [
|
||||||
|
'joomla' => 'Template-Joomla',
|
||||||
|
'dolibarr' => 'Template-Dolibarr',
|
||||||
|
'go' => 'Template-Go',
|
||||||
|
'mcp' => 'Template-MCP',
|
||||||
|
'platform' => 'Template-Generic',
|
||||||
|
'generic' => 'Template-Generic',
|
||||||
|
];
|
||||||
|
|
||||||
|
private const DEFAULT_TEMPLATE = 'Template-Generic';
|
||||||
|
private const GENERIC_TEMPLATE = 'Template-Generic';
|
||||||
|
|
||||||
|
private int $updated = 0;
|
||||||
|
private int $created = 0;
|
||||||
|
private int $skipped = 0;
|
||||||
|
private int $errors = 0;
|
||||||
|
|
||||||
|
protected function configure(): void
|
||||||
|
{
|
||||||
|
$this->setDescription('Sync workflows from Generic → platform templates → live repos based on manifest.platform');
|
||||||
|
$this->addArgument('--gitea-url', 'Gitea URL (default: https://git.mokoconsulting.tech)', 'https://git.mokoconsulting.tech');
|
||||||
|
$this->addArgument('--token', 'Gitea API token', '');
|
||||||
|
$this->addArgument('--org', 'Target organization', '');
|
||||||
|
$this->addArgument('--branch', 'Target branch (default: main)', 'main');
|
||||||
|
$this->addArgument('--phase', 'Phase to run: all, templates, repos (default: all)', 'all');
|
||||||
|
$this->addArgument('--platform-filter', 'Only sync repos matching this platform', '');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function run(): int
|
||||||
|
{
|
||||||
|
$giteaUrl = rtrim($this->getArgument('--gitea-url'), '/');
|
||||||
|
$token = $this->getArgument('--token');
|
||||||
|
$org = $this->getArgument('--org');
|
||||||
|
$branch = $this->getArgument('--branch');
|
||||||
|
$phase = $this->getArgument('--phase');
|
||||||
|
$platformFilter = $this->getArgument('--platform-filter');
|
||||||
|
|
||||||
|
if ($token === '') {
|
||||||
|
$this->log('ERROR', '--token is required.');
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($org === '') {
|
||||||
|
$this->log('ERROR', '--org is required.');
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!in_array($phase, ['all', 'templates', 'repos'], true)) {
|
||||||
|
$this->log('ERROR', "--phase must be one of: all, templates, repos (got: {$phase})");
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->log('INFO', "Workflow Sync — org: {$org}, branch: {$branch}, phase: {$phase}");
|
||||||
|
|
||||||
|
if ($platformFilter !== '') {
|
||||||
|
$this->log('INFO', "Platform filter: {$platformFilter}");
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->dryRun) {
|
||||||
|
$this->log('INFO', '[DRY RUN] No changes will be made.');
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "\n";
|
||||||
|
|
||||||
|
// Phase 1: Sync Generic → Platform Templates
|
||||||
|
if ($phase === 'all' || $phase === 'templates') {
|
||||||
|
$result = $this->syncGenericToTemplates($giteaUrl, $token, $org, $branch, $platformFilter);
|
||||||
|
|
||||||
|
if ($result !== 0) {
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Phase 2: Sync Platform Templates → Live Repos
|
||||||
|
if ($phase === 'all' || $phase === 'repos') {
|
||||||
|
$result = $this->syncTemplatesToRepos($giteaUrl, $token, $org, $branch, $platformFilter);
|
||||||
|
|
||||||
|
if ($result !== 0) {
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "\n";
|
||||||
|
$this->log('INFO', "Done: {$this->created} created, {$this->updated} updated, "
|
||||||
|
. "{$this->skipped} skipped, {$this->errors} error(s).");
|
||||||
|
|
||||||
|
return $this->errors > 0 ? 1 : 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Phase 1: Push all Generic workflows to each platform template repo.
|
||||||
|
* Skips platform-specific overrides (files that exist in the platform template but NOT in Generic).
|
||||||
|
*/
|
||||||
|
private function syncGenericToTemplates(
|
||||||
|
string $giteaUrl,
|
||||||
|
string $token,
|
||||||
|
string $org,
|
||||||
|
string $branch,
|
||||||
|
string $platformFilter
|
||||||
|
): int {
|
||||||
|
$this->log('INFO', '=== Phase 1: Sync Generic → Platform Templates ===');
|
||||||
|
echo "\n";
|
||||||
|
|
||||||
|
// Get all workflow files from Template-Generic
|
||||||
|
$genericWorkflows = $this->listWorkflows($giteaUrl, $token, $org, self::GENERIC_TEMPLATE, $branch);
|
||||||
|
|
||||||
|
if ($genericWorkflows === null) {
|
||||||
|
$this->log('ERROR', 'Could not list workflows from ' . self::GENERIC_TEMPLATE);
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (count($genericWorkflows) === 0) {
|
||||||
|
$this->log('WARN', 'No workflows found in ' . self::GENERIC_TEMPLATE);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->log('INFO', 'Found ' . count($genericWorkflows) . ' workflow(s) in ' . self::GENERIC_TEMPLATE);
|
||||||
|
echo "\n";
|
||||||
|
|
||||||
|
// Get unique platform templates (exclude Generic itself)
|
||||||
|
$platformTemplates = array_unique(array_filter(
|
||||||
|
array_values(self::PLATFORM_TEMPLATES),
|
||||||
|
fn(string $t) => $t !== self::GENERIC_TEMPLATE
|
||||||
|
));
|
||||||
|
|
||||||
|
// If platform-filter is set, only sync to the matching template
|
||||||
|
if ($platformFilter !== '') {
|
||||||
|
$targetTemplate = self::PLATFORM_TEMPLATES[$platformFilter] ?? null;
|
||||||
|
|
||||||
|
if ($targetTemplate === null || $targetTemplate === self::GENERIC_TEMPLATE) {
|
||||||
|
$this->log('INFO', "Platform filter '{$platformFilter}' does not map to a non-generic template, skipping Phase 1.");
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
$platformTemplates = [$targetTemplate];
|
||||||
|
}
|
||||||
|
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", 'Template / File', 'Status');
|
||||||
|
fprintf(STDERR, "%s\n", str_repeat('-', 70));
|
||||||
|
|
||||||
|
foreach ($platformTemplates as $templateRepo) {
|
||||||
|
foreach ($genericWorkflows as $workflow) {
|
||||||
|
$filename = $workflow['name'];
|
||||||
|
$destPath = '.mokogitea/workflows/' . $filename;
|
||||||
|
$label = "{$templateRepo}/{$filename}";
|
||||||
|
|
||||||
|
// Get file content from Generic
|
||||||
|
$sourceContent = $this->getFileContent(
|
||||||
|
$giteaUrl, $token, $org,
|
||||||
|
self::GENERIC_TEMPLATE, $destPath, $branch
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($sourceContent === null) {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, 'ERROR (read source)');
|
||||||
|
$this->errors++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$commitMsg = "chore: sync {$filename} from " . self::GENERIC_TEMPLATE . " [skip ci]";
|
||||||
|
|
||||||
|
$this->pushFile(
|
||||||
|
$giteaUrl, $token, $org, $templateRepo,
|
||||||
|
$destPath, $sourceContent, $branch, $commitMsg, $label
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "\n";
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Phase 2: Sync platform template workflows to live repos based on manifest.platform.
|
||||||
|
*/
|
||||||
|
private function syncTemplatesToRepos(
|
||||||
|
string $giteaUrl,
|
||||||
|
string $token,
|
||||||
|
string $org,
|
||||||
|
string $branch,
|
||||||
|
string $platformFilter
|
||||||
|
): int {
|
||||||
|
$this->log('INFO', '=== Phase 2: Sync Platform Templates → Live Repos ===');
|
||||||
|
echo "\n";
|
||||||
|
|
||||||
|
$repos = $this->fetchOrgRepos($giteaUrl, $token, $org);
|
||||||
|
|
||||||
|
if ($repos === null) {
|
||||||
|
return 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
$this->log('INFO', 'Found ' . count($repos) . " repo(s) in \"{$org}\".");
|
||||||
|
echo "\n";
|
||||||
|
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", 'Repo / File', 'Status');
|
||||||
|
fprintf(STDERR, "%s\n", str_repeat('-', 70));
|
||||||
|
|
||||||
|
// Cache template workflows to avoid repeated API calls
|
||||||
|
$templateWorkflowCache = [];
|
||||||
|
|
||||||
|
foreach ($repos as $repoFullName) {
|
||||||
|
[, $repoName] = explode('/', $repoFullName, 2);
|
||||||
|
|
||||||
|
// Skip template repos
|
||||||
|
if (str_starts_with($repoName, 'Template-')) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Read manifest.platform
|
||||||
|
$platform = $this->getRepoPlatform($giteaUrl, $token, $org, $repoName, $branch);
|
||||||
|
|
||||||
|
// Apply platform filter
|
||||||
|
if ($platformFilter !== '' && $platform !== $platformFilter) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Resolve template
|
||||||
|
$templateRepo = self::PLATFORM_TEMPLATES[$platform] ?? self::DEFAULT_TEMPLATE;
|
||||||
|
|
||||||
|
// Get workflows from the template (cached)
|
||||||
|
if (!isset($templateWorkflowCache[$templateRepo])) {
|
||||||
|
$workflows = $this->listWorkflows($giteaUrl, $token, $org, $templateRepo, $branch);
|
||||||
|
|
||||||
|
if ($workflows === null) {
|
||||||
|
$this->log('WARN', "Could not list workflows from {$templateRepo}, falling back to " . self::GENERIC_TEMPLATE);
|
||||||
|
$workflows = $this->listWorkflows($giteaUrl, $token, $org, self::GENERIC_TEMPLATE, $branch);
|
||||||
|
}
|
||||||
|
|
||||||
|
$templateWorkflowCache[$templateRepo] = $workflows ?? [];
|
||||||
|
}
|
||||||
|
|
||||||
|
$workflows = $templateWorkflowCache[$templateRepo];
|
||||||
|
|
||||||
|
if (count($workflows) === 0) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($workflows as $workflow) {
|
||||||
|
$filename = $workflow['name'];
|
||||||
|
$destPath = '.mokogitea/workflows/' . $filename;
|
||||||
|
$label = "{$repoFullName}/{$filename}";
|
||||||
|
|
||||||
|
// Get source content from template
|
||||||
|
$sourceContent = $this->getFileContent(
|
||||||
|
$giteaUrl, $token, $org,
|
||||||
|
$templateRepo, $destPath, $branch
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($sourceContent === null) {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, 'ERROR (read source)');
|
||||||
|
$this->errors++;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$commitMsg = "chore: sync {$filename} from {$templateRepo} [skip ci]";
|
||||||
|
|
||||||
|
$this->pushFile(
|
||||||
|
$giteaUrl, $token, $org, $repoName,
|
||||||
|
$destPath, $sourceContent, $branch, $commitMsg, $label
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
echo "\n";
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Push a file to a repo — create or update, skip if identical.
|
||||||
|
*/
|
||||||
|
private function pushFile(
|
||||||
|
string $giteaUrl,
|
||||||
|
string $token,
|
||||||
|
string $org,
|
||||||
|
string $repoName,
|
||||||
|
string $destPath,
|
||||||
|
string $localContent,
|
||||||
|
string $branch,
|
||||||
|
string $commitMsg,
|
||||||
|
string $label
|
||||||
|
): void {
|
||||||
|
$existing = $this->apiRequest(
|
||||||
|
$giteaUrl,
|
||||||
|
$token,
|
||||||
|
'GET',
|
||||||
|
"/api/v1/repos/{$org}/{$repoName}/contents/"
|
||||||
|
. "{$destPath}?ref={$branch}"
|
||||||
|
);
|
||||||
|
|
||||||
|
$encodedContent = base64_encode($localContent);
|
||||||
|
|
||||||
|
if ($existing['code'] === 200) {
|
||||||
|
$data = json_decode($existing['body'], true);
|
||||||
|
$remoteSha = $data['sha'] ?? '';
|
||||||
|
$remoteContent = base64_decode($data['content'] ?? '');
|
||||||
|
|
||||||
|
if ($remoteContent === $localContent) {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, 'IDENTICAL (skipped)');
|
||||||
|
$this->skipped++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($this->dryRun) {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, 'WOULD UPDATE');
|
||||||
|
$this->updated++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$payload = json_encode([
|
||||||
|
'content' => $encodedContent,
|
||||||
|
'sha' => $remoteSha,
|
||||||
|
'message' => $commitMsg,
|
||||||
|
'branch' => $branch,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$response = $this->apiRequest(
|
||||||
|
$giteaUrl,
|
||||||
|
$token,
|
||||||
|
'PUT',
|
||||||
|
"/api/v1/repos/{$org}/{$repoName}/contents/" . $destPath,
|
||||||
|
$payload
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($response['code'] === 200) {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, 'UPDATED');
|
||||||
|
$this->updated++;
|
||||||
|
} else {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, "ERROR (HTTP {$response['code']})");
|
||||||
|
$this->errors++;
|
||||||
|
}
|
||||||
|
} elseif ($existing['code'] === 404) {
|
||||||
|
if ($this->dryRun) {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, 'WOULD CREATE');
|
||||||
|
$this->created++;
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$payload = json_encode([
|
||||||
|
'content' => $encodedContent,
|
||||||
|
'message' => $commitMsg,
|
||||||
|
'branch' => $branch,
|
||||||
|
]);
|
||||||
|
|
||||||
|
$response = $this->apiRequest(
|
||||||
|
$giteaUrl,
|
||||||
|
$token,
|
||||||
|
'POST',
|
||||||
|
"/api/v1/repos/{$org}/{$repoName}/contents/" . $destPath,
|
||||||
|
$payload
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($response['code'] === 201) {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, 'CREATED');
|
||||||
|
$this->created++;
|
||||||
|
} else {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, "ERROR (HTTP {$response['code']})");
|
||||||
|
$this->errors++;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
fprintf(STDERR, "%-45s | %s\n", $label, "ERROR (HTTP {$existing['code']})");
|
||||||
|
$this->errors++;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List workflow files in a repo's .mokogitea/workflows/ directory.
|
||||||
|
*/
|
||||||
|
private function listWorkflows(
|
||||||
|
string $giteaUrl,
|
||||||
|
string $token,
|
||||||
|
string $org,
|
||||||
|
string $repoName,
|
||||||
|
string $branch
|
||||||
|
): ?array {
|
||||||
|
$response = $this->apiRequest(
|
||||||
|
$giteaUrl,
|
||||||
|
$token,
|
||||||
|
'GET',
|
||||||
|
"/api/v1/repos/{$org}/{$repoName}/contents/.mokogitea/workflows?ref={$branch}"
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($response['code'] !== 200) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$data = json_decode($response['body'], true);
|
||||||
|
|
||||||
|
if (!is_array($data)) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Filter to only files (not directories)
|
||||||
|
return array_values(array_filter($data, fn($item) => ($item['type'] ?? '') === 'file'));
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get file content from a repo as a raw string.
|
||||||
|
*/
|
||||||
|
private function getFileContent(
|
||||||
|
string $giteaUrl,
|
||||||
|
string $token,
|
||||||
|
string $org,
|
||||||
|
string $repoName,
|
||||||
|
string $filePath,
|
||||||
|
string $branch
|
||||||
|
): ?string {
|
||||||
|
$response = $this->apiRequest(
|
||||||
|
$giteaUrl,
|
||||||
|
$token,
|
||||||
|
'GET',
|
||||||
|
"/api/v1/repos/{$org}/{$repoName}/contents/{$filePath}?ref={$branch}"
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($response['code'] !== 200) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
$data = json_decode($response['body'], true);
|
||||||
|
|
||||||
|
if (!is_array($data) || !isset($data['content'])) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
return base64_decode($data['content']);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Read a repo's manifest.xml and extract the platform value.
|
||||||
|
* Returns 'generic' if the manifest is missing or has no platform field.
|
||||||
|
*/
|
||||||
|
private function getRepoPlatform(
|
||||||
|
string $giteaUrl,
|
||||||
|
string $token,
|
||||||
|
string $org,
|
||||||
|
string $repoName,
|
||||||
|
string $branch
|
||||||
|
): string {
|
||||||
|
$response = $this->apiRequest(
|
||||||
|
$giteaUrl,
|
||||||
|
$token,
|
||||||
|
'GET',
|
||||||
|
"/api/v1/repos/{$org}/{$repoName}/contents/.mokogitea/manifest.xml?ref={$branch}"
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($response['code'] !== 200) {
|
||||||
|
return 'generic';
|
||||||
|
}
|
||||||
|
|
||||||
|
$data = json_decode($response['body'], true);
|
||||||
|
|
||||||
|
if (!is_array($data) || !isset($data['content'])) {
|
||||||
|
return 'generic';
|
||||||
|
}
|
||||||
|
|
||||||
|
$xmlContent = base64_decode($data['content']);
|
||||||
|
|
||||||
|
if ($xmlContent === false || $xmlContent === '') {
|
||||||
|
return 'generic';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Suppress XML warnings for malformed manifests
|
||||||
|
$previous = libxml_use_internal_errors(true);
|
||||||
|
$xml = simplexml_load_string($xmlContent);
|
||||||
|
libxml_use_internal_errors($previous);
|
||||||
|
|
||||||
|
if ($xml === false) {
|
||||||
|
return 'generic';
|
||||||
|
}
|
||||||
|
|
||||||
|
// Try <governance><platform> (standard location)
|
||||||
|
$platform = '';
|
||||||
|
|
||||||
|
// Register namespace if present
|
||||||
|
$namespaces = $xml->getNamespaces(true);
|
||||||
|
|
||||||
|
if (!empty($namespaces)) {
|
||||||
|
$ns = reset($namespaces);
|
||||||
|
$xml->registerXPathNamespace('mp', $ns);
|
||||||
|
|
||||||
|
$nodes = $xml->xpath('//mp:governance/mp:platform');
|
||||||
|
|
||||||
|
if (!empty($nodes)) {
|
||||||
|
$platform = trim((string) $nodes[0]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback: <identity><platform>
|
||||||
|
if ($platform === '') {
|
||||||
|
$nodes = $xml->xpath('//mp:identity/mp:platform');
|
||||||
|
|
||||||
|
if (!empty($nodes)) {
|
||||||
|
$platform = trim((string) $nodes[0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Fallback: top-level <platform>
|
||||||
|
if ($platform === '') {
|
||||||
|
$nodes = $xml->xpath('//mp:platform');
|
||||||
|
|
||||||
|
if (!empty($nodes)) {
|
||||||
|
$platform = trim((string) $nodes[0]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// No namespace
|
||||||
|
if (isset($xml->governance->platform)) {
|
||||||
|
$platform = trim((string) $xml->governance->platform);
|
||||||
|
} elseif (isset($xml->identity->platform)) {
|
||||||
|
$platform = trim((string) $xml->identity->platform);
|
||||||
|
} elseif (isset($xml->platform)) {
|
||||||
|
$platform = trim((string) $xml->platform);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($platform === '') {
|
||||||
|
return 'generic';
|
||||||
|
}
|
||||||
|
|
||||||
|
return strtolower($platform);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Fetch all non-archived repos in an org (paginated).
|
||||||
|
*/
|
||||||
|
private function fetchOrgRepos(string $giteaUrl, string $token, string $org): ?array
|
||||||
|
{
|
||||||
|
$this->log('INFO', "Fetching repos from org: {$org}");
|
||||||
|
|
||||||
|
$page = 1;
|
||||||
|
$repos = [];
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
$response = $this->apiRequest(
|
||||||
|
$giteaUrl,
|
||||||
|
$token,
|
||||||
|
'GET',
|
||||||
|
"/api/v1/orgs/{$org}/repos?"
|
||||||
|
. "limit=50&page={$page}"
|
||||||
|
);
|
||||||
|
|
||||||
|
if ($response['code'] < 200 || $response['code'] >= 300) {
|
||||||
|
if ($page === 1) {
|
||||||
|
$this->log('ERROR', "Could not fetch repos "
|
||||||
|
. "(HTTP {$response['code']}).");
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
$data = json_decode($response['body'], true);
|
||||||
|
|
||||||
|
if (!is_array($data) || count($data) === 0) {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
|
||||||
|
foreach ($data as $repo) {
|
||||||
|
if (!empty($repo['archived'])) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
$fullName = $repo['full_name'] ?? '';
|
||||||
|
|
||||||
|
if ($fullName !== '') {
|
||||||
|
$repos[] = $fullName;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$page++;
|
||||||
|
}
|
||||||
|
|
||||||
|
return $repos;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Make an HTTP request to the Gitea API.
|
||||||
|
*/
|
||||||
|
private function apiRequest(
|
||||||
|
string $giteaUrl,
|
||||||
|
string $token,
|
||||||
|
string $method,
|
||||||
|
string $endpoint,
|
||||||
|
?string $body = null
|
||||||
|
): array {
|
||||||
|
$url = $giteaUrl . $endpoint;
|
||||||
|
|
||||||
|
$ch = curl_init();
|
||||||
|
curl_setopt($ch, CURLOPT_URL, $url);
|
||||||
|
curl_setopt($ch, CURLOPT_RETURNTRANSFER, true);
|
||||||
|
curl_setopt($ch, CURLOPT_CUSTOMREQUEST, $method);
|
||||||
|
curl_setopt($ch, CURLOPT_HTTPHEADER, [
|
||||||
|
'Content-Type: application/json',
|
||||||
|
'Accept: application/json',
|
||||||
|
"Authorization: token {$token}",
|
||||||
|
]);
|
||||||
|
|
||||||
|
if ($body !== null) {
|
||||||
|
curl_setopt($ch, CURLOPT_POSTFIELDS, $body);
|
||||||
|
}
|
||||||
|
|
||||||
|
$responseBody = curl_exec($ch);
|
||||||
|
$httpCode = (int) curl_getinfo(
|
||||||
|
$ch,
|
||||||
|
CURLINFO_HTTP_CODE
|
||||||
|
);
|
||||||
|
|
||||||
|
if (curl_errno($ch)) {
|
||||||
|
$error = curl_error($ch);
|
||||||
|
curl_close($ch);
|
||||||
|
|
||||||
|
return [
|
||||||
|
'code' => 0,
|
||||||
|
'body' => "cURL error: {$error}",
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
curl_close($ch);
|
||||||
|
|
||||||
|
return ['code' => $httpCode, 'body' => $responseBody];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$app = new WorkflowSyncCli();
|
||||||
|
exit($app->execute());
|
||||||
+4
-3
@@ -1,6 +1,6 @@
|
|||||||
{
|
{
|
||||||
"name": "mokoconsulting-tech/enterprise",
|
"name": "mokoconsulting-tech/mokocli",
|
||||||
"description": "moko-platform Enterprise API \u2014 PHP implementation",
|
"description": "MokoCLI Enterprise API \u2014 PHP implementation",
|
||||||
"type": "library",
|
"type": "library",
|
||||||
"version": "09.23.00",
|
"version": "09.23.00",
|
||||||
"license": "GPL-3.0-or-later",
|
"license": "GPL-3.0-or-later",
|
||||||
@@ -51,7 +51,8 @@
|
|||||||
"lib/Enterprise/CliFramework.php"
|
"lib/Enterprise/CliFramework.php"
|
||||||
],
|
],
|
||||||
"files": [
|
"files": [
|
||||||
"src/functions.php"
|
"src/functions.php",
|
||||||
|
"src/aliases.php"
|
||||||
]
|
]
|
||||||
},
|
},
|
||||||
"archive": {
|
"archive": {
|
||||||
|
|||||||
@@ -8,11 +8,11 @@
|
|||||||
* SPDX-License-Identifier: GPL-3.0-or-later
|
* SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
*
|
*
|
||||||
* FILE INFORMATION
|
* FILE INFORMATION
|
||||||
* DEFGROUP: MokoPlatform.Scripts.Deploy
|
* DEFGROUP: MokoCLI.Scripts.Deploy
|
||||||
* INGROUP: MokoPlatform
|
* INGROUP: MokoCLI
|
||||||
* REPO: https://git.mokoconsulting.tech/MokoConsulting/moko-platform
|
* REPO: https://git.mokoconsulting.tech/MokoConsulting/mokocli
|
||||||
* PATH: /deploy/backup-before-deploy.php
|
* PATH: /deploy/backup-before-deploy.php
|
||||||
* VERSION: 09.24.00
|
* VERSION: 09.25.05
|
||||||
* BRIEF: Snapshot Joomla directories before deployment for rollback capability
|
* BRIEF: Snapshot Joomla directories before deployment for rollback capability
|
||||||
*/
|
*/
|
||||||
|
|
||||||
|
|||||||
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user