From c2e97e5b77ae5f493712422e7854a20415f94f34 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:08 -0500 Subject: [PATCH 01/27] chore: update LICENSE from MokoStandards -- 2.52.0 From bb0cb0484df0e8fc49b0a5c297e6a89a122c4ee3 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:08 -0500 Subject: [PATCH 02/27] chore: update update.xml from MokoStandards -- 2.52.0 From 34ed5ef12be1ecc603b216c1cb7b035f70191195 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:09 -0500 Subject: [PATCH 03/27] chore: update phpstan.neon from MokoStandards -- 2.52.0 From eaefc4fa93d0fbb4b1915eae8d211b46432fb26f Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:09 -0500 Subject: [PATCH 04/27] chore: update Makefile from MokoStandards -- 2.52.0 From 3b91a11ac9fdc941da9caf8e8b3ebbc572aa5d39 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:10 -0500 Subject: [PATCH 05/27] chore: update .gitignore from MokoStandards -- 2.52.0 From 91b480cedf21234e5690890e47248c7c0f3aed59 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:11 -0500 Subject: [PATCH 06/27] chore: update composer.json from MokoStandards -- 2.52.0 From e61a40b43ee5dbcabe9d96da6d390fde1db4d662 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:11 -0500 Subject: [PATCH 07/27] chore: update .moko-standards from MokoStandards -- 2.52.0 From 0f8eb231e48441c3b0140b0d845cf14e35e11a38 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:12 -0500 Subject: [PATCH 08/27] chore: update .github/copilot.yml from MokoStandards -- 2.52.0 From dba6c0884b5bc8a17f325de9419905004c2199d8 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:13 -0500 Subject: [PATCH 09/27] chore: update .github/copilot-instructions.md from MokoStandards -- 2.52.0 From c206eed1fdd693feefe7b889717abae1d4ee44df Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:14 -0500 Subject: [PATCH 10/27] chore: update .github/CLAUDE.md from MokoStandards -- 2.52.0 From 9b7140979287e201466ba21b0bec7a46a2a5e7cf Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:14 -0500 Subject: [PATCH 11/27] chore: update .github/workflows/codeql-analysis.yml from MokoStandards -- 2.52.0 From 29f2b26e65f32cb0bb5b80677fcecfd35aa19598 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:15 -0500 Subject: [PATCH 12/27] chore: update .github/workflows/standards-compliance.yml from MokoStandards -- 2.52.0 From d1f5965a31b18b587f04f09b8fa2a7a83a8e14ab Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:16 -0500 Subject: [PATCH 13/27] chore: update .github/workflows/enterprise-firewall-setup.yml from MokoStandards -- 2.52.0 From 0aaafdfa8c39d9c82567955c5318a8c588a62a3f Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:16 -0500 Subject: [PATCH 14/27] chore: update .github/workflows/deploy-dev.yml from MokoStandards --- .github/workflows/deploy-dev.yml | 117 +++++++++++++++++++++---------- 1 file changed, 80 insertions(+), 37 deletions(-) diff --git a/.github/workflows/deploy-dev.yml b/.github/workflows/deploy-dev.yml index af69789..c93b7a9 100644 --- a/.github/workflows/deploy-dev.yml +++ b/.github/workflows/deploy-dev.yml @@ -22,7 +22,7 @@ # INGROUP: MokoStandards.Deploy # REPO: https://github.com/mokoconsulting-tech/MokoStandards # PATH: /templates/workflows/shared/deploy-dev.yml -# VERSION: 04.00.25 +# VERSION: 04.00.27 # BRIEF: SFTP deployment workflow for development server — synced to all governed repos # NOTE: Synced via bulk-repo-sync to .github/workflows/deploy-dev.yml in all governed repos. # Port is resolved in order: DEV_FTP_PORT variable → :port suffix in DEV_FTP_HOST → 22. @@ -35,14 +35,11 @@ name: Deploy to Dev Server (SFTP) # # Required org-level variables: DEV_FTP_HOST, DEV_FTP_PATH, DEV_FTP_USERNAME # Optional org-level variable: DEV_FTP_PORT (auto-detected from host or defaults to 22) -# Optional org/repo variable: DEV_FTP_PATH_SUFFIX -# Optional org/repo variable: CUSTOM_FOLDER — when set, appended to the remote path after -# DEV_FTP_PATH_SUFFIX; used automatically for Dolibarr modules -# Optional org/repo variable: FTP_IGNORE — comma-delimited list of regex patterns, each enclosed in -# double quotes, for files/paths to exclude from upload, e.g.: -# "\.git*", "\.DS_Store", "configuration\.php", "\.ps1" -# Patterns are tested against the forward-slash relative path of each -# file (e.g. "subdir/file.txt"). The repository .gitignore is also +# Optional org/repo variable: CUSTOM_NAME — when set, appended to DEV_FTP_PATH to form the +# full remote destination: DEV_FTP_PATH/CUSTOM_NAME +# Ignore rules: Place a .ftp_ignore file in the repository root. Each non-empty, +# non-comment line is a regex pattern tested against the relative path +# of each file (e.g. "subdir/file.txt"). The .gitignore is also # respected automatically. # Required org-level secret: DEV_FTP_KEY (preferred) or DEV_FTP_PASSWORD # @@ -56,8 +53,6 @@ on: - develop - dev - development - paths: - - 'src/**' pull_request: types: [opened, synchronize, reopened, closed] branches: @@ -163,15 +158,47 @@ jobs: if: steps.source.outputs.skip == 'false' env: SOURCE_DIR: ${{ steps.source.outputs.dir }} - FTP_IGNORE: ${{ vars.FTP_IGNORE }} run: | - # ── Parse FTP_IGNORE ───────────────────────────────────────────────── + # ── Convert a gitignore-style glob line to an ERE pattern ────────────── + ftp_ignore_to_regex() { + local line="$1" + local anchored=false + # Strip inline comments and whitespace + line=$(printf '%s' "$line" | sed 's/[[:space:]]*#.*$//' | sed 's/^[[:space:]]*//;s/[[:space:]]*$//') + [ -z "$line" ] && return + # Skip negation patterns (not supported) + [[ "$line" == !* ]] && return + # Trailing slash = directory marker; strip it + line="${line%/}" + # Leading slash = anchored to root; strip it + if [[ "$line" == /* ]]; then + anchored=true + line="${line#/}" + fi + # Escape ERE special chars, then restore glob semantics + local regex + regex=$(printf '%s' "$line" \ + | sed 's/[.+^${}()|[\\]/\\&/g' \ + | sed 's/\\\*\\\*/\x01/g' \ + | sed 's/\\\*/[^\/]*/g' \ + | sed 's/\x01/.*/g' \ + | sed 's/\\\?/[^\/]/g') + if $anchored; then + printf '^%s(/|$)' "$regex" + else + printf '(^|/)%s(/|$)' "$regex" + fi + } + + # ── Read .ftp_ignore (gitignore-style globs) ───────────────────────── IGNORE_PATTERNS=() - if [ -n "$FTP_IGNORE" ]; then - while IFS= read -r -d ',' token; do - pattern=$(printf '%s' "$token" | sed 's/^[[:space:]]*"//;s/"[[:space:]]*$//') - [ -n "$pattern" ] && IGNORE_PATTERNS+=("$pattern") - done <<< "${FTP_IGNORE}," + IGNORE_SOURCES=() + if [ -f ".ftp_ignore" ]; then + while IFS= read -r line; do + [[ "$line" =~ ^[[:space:]]*$ || "$line" =~ ^[[:space:]]*# ]] && continue + regex=$(ftp_ignore_to_regex "$line") + [ -n "$regex" ] && IGNORE_PATTERNS+=("$regex") && IGNORE_SOURCES+=("$line") + done < ".ftp_ignore" fi # ── Walk src/ and classify every file ──────────────────────────────── @@ -180,9 +207,9 @@ jobs: while IFS= read -r -d '' file; do rel="${file#${SOURCE_DIR}/}" SKIP=false - for pat in "${IGNORE_PATTERNS[@]}"; do - if echo "$rel" | grep -qE "$pat" 2>/dev/null; then - IGNORED_FILES+=("$rel | FTP_IGNORE \`$pat\`") + for i in "${!IGNORE_PATTERNS[@]}"; do + if echo "$rel" | grep -qE "${IGNORE_PATTERNS[$i]}" 2>/dev/null; then + IGNORED_FILES+=("$rel | .ftp_ignore \`${IGNORE_SOURCES[$i]}\`") SKIP=true; break fi done @@ -262,13 +289,10 @@ jobs: if: steps.source.outputs.skip == 'false' id: remote env: - DEV_FTP_PATH: ${{ vars.DEV_FTP_PATH }} - DEV_FTP_PATH_SUFFIX: ${{ vars.DEV_FTP_PATH_SUFFIX }} - CUSTOM_FOLDER: ${{ vars.CUSTOM_FOLDER }} + DEV_FTP_PATH: ${{ vars.DEV_FTP_PATH }} + CUSTOM_NAME: ${{ vars.CUSTOM_NAME }} run: | BASE="$DEV_FTP_PATH" - SUFFIX="$DEV_FTP_PATH_SUFFIX" - CUSTOM="$CUSTOM_FOLDER" if [ -z "$BASE" ]; then echo "❌ DEV_FTP_PATH is not set." @@ -277,21 +301,42 @@ jobs: exit 1 fi - # Always append suffix when set — path is BASE/SUFFIX - if [ -n "$SUFFIX" ]; then - REMOTE="${BASE%/}/${SUFFIX#/}" + # Path format: DEV_FTP_PATH/CUSTOM_NAME (CUSTOM_NAME is optional) + if [ -n "$CUSTOM_NAME" ]; then + REMOTE="${BASE%/}/${CUSTOM_NAME#/}" else REMOTE="$BASE" fi - # Append CUSTOM_FOLDER when set — makes Dolibarr module paths automatic - if [ -n "$CUSTOM" ]; then - REMOTE="${REMOTE%/}/${CUSTOM#/}" - echo "ℹ️ CUSTOM_FOLDER appended: ${CUSTOM}" + # ── Platform-specific path safety guards ────────────────────────────── + PLATFORM="" + if [ -f ".moko-standards" ]; then + PLATFORM=$(grep -E '^platform:' .moko-standards | sed 's/.*:[[:space:]]*//' | tr -d '"') fi + if [ "$PLATFORM" = "crm-module" ]; then + # Dolibarr modules must deploy under htdocs/custom/ — guard against + # accidentally overwriting server root or unrelated directories. + if [[ "$REMOTE" != *custom* ]]; then + echo "❌ Safety check failed: Dolibarr (crm-module) remote path must contain 'custom'." + echo " Current path: ${REMOTE}" + echo " Set CUSTOM_NAME to the module's htdocs/custom/ subdirectory." + exit 1 + fi + fi + + if [ "$PLATFORM" = "waas-component" ]; then + # Joomla extensions may only deploy to the server's tmp/ directory. + if [[ "$REMOTE" != *tmp* ]]; then + echo "❌ Safety check failed: Joomla (waas-component) remote path must contain 'tmp'." + echo " Current path: ${REMOTE}" + echo " Set CUSTOM_NAME to a path under the server tmp/ directory." + exit 1 + fi + fi + + echo "ℹ️ Remote path: ${REMOTE}" echo "path=${REMOTE}" >> "$GITHUB_OUTPUT" - echo "Remote path: ${REMOTE}" - name: Detect SFTP authentication method if: steps.source.outputs.skip == 'false' @@ -345,9 +390,7 @@ jobs: composer install --no-dev --no-interaction --quiet - name: Clear remote destination folder - if: >- - steps.source.outputs.skip == 'false' && - inputs.clear_remote == true + if: steps.source.outputs.skip == 'false' env: SFTP_HOST: ${{ steps.conn.outputs.host }} SFTP_PORT: ${{ steps.conn.outputs.port }} -- 2.52.0 From 0c0dcb5be27d7d1b3152e2661b5cd0e6ccef50d6 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:17 -0500 Subject: [PATCH 15/27] chore: add .github/workflows/sync-version-on-merge.yml from MokoStandards --- .github/workflows/sync-version-on-merge.yml | 140 ++++++++++++++++++++ 1 file changed, 140 insertions(+) create mode 100644 .github/workflows/sync-version-on-merge.yml diff --git a/.github/workflows/sync-version-on-merge.yml b/.github/workflows/sync-version-on-merge.yml new file mode 100644 index 0000000..54d8118 --- /dev/null +++ b/.github/workflows/sync-version-on-merge.yml @@ -0,0 +1,140 @@ +# Copyright (C) 2026 Moko Consulting +# +# This file is part of a Moko Consulting project. +# +# SPDX-License-Identifier: GPL-3.0-or-later +# +# FILE INFORMATION +# DEFGROUP: GitHub.Workflow +# INGROUP: MokoStandards.Automation +# REPO: https://github.com/mokoconsulting-tech/MokoStandards +# PATH: /templates/workflows/shared/sync-version-on-merge.yml +# VERSION: 04.00.35 +# BRIEF: Auto-bump patch version on every push to main and propagate to all file headers +# NOTE: Synced via bulk-repo-sync to .github/workflows/sync-version-on-merge.yml in all governed repos. +# README.md is the single source of truth for the repository version. + +name: Sync Version from README + +on: + push: + branches: + - main + - master + workflow_dispatch: + inputs: + dry_run: + description: 'Dry run (preview only, no commit)' + type: boolean + default: false + +permissions: + contents: write + issues: write + +jobs: + sync-version: + name: Propagate README version + runs-on: ubuntu-latest + + steps: + - name: Checkout repository + uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6 + with: + token: ${{ secrets.GH_TOKEN || github.token }} + fetch-depth: 0 + + - name: Set up PHP + uses: shivammathur/setup-php@44454db4f0199b8b9685a5d763dc37cbf79108e1 # v2.31.0 + with: + php-version: '8.1' + tools: composer + + - name: Setup MokoStandards tools + env: + GH_TOKEN: ${{ secrets.GH_TOKEN || github.token }} + COMPOSER_AUTH: '{"github-oauth":{"github.com":"${{ secrets.GH_TOKEN || github.token }}"}}' + run: | + git clone --depth 1 --quiet \ + "https://x-access-token:${GH_TOKEN}@github.com/mokoconsulting-tech/MokoStandards.git" \ + /tmp/mokostandards + cd /tmp/mokostandards + composer install --no-dev --no-interaction --quiet + + - name: Auto-bump patch version + if: ${{ github.event_name == 'push' && github.actor != 'github-actions[bot]' }} + run: | + # If README.md was part of this push, the author already bumped the version — skip. + if git diff --name-only HEAD~1 HEAD 2>/dev/null | grep -q '^README\.md$'; then + echo "README.md changed in this push — skipping auto-bump" + exit 0 + fi + + CURRENT=$(grep -oP '^\s*VERSION:\s*\K[0-9]{2}\.[0-9]{2}\.[0-9]{2}' README.md | head -1) + if [ -z "$CURRENT" ]; then + echo "⚠️ No VERSION found in README.md — skipping auto-bump" + exit 0 + fi + + # Increment the patch component (zero-padded to 2 digits) + MAJOR=$(echo "$CURRENT" | cut -d. -f1) + MINOR=$(echo "$CURRENT" | cut -d. -f2) + PATCH=$(echo "$CURRENT" | cut -d. -f3) + NEW_PATCH=$(printf '%02d' $(( 10#$PATCH + 1 ))) + NEW_VERSION="${MAJOR}.${MINOR}.${NEW_PATCH}" + + echo "Auto-bumping patch: $CURRENT → $NEW_VERSION" + sed -i "s/^\(\s*VERSION:\s*\)${CURRENT}/\1${NEW_VERSION}/" README.md + git config --local user.email "github-actions[bot]@users.noreply.github.com" + git config --local user.name "github-actions[bot]" + git add README.md + git commit -m "chore(version): auto-bump patch ${CURRENT} → ${NEW_VERSION} [skip ci]" \ + --author="github-actions[bot] " + git push + + - name: Extract version from README.md + id: readme_version + run: | + git pull --ff-only 2>/dev/null || true + VERSION=$(grep -oP '^\s*VERSION:\s*\K[0-9]{2}\.[0-9]{2}\.[0-9]{2}' README.md | head -1) + if [ -z "$VERSION" ]; then + echo "⚠️ No VERSION in README.md — skipping propagation" + echo "skip=true" >> $GITHUB_OUTPUT + exit 0 + fi + echo "version=$VERSION" >> $GITHUB_OUTPUT + echo "skip=false" >> $GITHUB_OUTPUT + echo "✅ README.md version: $VERSION" + + - name: Run version sync + if: ${{ steps.readme_version.outputs.skip != 'true' && inputs.dry_run != true }} + run: | + php /tmp/mokostandards/api/maintenance/update_version_from_readme.php \ + --path . \ + --create-issue \ + --repo "${{ github.repository }}" + env: + GH_TOKEN: ${{ secrets.GH_TOKEN || github.token }} + + - name: Commit updated files + if: ${{ steps.readme_version.outputs.skip != 'true' && inputs.dry_run != true }} + run: | + if git diff --quiet; then + echo "ℹ️ No version changes needed" + exit 0 + fi + VERSION="${{ steps.readme_version.outputs.version }}" + git config --local user.email "github-actions[bot]@users.noreply.github.com" + git config --local user.name "github-actions[bot]" + git add -A + git commit -m "chore(version): sync badges and headers to ${VERSION} [skip ci]" \ + --author="github-actions[bot] " + git push + + - name: Summary + run: | + VERSION="${{ steps.readme_version.outputs.version }}" + echo "## 📦 Version Sync — ${VERSION}" >> $GITHUB_STEP_SUMMARY + echo "" >> $GITHUB_STEP_SUMMARY + echo "**Source:** \`README.md\` FILE INFORMATION block" >> $GITHUB_STEP_SUMMARY + echo "**Version:** \`${VERSION}\`" >> $GITHUB_STEP_SUMMARY -- 2.52.0 From 3f4dc65244c953d16ca60b57bfec7bb4ccdd27e9 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:17 -0500 Subject: [PATCH 16/27] chore: update .github/ISSUE_TEMPLATE/config.yml from MokoStandards -- 2.52.0 From 8be521e453bca07aedf49d7b77de28b5aa4864bc Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:18 -0500 Subject: [PATCH 17/27] chore: update .github/ISSUE_TEMPLATE/adr.md from MokoStandards -- 2.52.0 From 3eec6be8f96c77e8b060043ccc5a1ac200169c7f Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:18 -0500 Subject: [PATCH 18/27] chore: update .github/ISSUE_TEMPLATE/bug_report.md from MokoStandards -- 2.52.0 From d104c12bf2ef7a7140b76d3147c8d6caf29247b2 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:19 -0500 Subject: [PATCH 19/27] chore: update .github/ISSUE_TEMPLATE/documentation.md from MokoStandards -- 2.52.0 From 2999240d972e3849072297ed8d02eeea5359b052 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:20 -0500 Subject: [PATCH 20/27] chore: update .github/ISSUE_TEMPLATE/enterprise_support.md from MokoStandards -- 2.52.0 From cde4cc1cb2a7e183acb267200cfc05ceecc4f695 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:20 -0500 Subject: [PATCH 21/27] chore: update .github/ISSUE_TEMPLATE/feature_request.md from MokoStandards -- 2.52.0 From a7fe992bb2c570107a0717f3582beaa5f499da95 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:21 -0500 Subject: [PATCH 22/27] chore: update .github/ISSUE_TEMPLATE/firewall-request.md from MokoStandards -- 2.52.0 From df1fde6ca7ee2909b83d4b2f9c8c8dfed50a42b1 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:22 -0500 Subject: [PATCH 23/27] chore: update .github/ISSUE_TEMPLATE/question.md from MokoStandards -- 2.52.0 From 7ebf01d0f5f62845c8390d3edd658ce2b3b1af08 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:22 -0500 Subject: [PATCH 24/27] chore: update .github/ISSUE_TEMPLATE/request-license.md from MokoStandards -- 2.52.0 From d5908530efcb0b24324fa37b023dd0cf9ea0493e Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:23 -0500 Subject: [PATCH 25/27] chore: update .github/ISSUE_TEMPLATE/rfc.md from MokoStandards -- 2.52.0 From f476b9349e9967c23cc2125f4a5ce92da02d1119 Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:23 -0500 Subject: [PATCH 26/27] chore: update .github/ISSUE_TEMPLATE/security.md from MokoStandards -- 2.52.0 From d3ef980f49c4289ef74dbfc9512259d27fb0150f Mon Sep 17 00:00:00 2001 From: Jonathan Miller <230051081+jmiller-moko@users.noreply.github.com> Date: Tue, 24 Mar 2026 14:51:24 -0500 Subject: [PATCH 27/27] chore: update .github/ISSUE_TEMPLATE/joomla_issue.md from MokoStandards -- 2.52.0