Compare commits
7 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| b3d955e1a8 | |||
| f5e8d0fe03 | |||
| 5815a65a39 | |||
| ad1c0cf349 | |||
| 8b6e260b28 | |||
| eb7f48d3a2 | |||
| 974b971340 |
@@ -0,0 +1,126 @@
|
|||||||
|
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
|
#
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
#
|
||||||
|
# FILE INFORMATION
|
||||||
|
# DEFGROUP: Gitea.Workflow
|
||||||
|
# INGROUP: MokoStandards.Deploy
|
||||||
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/MokoStandards-API
|
||||||
|
# PATH: /templates/workflows/joomla/deploy-manual.yml.template
|
||||||
|
# VERSION: 04.07.00
|
||||||
|
# BRIEF: Manual SFTP deploy to dev server for Joomla repos
|
||||||
|
|
||||||
|
name: "Universal: Deploy to Dev (Manual)"
|
||||||
|
|
||||||
|
on:
|
||||||
|
workflow_dispatch:
|
||||||
|
inputs:
|
||||||
|
clear_remote:
|
||||||
|
description: 'Delete all remote files before uploading'
|
||||||
|
required: false
|
||||||
|
default: 'false'
|
||||||
|
type: boolean
|
||||||
|
|
||||||
|
env:
|
||||||
|
FORCE_JAVASCRIPT_ACTIONS_TO_NODE24: true
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
deploy:
|
||||||
|
name: SFTP Deploy to Dev
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout repository
|
||||||
|
uses: actions/checkout@de0fac2e4500dabe0009e67214ff5f5447ce83dd # v6
|
||||||
|
|
||||||
|
- name: Setup PHP
|
||||||
|
run: |
|
||||||
|
php -v && composer --version
|
||||||
|
|
||||||
|
- name: Setup MokoStandards tools
|
||||||
|
env:
|
||||||
|
GA_TOKEN: ${{ secrets.GA_TOKEN || secrets.GA_TOKEN || github.token }}
|
||||||
|
MOKO_CLONE_TOKEN: ${{ secrets.GA_TOKEN || secrets.GA_TOKEN || github.token }}
|
||||||
|
MOKO_CLONE_HOST: ${{ secrets.GA_TOKEN && 'git.mokoconsulting.tech/MokoConsulting' || 'github.com/mokoconsulting-tech' }}
|
||||||
|
COMPOSER_AUTH: '{"github-oauth":{"github.com":"${{ secrets.GA_TOKEN || github.token }}"}}'
|
||||||
|
run: |
|
||||||
|
git clone --depth 1 --branch main --quiet \
|
||||||
|
"https://x-access-token:${MOKO_CLONE_TOKEN}@${MOKO_CLONE_HOST}/MokoStandards-API.git" \
|
||||||
|
/tmp/mokostandards-api 2>/dev/null || true
|
||||||
|
if [ -d "/tmp/mokostandards-api" ] && [ -f "/tmp/mokostandards-api/composer.json" ]; then
|
||||||
|
cd /tmp/mokostandards-api && composer install --no-dev --no-interaction --quiet 2>/dev/null || true
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Check FTP configuration
|
||||||
|
id: check
|
||||||
|
env:
|
||||||
|
HOST: ${{ vars.DEV_FTP_HOST }}
|
||||||
|
PATH_VAR: ${{ vars.DEV_FTP_PATH }}
|
||||||
|
PORT: ${{ vars.DEV_FTP_PORT }}
|
||||||
|
run: |
|
||||||
|
if [ -z "$HOST" ] || [ -z "$PATH_VAR" ]; then
|
||||||
|
echo "DEV_FTP_HOST or DEV_FTP_PATH not configured -- cannot deploy"
|
||||||
|
echo "skip=true" >> "$GITHUB_OUTPUT"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
echo "skip=false" >> "$GITHUB_OUTPUT"
|
||||||
|
echo "host=$HOST" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
REMOTE="${PATH_VAR%/}"
|
||||||
|
echo "remote=$REMOTE" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
[ -z "$PORT" ] && PORT="22"
|
||||||
|
echo "port=$PORT" >> "$GITHUB_OUTPUT"
|
||||||
|
|
||||||
|
- name: Deploy via SFTP
|
||||||
|
if: steps.check.outputs.skip != 'true'
|
||||||
|
env:
|
||||||
|
SFTP_KEY: ${{ secrets.DEV_FTP_KEY }}
|
||||||
|
SFTP_PASS: ${{ secrets.DEV_FTP_PASSWORD }}
|
||||||
|
SFTP_USER: ${{ vars.DEV_FTP_USERNAME }}
|
||||||
|
run: |
|
||||||
|
SOURCE_DIR="src"
|
||||||
|
[ ! -d "$SOURCE_DIR" ] && SOURCE_DIR="htdocs"
|
||||||
|
[ ! -d "$SOURCE_DIR" ] && { echo "No src/ or htdocs/ -- nothing to deploy"; exit 0; }
|
||||||
|
|
||||||
|
printf '{"host":"%s","port":%s,"username":"%s","remotePath":"%s"' \
|
||||||
|
"${{ steps.check.outputs.host }}" "${{ steps.check.outputs.port }}" "$SFTP_USER" "${{ steps.check.outputs.remote }}" \
|
||||||
|
> /tmp/sftp-config.json
|
||||||
|
|
||||||
|
if [ -n "$SFTP_KEY" ]; then
|
||||||
|
echo "$SFTP_KEY" > /tmp/deploy_key
|
||||||
|
chmod 600 /tmp/deploy_key
|
||||||
|
printf ',"privateKeyPath":"/tmp/deploy_key"}' >> /tmp/sftp-config.json
|
||||||
|
else
|
||||||
|
printf ',"password":"%s"}' "$SFTP_PASS" >> /tmp/sftp-config.json
|
||||||
|
fi
|
||||||
|
|
||||||
|
DEPLOY_ARGS=(--path . --src-dir "$SOURCE_DIR" --config /tmp/sftp-config.json)
|
||||||
|
[ "${{ inputs.clear_remote }}" = "true" ] && DEPLOY_ARGS+=(--clear-remote)
|
||||||
|
|
||||||
|
PLATFORM=$(php /tmp/mokostandards-api/cli/platform_detect.php --path . 2>/dev/null || true)
|
||||||
|
if [ "$PLATFORM" = "waas-component" ] && [ -f "/tmp/mokostandards-api/deploy/deploy-joomla.php" ]; then
|
||||||
|
php /tmp/mokostandards-api/deploy/deploy-joomla.php "${DEPLOY_ARGS[@]}"
|
||||||
|
else
|
||||||
|
php /tmp/mokostandards-api/deploy/deploy-sftp.php "${DEPLOY_ARGS[@]}"
|
||||||
|
fi
|
||||||
|
|
||||||
|
rm -f /tmp/deploy_key /tmp/sftp-config.json
|
||||||
|
|
||||||
|
- name: Summary
|
||||||
|
if: always()
|
||||||
|
run: |
|
||||||
|
if [ "${{ steps.check.outputs.skip }}" = "true" ]; then
|
||||||
|
echo "### Deploy Skipped -- FTP not configured" >> $GITHUB_STEP_SUMMARY
|
||||||
|
else
|
||||||
|
echo "### Manual Dev Deploy Complete" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Field | Value |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "|-------|-------|" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Host | \`${{ steps.check.outputs.host }}\` |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Remote | \`${{ steps.check.outputs.remote }}\` |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
echo "| Clear | ${{ inputs.clear_remote }} |" >> $GITHUB_STEP_SUMMARY
|
||||||
|
fi
|
||||||
@@ -5,7 +5,7 @@
|
|||||||
# FILE INFORMATION
|
# FILE INFORMATION
|
||||||
# DEFGROUP: Gitea.Workflow
|
# DEFGROUP: Gitea.Workflow
|
||||||
# INGROUP: mokocli.Automation
|
# INGROUP: mokocli.Automation
|
||||||
# VERSION: 01.39.02
|
# VERSION: 01.30.00
|
||||||
# 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"
|
||||||
|
|||||||
@@ -0,0 +1,82 @@
|
|||||||
|
# Copyright (C) 2026 Moko Consulting <hello@mokoconsulting.tech>
|
||||||
|
#
|
||||||
|
# SPDX-License-Identifier: GPL-3.0-or-later
|
||||||
|
#
|
||||||
|
# FILE INFORMATION
|
||||||
|
# DEFGROUP: Gitea.Workflow
|
||||||
|
# INGROUP: MokoStandards.Security
|
||||||
|
# REPO: https://git.mokoconsulting.tech/MokoConsulting/MokoStandards
|
||||||
|
# PATH: /.gitea/workflows/security-audit.yml
|
||||||
|
# VERSION: 01.00.00
|
||||||
|
# BRIEF: Dependency vulnerability scanning for composer and npm packages
|
||||||
|
|
||||||
|
name: "Universal: Security Audit"
|
||||||
|
|
||||||
|
on:
|
||||||
|
schedule:
|
||||||
|
- cron: '0 6 * * 1' # Weekly on Monday at 06:00 UTC
|
||||||
|
pull_request:
|
||||||
|
branches:
|
||||||
|
- main
|
||||||
|
paths:
|
||||||
|
- 'composer.json'
|
||||||
|
- 'composer.lock'
|
||||||
|
- 'package.json'
|
||||||
|
- 'package-lock.json'
|
||||||
|
workflow_dispatch:
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
env:
|
||||||
|
NTFY_URL: ${{ vars.NTFY_URL || 'https://ntfy.mokoconsulting.tech' }}
|
||||||
|
NTFY_TOPIC: ${{ vars.NTFY_TOPIC || 'gitea-security' }}
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
audit:
|
||||||
|
name: Dependency Audit
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
|
||||||
|
- name: Composer audit
|
||||||
|
if: hashFiles('composer.lock') != ''
|
||||||
|
run: |
|
||||||
|
echo "=== Composer Security Audit ==="
|
||||||
|
if ! command -v composer &> /dev/null; then
|
||||||
|
sudo apt-get update -qq
|
||||||
|
sudo apt-get install -y -qq php-cli composer >/dev/null 2>&1
|
||||||
|
fi
|
||||||
|
composer audit --format=plain 2>&1 | tee /tmp/composer-audit.txt
|
||||||
|
RESULT=$?
|
||||||
|
if [ $RESULT -ne 0 ]; then
|
||||||
|
echo "::warning::Composer vulnerabilities found"
|
||||||
|
echo "composer_vulnerable=true" >> "$GITHUB_ENV"
|
||||||
|
else
|
||||||
|
echo "No known vulnerabilities in composer dependencies"
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: NPM audit
|
||||||
|
if: hashFiles('package-lock.json') != ''
|
||||||
|
run: |
|
||||||
|
echo "=== NPM Security Audit ==="
|
||||||
|
npm audit --production 2>&1 | tee /tmp/npm-audit.txt || true
|
||||||
|
if npm audit --production 2>&1 | grep -q "found 0 vulnerabilities"; then
|
||||||
|
echo "No known vulnerabilities in npm dependencies"
|
||||||
|
else
|
||||||
|
echo "::warning::NPM vulnerabilities found"
|
||||||
|
echo "npm_vulnerable=true" >> "$GITHUB_ENV"
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Notify on vulnerabilities
|
||||||
|
if: env.composer_vulnerable == 'true' || env.npm_vulnerable == 'true'
|
||||||
|
run: |
|
||||||
|
REPO="${{ github.event.repository.name }}"
|
||||||
|
curl -sS \
|
||||||
|
-H "Title: ${REPO} has vulnerable dependencies" \
|
||||||
|
-H "Tags: lock,warning" \
|
||||||
|
-H "Priority: high" \
|
||||||
|
-d "Security audit found vulnerabilities. Review dependency updates." \
|
||||||
|
"${NTFY_URL}/${NTFY_TOPIC}" || true
|
||||||
+11
-4
@@ -1,6 +1,17 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
## [01.30.00] --- 2026-06-22
|
||||||
|
|
||||||
|
## [01.30.00] --- 2026-06-22
|
||||||
|
|
||||||
|
### Changed
|
||||||
|
- Remote upload failure no longer marks the entire backup as failed — local archive is preserved with status 'complete' (#66)
|
||||||
|
|
||||||
|
### Added
|
||||||
|
- Snapshots now capture tags, custom fields, field values, and field-category mappings when articles are included (#57)
|
||||||
|
- Snapshot retention settings: max count and max age with automatic cleanup (#63)
|
||||||
|
|
||||||
## [01.27.03] --- 2026-06-21
|
## [01.27.03] --- 2026-06-21
|
||||||
|
|
||||||
## [01.27.03] --- 2026-06-21
|
## [01.27.03] --- 2026-06-21
|
||||||
@@ -8,7 +19,3 @@
|
|||||||
## [01.27.00] --- 2026-06-21
|
## [01.27.00] --- 2026-06-21
|
||||||
|
|
||||||
## [01.27.00] --- 2026-06-21
|
## [01.27.00] --- 2026-06-21
|
||||||
|
|
||||||
## [01.27.00] --- 2026-06-21
|
|
||||||
|
|
||||||
## [01.27.00] --- 2026-06-21
|
|
||||||
|
|||||||
@@ -1,6 +1,6 @@
|
|||||||
# MokoSuiteBackup
|
# MokoSuiteBackup
|
||||||
|
|
||||||
<!-- VERSION: 01.39.02 -->
|
<!-- VERSION: 01.30.00 -->
|
||||||
|
|
||||||
Full-site backup and restore for Joomla — database, files, and configuration.
|
Full-site backup and restore for Joomla — database, files, and configuration.
|
||||||
|
|
||||||
|
|||||||
@@ -118,6 +118,27 @@
|
|||||||
/>
|
/>
|
||||||
</fieldset>
|
</fieldset>
|
||||||
|
|
||||||
|
<fieldset name="snapshot_cleanup" label="COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_RETENTION">
|
||||||
|
<field
|
||||||
|
name="snapshot_retention_count"
|
||||||
|
type="number"
|
||||||
|
label="COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_MAX_COUNT"
|
||||||
|
description="COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_MAX_COUNT_DESC"
|
||||||
|
default="20"
|
||||||
|
min="0"
|
||||||
|
max="100"
|
||||||
|
/>
|
||||||
|
<field
|
||||||
|
name="snapshot_retention_days"
|
||||||
|
type="number"
|
||||||
|
label="COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_MAX_AGE"
|
||||||
|
description="COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_MAX_AGE_DESC"
|
||||||
|
default="30"
|
||||||
|
min="0"
|
||||||
|
max="365"
|
||||||
|
/>
|
||||||
|
</fieldset>
|
||||||
|
|
||||||
<fieldset name="notifications" label="COM_MOKOJOOMBACKUP_CONFIG_NOTIFICATIONS">
|
<fieldset name="notifications" label="COM_MOKOJOOMBACKUP_CONFIG_NOTIFICATIONS">
|
||||||
<field
|
<field
|
||||||
name="notify_email"
|
name="notify_email"
|
||||||
|
|||||||
@@ -269,6 +269,13 @@ COM_MOKOJOOMBACKUP_CONFIG_NOTIFY_SUCCESS_DESC="Send email when any backup comple
|
|||||||
COM_MOKOJOOMBACKUP_CONFIG_NOTIFY_FAILURE="Notify on Failure"
|
COM_MOKOJOOMBACKUP_CONFIG_NOTIFY_FAILURE="Notify on Failure"
|
||||||
COM_MOKOJOOMBACKUP_CONFIG_NOTIFY_FAILURE_DESC="Send email when any backup fails (unless overridden by profile)."
|
COM_MOKOJOOMBACKUP_CONFIG_NOTIFY_FAILURE_DESC="Send email when any backup fails (unless overridden by profile)."
|
||||||
|
|
||||||
|
; Snapshot Retention
|
||||||
|
COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_RETENTION="Snapshot Retention"
|
||||||
|
COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_MAX_COUNT="Max Snapshot Count"
|
||||||
|
COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_MAX_COUNT_DESC="Maximum number of content snapshots to keep. Oldest are removed first. Set to 0 for unlimited."
|
||||||
|
COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_MAX_AGE="Max Snapshot Age (days)"
|
||||||
|
COM_MOKOJOOMBACKUP_CONFIG_SNAPSHOT_MAX_AGE_DESC="Delete snapshots older than this many days. Set to 0 for unlimited."
|
||||||
|
|
||||||
; Web Cron
|
; Web Cron
|
||||||
COM_MOKOJOOMBACKUP_CONFIG_WEBCRON="Web Cron"
|
COM_MOKOJOOMBACKUP_CONFIG_WEBCRON="Web Cron"
|
||||||
COM_MOKOJOOMBACKUP_CONFIG_WEBCRON_ENABLED="Enable Web Cron"
|
COM_MOKOJOOMBACKUP_CONFIG_WEBCRON_ENABLED="Enable Web Cron"
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="component" method="upgrade">
|
<extension type="component" method="upgrade">
|
||||||
<name>MokoSuiteBackup</name>
|
<name>MokoSuiteBackup</name>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-02</creationDate>
|
<creationDate>2026-06-02</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -255,26 +255,36 @@ class BackupEngine
|
|||||||
}
|
}
|
||||||
|
|
||||||
$remoteFilename = '';
|
$remoteFilename = '';
|
||||||
|
$uploadFailed = false;
|
||||||
|
|
||||||
// Step 3: Remote upload (if configured)
|
// Step 3: Remote upload (if configured)
|
||||||
|
// Wrapped in its own try-catch so a remote failure does not mark
|
||||||
|
// the entire backup as failed — the local archive is preserved.
|
||||||
$remoteStorage = $profile->remote_storage ?? 'none';
|
$remoteStorage = $profile->remote_storage ?? 'none';
|
||||||
|
|
||||||
if ($remoteStorage !== 'none') {
|
if ($remoteStorage !== 'none') {
|
||||||
$this->log('Starting remote upload (' . $remoteStorage . ')...');
|
try {
|
||||||
$uploader = $this->createUploader($remoteStorage, $profile);
|
$this->log('Starting remote upload (' . $remoteStorage . ')...');
|
||||||
$uploadResult = $uploader->upload($archivePath, $archiveName);
|
$uploader = $this->createUploader($remoteStorage, $profile);
|
||||||
|
$uploadResult = $uploader->upload($archivePath, $archiveName);
|
||||||
|
|
||||||
if ($uploadResult['success']) {
|
if ($uploadResult['success']) {
|
||||||
$remoteFilename = $uploadResult['remote_path'] ?? $archiveName;
|
$remoteFilename = $uploadResult['remote_path'] ?? $archiveName;
|
||||||
$this->log('Remote upload complete: ' . $uploadResult['message']);
|
$this->log('Remote upload complete: ' . $uploadResult['message']);
|
||||||
|
|
||||||
// Delete local copy if configured
|
// Delete local copy if configured
|
||||||
if (empty($profile->remote_keep_local) && is_file($archivePath)) {
|
if (empty($profile->remote_keep_local) && is_file($archivePath)) {
|
||||||
@unlink($archivePath);
|
@unlink($archivePath);
|
||||||
$this->log('Local copy removed (remote_keep_local = off)');
|
$this->log('Local copy removed (remote_keep_local = off)');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$uploadFailed = true;
|
||||||
|
$this->log('WARNING: Remote upload failed: ' . $uploadResult['message']);
|
||||||
|
$this->log('Local backup is preserved.');
|
||||||
}
|
}
|
||||||
} else {
|
} catch (\Throwable $e) {
|
||||||
$this->log('WARNING: Remote upload failed: ' . $uploadResult['message']);
|
$uploadFailed = true;
|
||||||
|
$this->log('WARNING: Remote upload threw an exception: ' . $e->getMessage());
|
||||||
$this->log('Local backup is preserved.');
|
$this->log('Local backup is preserved.');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -309,9 +319,14 @@ class BackupEngine
|
|||||||
|
|
||||||
$db->updateObject('#__mokosuitebackup_records', $update, 'id');
|
$db->updateObject('#__mokosuitebackup_records', $update, 'id');
|
||||||
|
|
||||||
// Send success notification
|
// Send success notification (backup completed, even if upload failed)
|
||||||
NotificationSender::send($profile, $update, true, implode("\n", $this->log));
|
NotificationSender::send($profile, $update, true, implode("\n", $this->log));
|
||||||
|
|
||||||
|
// If remote upload failed, also send a failure notification for the upload
|
||||||
|
if ($uploadFailed) {
|
||||||
|
NotificationSender::send($profile, $update, false, "Remote upload failed — see backup log for details.\n\n" . implode("\n", $this->log));
|
||||||
|
}
|
||||||
|
|
||||||
// Dispatch event for actionlog and other listeners
|
// Dispatch event for actionlog and other listeners
|
||||||
$this->dispatchAfterRun(true, $recordId, $description, $profileId, $origin);
|
$this->dispatchAfterRun(true, $recordId, $description, $profileId, $origin);
|
||||||
|
|
||||||
|
|||||||
@@ -41,6 +41,10 @@ class SnapshotEngine
|
|||||||
private const ARTICLE_RELATED = [
|
private const ARTICLE_RELATED = [
|
||||||
'#__workflow_associations',
|
'#__workflow_associations',
|
||||||
'#__contentitem_tag_map',
|
'#__contentitem_tag_map',
|
||||||
|
'#__tags',
|
||||||
|
'#__fields',
|
||||||
|
'#__fields_values',
|
||||||
|
'#__fields_categories',
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -107,6 +111,32 @@ class SnapshotEngine
|
|||||||
$rows = $this->dumpTagMap($db, $prefix);
|
$rows = $this->dumpTagMap($db, $prefix);
|
||||||
$data['tables']['#__contentitem_tag_map'] = $rows;
|
$data['tables']['#__contentitem_tag_map'] = $rows;
|
||||||
$this->log(' #__contentitem_tag_map: ' . count($rows) . ' rows');
|
$this->log(' #__contentitem_tag_map: ' . count($rows) . ' rows');
|
||||||
|
|
||||||
|
// Tags — dump all (shared, small table)
|
||||||
|
$rows = $this->dumpTable($db, str_replace('#__', $prefix, '#__tags'), '#__tags', 'articles');
|
||||||
|
$data['tables']['#__tags'] = $rows;
|
||||||
|
$this->log(' #__tags: ' . count($rows) . ' rows');
|
||||||
|
|
||||||
|
// Custom fields — only com_content.article context
|
||||||
|
$rows = $this->dumpFilteredTable(
|
||||||
|
$db,
|
||||||
|
str_replace('#__', $prefix, '#__fields'),
|
||||||
|
'#__fields',
|
||||||
|
'context',
|
||||||
|
'com_content.article'
|
||||||
|
);
|
||||||
|
$data['tables']['#__fields'] = $rows;
|
||||||
|
$this->log(' #__fields: ' . count($rows) . ' rows');
|
||||||
|
|
||||||
|
// Field values — only for com_content.article fields (table is shared across extensions)
|
||||||
|
$rows = $this->dumpFieldValues($db, $prefix);
|
||||||
|
$data['tables']['#__fields_values'] = $rows;
|
||||||
|
$this->log(' #__fields_values: ' . count($rows) . ' rows');
|
||||||
|
|
||||||
|
// Field-category mappings — only for com_content.article fields
|
||||||
|
$rows = $this->dumpFieldCategories($db, $prefix);
|
||||||
|
$data['tables']['#__fields_categories'] = $rows;
|
||||||
|
$this->log(' #__fields_categories: ' . count($rows) . ' rows');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Count items
|
// Count items
|
||||||
@@ -231,6 +261,52 @@ class SnapshotEngine
|
|||||||
return $db->loadAssocList() ?: [];
|
return $db->loadAssocList() ?: [];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Dump field-category mappings for com_content.article fields.
|
||||||
|
*
|
||||||
|
* Uses a subquery: field_id IN (SELECT id FROM #__fields WHERE context = 'com_content.article')
|
||||||
|
*/
|
||||||
|
/**
|
||||||
|
* Dump field values only for com_content.article fields.
|
||||||
|
*/
|
||||||
|
private function dumpFieldValues(object $db, string $prefix): array
|
||||||
|
{
|
||||||
|
$fvTable = $prefix . 'fields_values';
|
||||||
|
$fTable = $prefix . 'fields';
|
||||||
|
|
||||||
|
$subQuery = $db->getQuery(true)
|
||||||
|
->select($db->quoteName('id'))
|
||||||
|
->from($db->quoteName($fTable))
|
||||||
|
->where($db->quoteName('context') . ' = ' . $db->quote('com_content.article'));
|
||||||
|
|
||||||
|
$query = $db->getQuery(true)
|
||||||
|
->select('*')
|
||||||
|
->from($db->quoteName($fvTable))
|
||||||
|
->where($db->quoteName('field_id') . ' IN (' . $subQuery . ')');
|
||||||
|
$db->setQuery($query);
|
||||||
|
|
||||||
|
return $db->loadAssocList() ?: [];
|
||||||
|
}
|
||||||
|
|
||||||
|
private function dumpFieldCategories(object $db, string $prefix): array
|
||||||
|
{
|
||||||
|
$fcTable = $prefix . 'fields_categories';
|
||||||
|
$fTable = $prefix . 'fields';
|
||||||
|
|
||||||
|
$subQuery = $db->getQuery(true)
|
||||||
|
->select($db->quoteName('id'))
|
||||||
|
->from($db->quoteName($fTable))
|
||||||
|
->where($db->quoteName('context') . ' = ' . $db->quote('com_content.article'));
|
||||||
|
|
||||||
|
$query = $db->getQuery(true)
|
||||||
|
->select('*')
|
||||||
|
->from($db->quoteName($fcTable))
|
||||||
|
->where($db->quoteName('field_id') . ' IN (' . $subQuery . ')');
|
||||||
|
$db->setQuery($query);
|
||||||
|
|
||||||
|
return $db->loadAssocList() ?: [];
|
||||||
|
}
|
||||||
|
|
||||||
private function log(string $message): void
|
private function log(string $message): void
|
||||||
{
|
{
|
||||||
$this->log[] = '[' . date('H:i:s') . '] ' . $message;
|
$this->log[] = '[' . date('H:i:s') . '] ' . $message;
|
||||||
|
|||||||
@@ -33,6 +33,10 @@ class SnapshotRestoreEngine
|
|||||||
'#__contentitem_tag_map' => null, // composite key, handled specially
|
'#__contentitem_tag_map' => null, // composite key, handled specially
|
||||||
'#__modules' => 'id',
|
'#__modules' => 'id',
|
||||||
'#__modules_menu' => null, // composite key, handled specially
|
'#__modules_menu' => null, // composite key, handled specially
|
||||||
|
'#__tags' => 'id',
|
||||||
|
'#__fields' => 'id',
|
||||||
|
'#__fields_values' => null, // composite key, handled specially
|
||||||
|
'#__fields_categories' => null, // composite key, handled specially
|
||||||
];
|
];
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -282,6 +286,48 @@ class SnapshotRestoreEngine
|
|||||||
$query->where($db->quoteName('moduleid') . ' IN (' . implode(',', $moduleIds) . ')');
|
$query->where($db->quoteName('moduleid') . ' IN (' . implode(',', $moduleIds) . ')');
|
||||||
break;
|
break;
|
||||||
|
|
||||||
|
case '#__tags':
|
||||||
|
// Only delete tags that exist in the snapshot — never wipe all tags
|
||||||
|
$ids = array_filter(array_column($rows, 'id'));
|
||||||
|
|
||||||
|
if (empty($ids)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$ids = array_map('intval', $ids);
|
||||||
|
$query->where($db->quoteName('id') . ' IN (' . implode(',', $ids) . ')');
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '#__fields':
|
||||||
|
// Only delete custom fields scoped to com_content.article
|
||||||
|
$query->where($db->quoteName('context') . ' = ' . $db->quote('com_content.article'));
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '#__fields_values':
|
||||||
|
// Only delete field values for com_content.article fields
|
||||||
|
$prefix = $db->getPrefix();
|
||||||
|
$fTable = $prefix . 'fields';
|
||||||
|
|
||||||
|
$subQuery = $db->getQuery(true)
|
||||||
|
->select($db->quoteName('id'))
|
||||||
|
->from($db->quoteName($fTable))
|
||||||
|
->where($db->quoteName('context') . ' = ' . $db->quote('com_content.article'));
|
||||||
|
$query->where($db->quoteName('field_id') . ' IN (' . $subQuery . ')');
|
||||||
|
break;
|
||||||
|
|
||||||
|
case '#__fields_categories':
|
||||||
|
// Delete field-category mappings for com_content.article fields only
|
||||||
|
$prefix = $db->getPrefix();
|
||||||
|
$fTable = $prefix . 'fields';
|
||||||
|
|
||||||
|
$subQuery = $db->getQuery(true)
|
||||||
|
->select($db->quoteName('id'))
|
||||||
|
->from($db->quoteName($fTable))
|
||||||
|
->where($db->quoteName('context') . ' = ' . $db->quote('com_content.article'));
|
||||||
|
|
||||||
|
$query->where($db->quoteName('field_id') . ' IN (' . $subQuery . ')');
|
||||||
|
break;
|
||||||
|
|
||||||
// #__content and #__content_frontpage are fully owned by com_content
|
// #__content and #__content_frontpage are fully owned by com_content
|
||||||
default:
|
default:
|
||||||
break;
|
break;
|
||||||
@@ -303,6 +349,10 @@ class SnapshotRestoreEngine
|
|||||||
$tables[] = '#__content_frontpage';
|
$tables[] = '#__content_frontpage';
|
||||||
$tables[] = '#__workflow_associations';
|
$tables[] = '#__workflow_associations';
|
||||||
$tables[] = '#__contentitem_tag_map';
|
$tables[] = '#__contentitem_tag_map';
|
||||||
|
$tables[] = '#__tags';
|
||||||
|
$tables[] = '#__fields';
|
||||||
|
$tables[] = '#__fields_values';
|
||||||
|
$tables[] = '#__fields_categories';
|
||||||
}
|
}
|
||||||
|
|
||||||
if (in_array('categories', $types)) {
|
if (in_array('categories', $types)) {
|
||||||
|
|||||||
@@ -389,37 +389,47 @@ class SteppedBackupEngine
|
|||||||
private function stepUpload(SteppedSession $session): void
|
private function stepUpload(SteppedSession $session): void
|
||||||
{
|
{
|
||||||
$db = Factory::getDbo();
|
$db = Factory::getDbo();
|
||||||
|
|
||||||
// Reload profile for remote settings
|
|
||||||
$query = $db->getQuery(true)
|
|
||||||
->select('*')
|
|
||||||
->from($db->quoteName('#__mokosuitebackup_profiles'))
|
|
||||||
->where($db->quoteName('id') . ' = ' . $session->profileId);
|
|
||||||
$db->setQuery($query);
|
|
||||||
$profile = $db->loadObject();
|
|
||||||
|
|
||||||
$uploader = match ($session->remoteStorage) {
|
|
||||||
'ftp' => new FtpUploader($profile),
|
|
||||||
'google_drive' => new GoogleDriveUploader($profile),
|
|
||||||
's3' => new S3Uploader($profile),
|
|
||||||
default => throw new \InvalidArgumentException('Unknown storage: ' . $session->remoteStorage),
|
|
||||||
};
|
|
||||||
|
|
||||||
$session->log('Starting remote upload (' . $session->remoteStorage . ')...');
|
|
||||||
$result = $uploader->upload($session->archivePath, $session->archiveName);
|
|
||||||
|
|
||||||
$remoteFilename = '';
|
$remoteFilename = '';
|
||||||
|
$uploadFailed = false;
|
||||||
|
|
||||||
if ($result['success']) {
|
// Wrapped in its own try-catch so a remote failure does not mark
|
||||||
$remoteFilename = $result['remote_path'] ?? $session->archiveName;
|
// the entire backup as failed — the local archive is preserved.
|
||||||
$session->log('Remote upload complete: ' . $result['message']);
|
try {
|
||||||
|
// Reload profile for remote settings
|
||||||
|
$query = $db->getQuery(true)
|
||||||
|
->select('*')
|
||||||
|
->from($db->quoteName('#__mokosuitebackup_profiles'))
|
||||||
|
->where($db->quoteName('id') . ' = ' . $session->profileId);
|
||||||
|
$db->setQuery($query);
|
||||||
|
$profile = $db->loadObject();
|
||||||
|
|
||||||
if (!$session->remoteKeepLocal && is_file($session->archivePath)) {
|
$uploader = match ($session->remoteStorage) {
|
||||||
@unlink($session->archivePath);
|
'ftp' => new FtpUploader($profile),
|
||||||
$session->log('Local copy removed');
|
'google_drive' => new GoogleDriveUploader($profile),
|
||||||
|
's3' => new S3Uploader($profile),
|
||||||
|
default => throw new \InvalidArgumentException('Unknown storage: ' . $session->remoteStorage),
|
||||||
|
};
|
||||||
|
|
||||||
|
$session->log('Starting remote upload (' . $session->remoteStorage . ')...');
|
||||||
|
$result = $uploader->upload($session->archivePath, $session->archiveName);
|
||||||
|
|
||||||
|
if ($result['success']) {
|
||||||
|
$remoteFilename = $result['remote_path'] ?? $session->archiveName;
|
||||||
|
$session->log('Remote upload complete: ' . $result['message']);
|
||||||
|
|
||||||
|
if (!$session->remoteKeepLocal && is_file($session->archivePath)) {
|
||||||
|
@unlink($session->archivePath);
|
||||||
|
$session->log('Local copy removed');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$uploadFailed = true;
|
||||||
|
$session->log('WARNING: Remote upload failed: ' . $result['message']);
|
||||||
|
$session->log('Local backup is preserved.');
|
||||||
}
|
}
|
||||||
} else {
|
} catch (\Throwable $e) {
|
||||||
$session->log('WARNING: Remote upload failed: ' . $result['message']);
|
$uploadFailed = true;
|
||||||
|
$session->log('WARNING: Remote upload threw an exception: ' . $e->getMessage());
|
||||||
|
$session->log('Local backup is preserved.');
|
||||||
}
|
}
|
||||||
|
|
||||||
// Update record with remote filename
|
// Update record with remote filename
|
||||||
@@ -433,14 +443,16 @@ class SteppedBackupEngine
|
|||||||
|
|
||||||
$session->currentStep++;
|
$session->currentStep++;
|
||||||
$session->phase = 'complete';
|
$session->phase = 'complete';
|
||||||
$session->statusMessage = 'Backup complete';
|
$session->statusMessage = $uploadFailed
|
||||||
$this->completeRecord($session);
|
? 'Backup complete (remote upload failed — local archive preserved)'
|
||||||
|
: 'Backup complete';
|
||||||
|
$this->completeRecord($session, $uploadFailed);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Mark the backup record as complete.
|
* Mark the backup record as complete.
|
||||||
*/
|
*/
|
||||||
private function completeRecord(SteppedSession $session): void
|
private function completeRecord(SteppedSession $session, bool $uploadFailed = false): void
|
||||||
{
|
{
|
||||||
$db = Factory::getDbo();
|
$db = Factory::getDbo();
|
||||||
$logContent = implode("\n", $session->log);
|
$logContent = implode("\n", $session->log);
|
||||||
@@ -490,6 +502,11 @@ class SteppedBackupEngine
|
|||||||
];
|
];
|
||||||
|
|
||||||
NotificationSender::send($profile, $record, true, $logContent);
|
NotificationSender::send($profile, $record, true, $logContent);
|
||||||
|
|
||||||
|
// If remote upload failed, also send a failure notification for the upload
|
||||||
|
if ($uploadFailed) {
|
||||||
|
NotificationSender::send($profile, $record, false, "Remote upload failed — see backup log for details.\n\n" . $logContent);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
} catch (\Throwable $e) {
|
} catch (\Throwable $e) {
|
||||||
error_log('MokoSuiteBackup: SteppedBackupEngine notification failed: ' . $e->getMessage());
|
error_log('MokoSuiteBackup: SteppedBackupEngine notification failed: ' . $e->getMessage());
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="plugin" group="actionlog" method="upgrade">
|
<extension type="plugin" group="actionlog" method="upgrade">
|
||||||
<name>Action Log - MokoSuiteBackup</name>
|
<name>Action Log - MokoSuiteBackup</name>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-04</creationDate>
|
<creationDate>2026-06-04</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="plugin" group="console" method="upgrade">
|
<extension type="plugin" group="console" method="upgrade">
|
||||||
<name>Console - MokoSuiteBackup</name>
|
<name>Console - MokoSuiteBackup</name>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-04</creationDate>
|
<creationDate>2026-06-04</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="plugin" group="content" method="upgrade">
|
<extension type="plugin" group="content" method="upgrade">
|
||||||
<name>Content - MokoSuiteBackup</name>
|
<name>Content - MokoSuiteBackup</name>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-04</creationDate>
|
<creationDate>2026-06-04</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
<?xml version="1.0" encoding="UTF-8"?>
|
<?xml version="1.0" encoding="UTF-8"?>
|
||||||
<extension type="plugin" group="quickicon" method="upgrade">
|
<extension type="plugin" group="quickicon" method="upgrade">
|
||||||
<name>Quick Icon - MokoSuiteBackup</name>
|
<name>Quick Icon - MokoSuiteBackup</name>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-02</creationDate>
|
<creationDate>2026-06-02</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="plugin" group="system" method="upgrade">
|
<extension type="plugin" group="system" method="upgrade">
|
||||||
<name>System - MokoSuiteBackup</name>
|
<name>System - MokoSuiteBackup</name>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-02</creationDate>
|
<creationDate>2026-06-02</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -136,6 +136,7 @@ final class MokoSuiteBackup extends CMSPlugin implements SubscriberInterface
|
|||||||
$session->set('mokosuitebackup.last_cleanup', time());
|
$session->set('mokosuitebackup.last_cleanup', time());
|
||||||
|
|
||||||
$this->cleanupOldBackups();
|
$this->cleanupOldBackups();
|
||||||
|
$this->cleanupOldSnapshots();
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -152,6 +153,93 @@ final class MokoSuiteBackup extends CMSPlugin implements SubscriberInterface
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Remove old content snapshots per component retention settings.
|
||||||
|
*
|
||||||
|
* Respects snapshot_retention_days (max age) and snapshot_retention_count
|
||||||
|
* (max number to keep). A value of 0 means unlimited for that setting.
|
||||||
|
*/
|
||||||
|
private function cleanupOldSnapshots(): void
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
$this->doSnapshotCleanup();
|
||||||
|
} catch (\Throwable $e) {
|
||||||
|
error_log('MokoSuiteBackup: cleanupOldSnapshots() failed: ' . $e->getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
private function doSnapshotCleanup(): void
|
||||||
|
{
|
||||||
|
$db = Factory::getDbo();
|
||||||
|
$params = ComponentHelper::getParams('com_mokosuitebackup');
|
||||||
|
$retentionDays = (int) $params->get('snapshot_retention_days', 30);
|
||||||
|
$retentionCount = (int) $params->get('snapshot_retention_count', 20);
|
||||||
|
|
||||||
|
// Delete snapshots older than retention_days
|
||||||
|
if ($retentionDays > 0) {
|
||||||
|
$cutoff = date('Y-m-d H:i:s', strtotime("-{$retentionDays} days"));
|
||||||
|
$query = $db->getQuery(true)
|
||||||
|
->select([$db->quoteName('id'), $db->quoteName('data_file')])
|
||||||
|
->from($db->quoteName('#__mokosuitebackup_snapshots'))
|
||||||
|
->where($db->quoteName('created') . ' < ' . $db->quote($cutoff))
|
||||||
|
->order($db->quoteName('created') . ' DESC');
|
||||||
|
$db->setQuery($query);
|
||||||
|
$expired = $db->loadObjectList();
|
||||||
|
|
||||||
|
foreach ($expired as $snapshot) {
|
||||||
|
$this->deleteSnapshotRecord($db, $snapshot);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Enforce max count (keep newest)
|
||||||
|
if ($retentionCount > 0) {
|
||||||
|
$query = $db->getQuery(true)
|
||||||
|
->select('COUNT(*)')
|
||||||
|
->from($db->quoteName('#__mokosuitebackup_snapshots'));
|
||||||
|
$db->setQuery($query);
|
||||||
|
$totalCount = (int) $db->loadResult();
|
||||||
|
|
||||||
|
if ($totalCount > $retentionCount) {
|
||||||
|
$excess = $totalCount - $retentionCount;
|
||||||
|
$query = $db->getQuery(true)
|
||||||
|
->select([$db->quoteName('id'), $db->quoteName('data_file')])
|
||||||
|
->from($db->quoteName('#__mokosuitebackup_snapshots'))
|
||||||
|
->order($db->quoteName('created') . ' ASC');
|
||||||
|
$db->setQuery($query, 0, $excess);
|
||||||
|
$oldest = $db->loadObjectList();
|
||||||
|
|
||||||
|
foreach ($oldest as $snapshot) {
|
||||||
|
$this->deleteSnapshotRecord($db, $snapshot);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete a snapshot record and its JSON data file.
|
||||||
|
*/
|
||||||
|
private function deleteSnapshotRecord(object $db, object $snapshot): void
|
||||||
|
{
|
||||||
|
if (!empty($snapshot->data_file) && is_file($snapshot->data_file)) {
|
||||||
|
if (!@unlink($snapshot->data_file)) {
|
||||||
|
error_log('MokoSuiteBackup: Could not delete snapshot file (id=' . $snapshot->id . '): ' . $snapshot->data_file);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
$db->setQuery(
|
||||||
|
$db->getQuery(true)
|
||||||
|
->delete($db->quoteName('#__mokosuitebackup_snapshots'))
|
||||||
|
->where($db->quoteName('id') . ' = ' . (int) $snapshot->id)
|
||||||
|
);
|
||||||
|
$db->execute();
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
error_log('MokoSuiteBackup: Could not delete snapshot record ' . $snapshot->id . ': ' . $e->getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
private function doCleanup(): void
|
private function doCleanup(): void
|
||||||
{
|
{
|
||||||
$db = Factory::getDbo();
|
$db = Factory::getDbo();
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="plugin" group="task" method="upgrade">
|
<extension type="plugin" group="task" method="upgrade">
|
||||||
<name>Task - MokoSuiteBackup</name>
|
<name>Task - MokoSuiteBackup</name>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-02</creationDate>
|
<creationDate>2026-06-02</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="plugin" group="webservices" method="upgrade">
|
<extension type="plugin" group="webservices" method="upgrade">
|
||||||
<name>Web Services - MokoSuiteBackup</name>
|
<name>Web Services - MokoSuiteBackup</name>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-02</creationDate>
|
<creationDate>2026-06-02</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
<extension type="package" method="upgrade">
|
<extension type="package" method="upgrade">
|
||||||
<name>Package - MokoSuiteBackup</name>
|
<name>Package - MokoSuiteBackup</name>
|
||||||
<packagename>mokosuitebackup</packagename>
|
<packagename>mokosuitebackup</packagename>
|
||||||
<version>01.39.02</version>
|
<version>01.30.00</version>
|
||||||
<creationDate>2026-06-02</creationDate>
|
<creationDate>2026-06-02</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
Reference in New Issue
Block a user