Compare commits
2 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| f560b0c63e | |||
| 430b25cea5 |
@@ -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:
|
||||||
|
MOKOGITEA_TOKEN: ${{ secrets.MOKOGITEA_TOKEN || github.token }}
|
||||||
|
MOKO_CLONE_TOKEN: ${{ secrets.MOKOGITEA_TOKEN || github.token }}
|
||||||
|
MOKO_CLONE_HOST: ${{ secrets.MOKOGITEA_TOKEN && 'git.mokoconsulting.tech/MokoConsulting' || 'github.com/mokoconsulting-tech' }}
|
||||||
|
COMPOSER_AUTH: '{"github-oauth":{"github.com":"${{ secrets.MOKOGITEA_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: 02.52.24
|
# VERSION: 02.52.19
|
||||||
# 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
-19
@@ -1,28 +1,13 @@
|
|||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
## [02.52.22] --- 2026-06-30
|
## [02.52.18] --- 2026-06-30
|
||||||
|
|
||||||
### Added
|
|
||||||
- Cancel Stalled toolbar button on Backup Records view to cancel backups stuck in "running" status
|
|
||||||
- New ACL permission `mokosuitebackup.backup.cancel` for cancel stalled action
|
|
||||||
- AJAX endpoint `ajax.cancelBackup` for programmatic/API cancel
|
|
||||||
- Auto-timeout failsafe: preflight auto-cancels "running" backups older than 30 minutes
|
|
||||||
- Pre-extension-update backup progress modal (Bootstrap 5 modal with stepped AJAX progress bar)
|
|
||||||
- New `warning` backup status for records where archive succeeded but remote upload failed
|
|
||||||
- Warning-status records are downloadable, browsable, restorable, and purgeable
|
|
||||||
- Warning status filter option in Backup Records dropdown
|
|
||||||
- Yellow "Warning" badge in backup list, detail view, and cpanel module
|
|
||||||
|
|
||||||
### Fixed
|
|
||||||
- Pre-update backup ran synchronously with no browser feedback — page hung until complete
|
|
||||||
- Stalled backups permanently blocked future backups for the same profile
|
|
||||||
- Preflight error message now directs users to Cancel Stalled action
|
|
||||||
- Backups with failed remote uploads were marked as "complete", hiding the upload failure
|
|
||||||
|
|
||||||
## [02.52.18] --- 2026-06-30
|
## [02.52.18] --- 2026-06-30
|
||||||
|
|
||||||
|
## [01.45.00] --- 2026-06-28
|
||||||
|
|
||||||
|
|
||||||
## [01.45.00] --- 2026-06-28
|
## [01.45.00] --- 2026-06-28
|
||||||
|
|
||||||
## [01.43.35] --- 2026-06-28
|
## [01.43.35] --- 2026-06-28
|
||||||
|
|||||||
+1
-1
@@ -23,7 +23,7 @@ DEFGROUP: Template-Joomla
|
|||||||
INGROUP: Template-Joomla.Documentation
|
INGROUP: Template-Joomla.Documentation
|
||||||
REPO: https://git.mokoconsulting.tech/MokoConsulting/Template-Joomla
|
REPO: https://git.mokoconsulting.tech/MokoConsulting/Template-Joomla
|
||||||
PATH: /SECURITY.md
|
PATH: /SECURITY.md
|
||||||
VERSION: 02.52.24
|
VERSION: 02.52.19
|
||||||
BRIEF: Security vulnerability reporting and handling policy
|
BRIEF: Security vulnerability reporting and handling policy
|
||||||
-->
|
-->
|
||||||
|
|
||||||
|
|||||||
Submodule source/packages/MokoSuiteClient updated: c7e6670544...0a9125e519
@@ -15,6 +15,5 @@
|
|||||||
<action name="mokosuitebackup.backup.purge" title="COM_MOKOSUITEBACKUP_ACTION_BACKUP_PURGE" />
|
<action name="mokosuitebackup.backup.purge" title="COM_MOKOSUITEBACKUP_ACTION_BACKUP_PURGE" />
|
||||||
<action name="mokosuitebackup.backup.compare" title="COM_MOKOSUITEBACKUP_ACTION_BACKUP_COMPARE" />
|
<action name="mokosuitebackup.backup.compare" title="COM_MOKOSUITEBACKUP_ACTION_BACKUP_COMPARE" />
|
||||||
<action name="mokosuitebackup.backup.browse" title="COM_MOKOSUITEBACKUP_ACTION_BACKUP_BROWSE" />
|
<action name="mokosuitebackup.backup.browse" title="COM_MOKOSUITEBACKUP_ACTION_BACKUP_BROWSE" />
|
||||||
<action name="mokosuitebackup.backup.cancel" title="COM_MOKOSUITEBACKUP_ACTION_BACKUP_CANCEL" />
|
|
||||||
</section>
|
</section>
|
||||||
</access>
|
</access>
|
||||||
|
|||||||
@@ -15,7 +15,6 @@
|
|||||||
>
|
>
|
||||||
<option value="">COM_MOKOJOOMBACKUP_FILTER_STATUS_ALL</option>
|
<option value="">COM_MOKOJOOMBACKUP_FILTER_STATUS_ALL</option>
|
||||||
<option value="complete">COM_MOKOJOOMBACKUP_STATUS_COMPLETE</option>
|
<option value="complete">COM_MOKOJOOMBACKUP_STATUS_COMPLETE</option>
|
||||||
<option value="warning">COM_MOKOJOOMBACKUP_STATUS_WARNING</option>
|
|
||||||
<option value="running">COM_MOKOJOOMBACKUP_STATUS_RUNNING</option>
|
<option value="running">COM_MOKOJOOMBACKUP_STATUS_RUNNING</option>
|
||||||
<option value="fail">COM_MOKOJOOMBACKUP_STATUS_FAIL</option>
|
<option value="fail">COM_MOKOJOOMBACKUP_STATUS_FAIL</option>
|
||||||
<option value="pending">COM_MOKOJOOMBACKUP_STATUS_PENDING</option>
|
<option value="pending">COM_MOKOJOOMBACKUP_STATUS_PENDING</option>
|
||||||
|
|||||||
@@ -207,7 +207,6 @@ COM_MOKOJOOMBACKUP_TYPE_DIFFERENTIAL="Differential (changed files + full DB)"
|
|||||||
|
|
||||||
; Status labels
|
; Status labels
|
||||||
COM_MOKOJOOMBACKUP_STATUS_COMPLETE="Complete"
|
COM_MOKOJOOMBACKUP_STATUS_COMPLETE="Complete"
|
||||||
COM_MOKOJOOMBACKUP_STATUS_WARNING="Warning"
|
|
||||||
COM_MOKOJOOMBACKUP_STATUS_RUNNING="Running"
|
COM_MOKOJOOMBACKUP_STATUS_RUNNING="Running"
|
||||||
COM_MOKOJOOMBACKUP_STATUS_FAIL="Failed"
|
COM_MOKOJOOMBACKUP_STATUS_FAIL="Failed"
|
||||||
COM_MOKOJOOMBACKUP_STATUS_PENDING="Pending"
|
COM_MOKOJOOMBACKUP_STATUS_PENDING="Pending"
|
||||||
@@ -451,8 +450,6 @@ COM_MOKOSUITEBACKUP_ACTION_BACKUP_COMPARE="Compare Backups"
|
|||||||
COM_MOKOSUITEBACKUP_ACTION_BACKUP_COMPARE_DESC="Allows users to compare two backup records side-by-side."
|
COM_MOKOSUITEBACKUP_ACTION_BACKUP_COMPARE_DESC="Allows users to compare two backup records side-by-side."
|
||||||
COM_MOKOSUITEBACKUP_ACTION_BACKUP_BROWSE="Browse Archives"
|
COM_MOKOSUITEBACKUP_ACTION_BACKUP_BROWSE="Browse Archives"
|
||||||
COM_MOKOSUITEBACKUP_ACTION_BACKUP_BROWSE_DESC="Allows users to view file listings inside backup archives without extracting."
|
COM_MOKOSUITEBACKUP_ACTION_BACKUP_BROWSE_DESC="Allows users to view file listings inside backup archives without extracting."
|
||||||
COM_MOKOSUITEBACKUP_ACTION_BACKUP_CANCEL="Cancel Stalled Backup"
|
|
||||||
COM_MOKOSUITEBACKUP_ACTION_BACKUP_CANCEL_DESC="Allows users to cancel backup records stuck in running status and clean up partial archive files."
|
|
||||||
|
|
||||||
; Snapshot ACL
|
; Snapshot ACL
|
||||||
COM_MOKOSUITEBACKUP_ACTION_SNAPSHOT_MANAGE="Manage Snapshots"
|
COM_MOKOSUITEBACKUP_ACTION_SNAPSHOT_MANAGE="Manage Snapshots"
|
||||||
@@ -503,12 +500,6 @@ COM_MOKOJOOMBACKUP_PURGE_INVALID_DATE="Invalid date. Please select a valid date.
|
|||||||
COM_MOKOJOOMBACKUP_PURGE_SUCCESS="%d backup(s) purged successfully."
|
COM_MOKOJOOMBACKUP_PURGE_SUCCESS="%d backup(s) purged successfully."
|
||||||
COM_MOKOJOOMBACKUP_PURGE_PARTIAL="%d backup(s) purged, but %d could not be deleted."
|
COM_MOKOJOOMBACKUP_PURGE_PARTIAL="%d backup(s) purged, but %d could not be deleted."
|
||||||
|
|
||||||
; Cancel Stalled Backup
|
|
||||||
COM_MOKOJOOMBACKUP_TOOLBAR_CANCEL_STALLED="Cancel Stalled"
|
|
||||||
COM_MOKOJOOMBACKUP_CANCEL_NONE_SELECTED="No backup records selected."
|
|
||||||
COM_MOKOJOOMBACKUP_CANCEL_NONE_RUNNING="None of the selected backups are in running status."
|
|
||||||
COM_MOKOJOOMBACKUP_CANCEL_SUCCESS="%d stalled backup(s) cancelled."
|
|
||||||
|
|
||||||
; Remote Destinations (multi-remote)
|
; Remote Destinations (multi-remote)
|
||||||
COM_MOKOJOOMBACKUP_REMOTE_DESTINATIONS="Remote Destinations"
|
COM_MOKOJOOMBACKUP_REMOTE_DESTINATIONS="Remote Destinations"
|
||||||
COM_MOKOJOOMBACKUP_REMOTE_ADD="Add Destination"
|
COM_MOKOJOOMBACKUP_REMOTE_ADD="Add Destination"
|
||||||
|
|||||||
@@ -116,16 +116,3 @@ COM_MOKOJOOMBACKUP_PURGE_NONE_FOUND="No completed backups found before the selec
|
|||||||
COM_MOKOJOOMBACKUP_PURGE_INVALID_DATE="Invalid date. Please select a valid date."
|
COM_MOKOJOOMBACKUP_PURGE_INVALID_DATE="Invalid date. Please select a valid date."
|
||||||
COM_MOKOJOOMBACKUP_PURGE_SUCCESS="%d backup(s) purged successfully."
|
COM_MOKOJOOMBACKUP_PURGE_SUCCESS="%d backup(s) purged successfully."
|
||||||
COM_MOKOJOOMBACKUP_PURGE_PARTIAL="%d backup(s) purged, but %d could not be deleted."
|
COM_MOKOJOOMBACKUP_PURGE_PARTIAL="%d backup(s) purged, but %d could not be deleted."
|
||||||
|
|
||||||
; Cancel Stalled Backup
|
|
||||||
COM_MOKOJOOMBACKUP_TOOLBAR_CANCEL_STALLED="Cancel Stalled"
|
|
||||||
COM_MOKOJOOMBACKUP_CANCEL_NONE_SELECTED="No backup records selected."
|
|
||||||
COM_MOKOJOOMBACKUP_CANCEL_NONE_RUNNING="None of the selected backups are in running status."
|
|
||||||
COM_MOKOJOOMBACKUP_CANCEL_SUCCESS="%d stalled backup(s) cancelled."
|
|
||||||
|
|
||||||
; Backup status
|
|
||||||
COM_MOKOJOOMBACKUP_STATUS_WARNING="Warning"
|
|
||||||
|
|
||||||
; ACL - Cancel
|
|
||||||
COM_MOKOSUITEBACKUP_ACTION_BACKUP_CANCEL="Cancel Stalled Backup"
|
|
||||||
COM_MOKOSUITEBACKUP_ACTION_BACKUP_CANCEL_DESC="Allows users to cancel backup records stuck in running status and clean up partial archive files."
|
|
||||||
|
|||||||
@@ -7,7 +7,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="component" method="upgrade">
|
<extension type="component" method="upgrade">
|
||||||
<name>MokoSuiteBackup</name>
|
<name>MokoSuiteBackup</name>
|
||||||
<version>02.52.24</version>
|
<version>02.52.19</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>
|
||||||
|
|||||||
@@ -65,7 +65,7 @@ CREATE TABLE IF NOT EXISTS `#__mokosuitebackup_records` (
|
|||||||
`id` INT(11) UNSIGNED NOT NULL AUTO_INCREMENT,
|
`id` INT(11) UNSIGNED NOT NULL AUTO_INCREMENT,
|
||||||
`profile_id` INT(11) UNSIGNED NOT NULL DEFAULT 1,
|
`profile_id` INT(11) UNSIGNED NOT NULL DEFAULT 1,
|
||||||
`description` VARCHAR(255) NOT NULL DEFAULT '',
|
`description` VARCHAR(255) NOT NULL DEFAULT '',
|
||||||
`status` VARCHAR(20) NOT NULL DEFAULT 'pending' COMMENT 'pending, running, complete, warning, fail',
|
`status` VARCHAR(20) NOT NULL DEFAULT 'pending' COMMENT 'pending, running, complete, fail',
|
||||||
`origin` VARCHAR(20) NOT NULL DEFAULT 'backend' COMMENT 'backend, cli, api, scheduled',
|
`origin` VARCHAR(20) NOT NULL DEFAULT 'backend' COMMENT 'backend, cli, api, scheduled',
|
||||||
`backup_type` VARCHAR(20) NOT NULL DEFAULT 'full' COMMENT 'full, database, files',
|
`backup_type` VARCHAR(20) NOT NULL DEFAULT 'full' COMMENT 'full, database, files',
|
||||||
`archivename` VARCHAR(512) NOT NULL DEFAULT '',
|
`archivename` VARCHAR(512) NOT NULL DEFAULT '',
|
||||||
|
|||||||
@@ -0,0 +1 @@
|
|||||||
|
/* 02.52.19 — no schema changes */
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 02.52.20 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 02.52.21 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 02.52.22 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 02.52.23 — no schema changes */
|
|
||||||
@@ -1 +0,0 @@
|
|||||||
/* 02.52.24 — no schema changes */
|
|
||||||
@@ -85,10 +85,11 @@ class AjaxController extends BaseController
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Cancel a backup record stuck in "running" status.
|
* Mark that the JS-driven pre-update backup has completed so the
|
||||||
* POST: task=ajax.cancelBackup&id=123
|
* server-side onExtensionBeforeUpdate handler skips its own run.
|
||||||
|
* POST: task=ajax.markPreUpdateDone
|
||||||
*/
|
*/
|
||||||
public function cancelBackup(): void
|
public function markPreUpdateDone(): void
|
||||||
{
|
{
|
||||||
if (!Session::checkToken('get') && !Session::checkToken('post')) {
|
if (!Session::checkToken('get') && !Session::checkToken('post')) {
|
||||||
$this->sendJson(['error' => true, 'message' => 'Invalid token'], 403);
|
$this->sendJson(['error' => true, 'message' => 'Invalid token'], 403);
|
||||||
@@ -96,53 +97,9 @@ class AjaxController extends BaseController
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!$this->app->getIdentity()->authorise('mokosuitebackup.backup.cancel', 'com_mokosuitebackup')) {
|
Factory::getSession()->set('mokosuitebackup.preupdate_js_done', true);
|
||||||
$this->sendJson(['error' => true, 'message' => 'Access denied'], 403);
|
|
||||||
|
|
||||||
return;
|
$this->sendJson(['success' => true]);
|
||||||
}
|
|
||||||
|
|
||||||
$id = $this->input->getInt('id', 0);
|
|
||||||
|
|
||||||
if (!$id) {
|
|
||||||
$this->sendJson(['error' => true, 'message' => 'Missing record ID']);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$db = Factory::getDbo();
|
|
||||||
$query = $db->getQuery(true)
|
|
||||||
->select($db->quoteName(['id', 'status', 'absolute_path']))
|
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
|
||||||
->where($db->quoteName('id') . ' = ' . $id);
|
|
||||||
$db->setQuery($query);
|
|
||||||
$record = $db->loadObject();
|
|
||||||
|
|
||||||
if (!$record) {
|
|
||||||
$this->sendJson(['error' => true, 'message' => 'Record not found'], 404);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($record->status !== 'running') {
|
|
||||||
$this->sendJson(['error' => true, 'message' => 'Backup is not in running status']);
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$update = $db->getQuery(true)
|
|
||||||
->update($db->quoteName('#__mokosuitebackup_records'))
|
|
||||||
->set($db->quoteName('status') . ' = ' . $db->quote('fail'))
|
|
||||||
->set($db->quoteName('backupend') . ' = ' . $db->quote(date('Y-m-d H:i:s')))
|
|
||||||
->where($db->quoteName('id') . ' = ' . $id);
|
|
||||||
$db->setQuery($update);
|
|
||||||
$db->execute();
|
|
||||||
|
|
||||||
if (!empty($record->absolute_path) && is_file($record->absolute_path)) {
|
|
||||||
@unlink($record->absolute_path);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->sendJson(['error' => false, 'message' => 'Backup cancelled']);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@@ -512,7 +469,7 @@ class AjaxController extends BaseController
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!\in_array($record->status, ['complete', 'warning'], true) || !$record->filesexist) {
|
if ($record->status !== 'complete' || !$record->filesexist) {
|
||||||
$this->sendJson(['error' => true, 'message' => 'Archive not available']);
|
$this->sendJson(['error' => true, 'message' => 'Archive not available']);
|
||||||
|
|
||||||
return;
|
return;
|
||||||
@@ -808,7 +765,7 @@ class AjaxController extends BaseController
|
|||||||
->select('COUNT(*)')
|
->select('COUNT(*)')
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
->from($db->quoteName('#__mokosuitebackup_records'))
|
||||||
->where($db->quoteName('backupstart') . ' < ' . $db->quote($cutoff))
|
->where($db->quoteName('backupstart') . ' < ' . $db->quote($cutoff))
|
||||||
->where($db->quoteName('status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning'])) . ')');
|
->where($db->quoteName('status') . ' = ' . $db->quote('complete'));
|
||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
$count = (int) $db->loadResult();
|
$count = (int) $db->loadResult();
|
||||||
} catch (\Exception $e) {
|
} catch (\Exception $e) {
|
||||||
|
|||||||
@@ -199,7 +199,7 @@ class BackupsController extends AdminController
|
|||||||
->select($db->quoteName('id'))
|
->select($db->quoteName('id'))
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
->from($db->quoteName('#__mokosuitebackup_records'))
|
||||||
->where($db->quoteName('backupstart') . ' < ' . $db->quote($cutoff))
|
->where($db->quoteName('backupstart') . ' < ' . $db->quote($cutoff))
|
||||||
->where($db->quoteName('status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning'])) . ')');
|
->where($db->quoteName('status') . ' = ' . $db->quote('complete'));
|
||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
$ids = $db->loadColumn();
|
$ids = $db->loadColumn();
|
||||||
|
|
||||||
@@ -235,76 +235,6 @@ class BackupsController extends AdminController
|
|||||||
$this->setRedirect(Route::_('index.php?option=com_mokosuitebackup&view=backups', false));
|
$this->setRedirect(Route::_('index.php?option=com_mokosuitebackup&view=backups', false));
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
|
||||||
* Cancel selected backup records that are stuck in "running" status.
|
|
||||||
*
|
|
||||||
* Sets their status to "fail", cleans up partial archive files,
|
|
||||||
* and destroys any associated stepped session.
|
|
||||||
*/
|
|
||||||
public function cancelStalled(): void
|
|
||||||
{
|
|
||||||
$this->checkToken();
|
|
||||||
|
|
||||||
if (!$this->app->getIdentity()->authorise('mokosuitebackup.backup.cancel', 'com_mokosuitebackup')) {
|
|
||||||
$this->setMessage(Text::_('JLIB_APPLICATION_ERROR_ACCESS_FORBIDDEN'), 'error');
|
|
||||||
$this->setRedirect(Route::_('index.php?option=com_mokosuitebackup&view=backups', false));
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$cid = $this->input->get('cid', [], 'array');
|
|
||||||
|
|
||||||
if (empty($cid)) {
|
|
||||||
$this->setMessage(Text::_('COM_MOKOJOOMBACKUP_CANCEL_NONE_SELECTED'), 'warning');
|
|
||||||
$this->setRedirect(Route::_('index.php?option=com_mokosuitebackup&view=backups', false));
|
|
||||||
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$db = $this->app->getContainer()->get('DatabaseDriver');
|
|
||||||
$cancelled = 0;
|
|
||||||
$skipped = 0;
|
|
||||||
|
|
||||||
foreach ($cid as $id) {
|
|
||||||
$id = (int) $id;
|
|
||||||
|
|
||||||
$query = $db->getQuery(true)
|
|
||||||
->select($db->quoteName(['id', 'status', 'absolute_path']))
|
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
|
||||||
->where($db->quoteName('id') . ' = ' . $id);
|
|
||||||
$db->setQuery($query);
|
|
||||||
$record = $db->loadObject();
|
|
||||||
|
|
||||||
if (!$record || $record->status !== 'running') {
|
|
||||||
$skipped++;
|
|
||||||
|
|
||||||
continue;
|
|
||||||
}
|
|
||||||
|
|
||||||
$update = $db->getQuery(true)
|
|
||||||
->update($db->quoteName('#__mokosuitebackup_records'))
|
|
||||||
->set($db->quoteName('status') . ' = ' . $db->quote('fail'))
|
|
||||||
->set($db->quoteName('backupend') . ' = ' . $db->quote(date('Y-m-d H:i:s')))
|
|
||||||
->where($db->quoteName('id') . ' = ' . $id);
|
|
||||||
$db->setQuery($update);
|
|
||||||
$db->execute();
|
|
||||||
|
|
||||||
if (!empty($record->absolute_path) && is_file($record->absolute_path)) {
|
|
||||||
@unlink($record->absolute_path);
|
|
||||||
}
|
|
||||||
|
|
||||||
$cancelled++;
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($cancelled > 0) {
|
|
||||||
$this->setMessage(Text::sprintf('COM_MOKOJOOMBACKUP_CANCEL_SUCCESS', $cancelled));
|
|
||||||
} elseif ($skipped > 0) {
|
|
||||||
$this->setMessage(Text::_('COM_MOKOJOOMBACKUP_CANCEL_NONE_RUNNING'), 'warning');
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->setRedirect(Route::_('index.php?option=com_mokosuitebackup&view=backups', false));
|
|
||||||
}
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* No-op target for the purge toolbar button.
|
* No-op target for the purge toolbar button.
|
||||||
*
|
*
|
||||||
|
|||||||
@@ -375,7 +375,7 @@ class BackupEngine
|
|||||||
// Final record update (includes fields needed by NotificationSender)
|
// Final record update (includes fields needed by NotificationSender)
|
||||||
$update = (object) [
|
$update = (object) [
|
||||||
'id' => $recordId,
|
'id' => $recordId,
|
||||||
'status' => $uploadFailed ? 'warning' : 'complete',
|
'status' => 'complete',
|
||||||
'description' => $description,
|
'description' => $description,
|
||||||
'backup_type' => $profile->backup_type,
|
'backup_type' => $profile->backup_type,
|
||||||
'archivename' => $archiveName,
|
'archivename' => $archiveName,
|
||||||
@@ -606,7 +606,7 @@ class BackupEngine
|
|||||||
->select($db->quoteName('manifest'))
|
->select($db->quoteName('manifest'))
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
->from($db->quoteName('#__mokosuitebackup_records'))
|
||||||
->where($db->quoteName('profile_id') . ' = ' . $profileId)
|
->where($db->quoteName('profile_id') . ' = ' . $profileId)
|
||||||
->where($db->quoteName('status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning'])) . ')')
|
->where($db->quoteName('status') . ' = ' . $db->quote('complete'))
|
||||||
->where($db->quoteName('manifest') . ' != ' . $db->quote(''))
|
->where($db->quoteName('manifest') . ' != ' . $db->quote(''))
|
||||||
->where($db->quoteName('backup_type') . ' = ' . $db->quote('full'))
|
->where($db->quoteName('backup_type') . ' = ' . $db->quote('full'))
|
||||||
->order($db->quoteName('backupstart') . ' DESC');
|
->order($db->quoteName('backupstart') . ' DESC');
|
||||||
|
|||||||
@@ -165,7 +165,7 @@ class PreflightCheck
|
|||||||
->select($db->quoteName('total_size'))
|
->select($db->quoteName('total_size'))
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
->from($db->quoteName('#__mokosuitebackup_records'))
|
||||||
->where($db->quoteName('profile_id') . ' = ' . (int) $profile->id)
|
->where($db->quoteName('profile_id') . ' = ' . (int) $profile->id)
|
||||||
->where($db->quoteName('status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning'])) . ')')
|
->where($db->quoteName('status') . ' = ' . $db->quote('complete'))
|
||||||
->where($db->quoteName('total_size') . ' > 0')
|
->where($db->quoteName('total_size') . ' > 0')
|
||||||
->order($db->quoteName('backupstart') . ' DESC');
|
->order($db->quoteName('backupstart') . ' DESC');
|
||||||
$db->setQuery($query, 0, 1);
|
$db->setQuery($query, 0, 1);
|
||||||
@@ -194,58 +194,22 @@ class PreflightCheck
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
private const STALE_TIMEOUT_MINUTES = 30;
|
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Check if another backup is already running for this profile.
|
* Check if another backup is already running for this profile.
|
||||||
*
|
|
||||||
* Backups running longer than STALE_TIMEOUT_MINUTES are automatically
|
|
||||||
* marked as failed so they don't permanently block future runs.
|
|
||||||
*/
|
*/
|
||||||
private function checkRunningBackup(object $profile, object $db): void
|
private function checkRunningBackup(object $profile, object $db): void
|
||||||
{
|
{
|
||||||
$query = $db->getQuery(true)
|
$query = $db->getQuery(true)
|
||||||
->select($db->quoteName(['id', 'backupstart', 'absolute_path']))
|
->select('COUNT(*)')
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
->from($db->quoteName('#__mokosuitebackup_records'))
|
||||||
->where($db->quoteName('profile_id') . ' = ' . (int) $profile->id)
|
->where($db->quoteName('profile_id') . ' = ' . (int) $profile->id)
|
||||||
->where($db->quoteName('status') . ' = ' . $db->quote('running'));
|
->where($db->quoteName('status') . ' = ' . $db->quote('running'));
|
||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
$rows = $db->loadObjectList();
|
$running = (int) $db->loadResult();
|
||||||
|
|
||||||
if (empty($rows)) {
|
if ($running > 0) {
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
$cutoff = time() - (self::STALE_TIMEOUT_MINUTES * 60);
|
|
||||||
$stillAlive = 0;
|
|
||||||
|
|
||||||
foreach ($rows as $row) {
|
|
||||||
$started = strtotime($row->backupstart);
|
|
||||||
|
|
||||||
if ($started !== false && $started < $cutoff) {
|
|
||||||
$update = $db->getQuery(true)
|
|
||||||
->update($db->quoteName('#__mokosuitebackup_records'))
|
|
||||||
->set($db->quoteName('status') . ' = ' . $db->quote('fail'))
|
|
||||||
->set($db->quoteName('backupend') . ' = ' . $db->quote(date('Y-m-d H:i:s')))
|
|
||||||
->where($db->quoteName('id') . ' = ' . (int) $row->id);
|
|
||||||
$db->setQuery($update);
|
|
||||||
$db->execute();
|
|
||||||
|
|
||||||
if (!empty($row->absolute_path) && is_file($row->absolute_path)) {
|
|
||||||
@unlink($row->absolute_path);
|
|
||||||
}
|
|
||||||
|
|
||||||
$this->warnings[] = 'Auto-cancelled stalled backup #' . $row->id
|
|
||||||
. ' (started ' . $row->backupstart . ', exceeded '
|
|
||||||
. self::STALE_TIMEOUT_MINUTES . ' min timeout)';
|
|
||||||
} else {
|
|
||||||
$stillAlive++;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($stillAlive > 0) {
|
|
||||||
$this->errors[] = 'Another backup is already running for profile: ' . $profile->title
|
$this->errors[] = 'Another backup is already running for profile: ' . $profile->title
|
||||||
. ' — wait for it to finish or use Cancel Stalled from the Backup Records toolbar';
|
. ' — wait for it to finish or delete the stale record';
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -67,7 +67,7 @@ class RestoreEngine
|
|||||||
return ['success' => false, 'message' => 'Backup record not found: ' . $recordId];
|
return ['success' => false, 'message' => 'Backup record not found: ' . $recordId];
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($record->status !== 'complete' && $record->status !== 'warning') {
|
if ($record->status !== 'complete') {
|
||||||
return ['success' => false, 'message' => 'Cannot restore from incomplete backup (status: ' . $record->status . ')'];
|
return ['success' => false, 'message' => 'Cannot restore from incomplete backup (status: ' . $record->status . ')'];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -647,7 +647,7 @@ class SteppedBackupEngine
|
|||||||
|
|
||||||
$update = (object) [
|
$update = (object) [
|
||||||
'id' => $session->recordId,
|
'id' => $session->recordId,
|
||||||
'status' => $uploadFailed ? 'warning' : 'complete',
|
'status' => 'complete',
|
||||||
'backupend' => date('Y-m-d H:i:s'),
|
'backupend' => date('Y-m-d H:i:s'),
|
||||||
'total_size' => $totalSize,
|
'total_size' => $totalSize,
|
||||||
'checksum' => $checksum,
|
'checksum' => $checksum,
|
||||||
|
|||||||
@@ -64,7 +64,7 @@ class SteppedRestoreEngine
|
|||||||
return ['error' => true, 'message' => 'Backup record not found: ' . $recordId];
|
return ['error' => true, 'message' => 'Backup record not found: ' . $recordId];
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($record->status !== 'complete' && $record->status !== 'warning') {
|
if ($record->status !== 'complete') {
|
||||||
return ['error' => true, 'message' => 'Cannot restore from incomplete backup (status: ' . $record->status . ')'];
|
return ['error' => true, 'message' => 'Cannot restore from incomplete backup (status: ' . $record->status . ')'];
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|||||||
@@ -70,7 +70,7 @@ class BackupStatusHelper
|
|||||||
])
|
])
|
||||||
->from($db->quoteName('#__mokosuitebackup_records', 'r'))
|
->from($db->quoteName('#__mokosuitebackup_records', 'r'))
|
||||||
->join('LEFT', $db->quoteName('#__mokosuitebackup_profiles', 'p') . ' ON p.id = r.profile_id')
|
->join('LEFT', $db->quoteName('#__mokosuitebackup_profiles', 'p') . ' ON p.id = r.profile_id')
|
||||||
->where($db->quoteName('r.status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning', 'fail'])) . ')')
|
->where($db->quoteName('r.status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'fail'])) . ')')
|
||||||
->order($db->quoteName('r.backupstart') . ' DESC');
|
->order($db->quoteName('r.backupstart') . ' DESC');
|
||||||
|
|
||||||
if ($profileId !== null) {
|
if ($profileId !== null) {
|
||||||
@@ -148,7 +148,7 @@ class BackupStatusHelper
|
|||||||
$query = $db->getQuery(true)
|
$query = $db->getQuery(true)
|
||||||
->select($db->quoteName('status'))
|
->select($db->quoteName('status'))
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
->from($db->quoteName('#__mokosuitebackup_records'))
|
||||||
->where($db->quoteName('status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning', 'fail'])) . ')')
|
->where($db->quoteName('status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'fail'])) . ')')
|
||||||
->order($db->quoteName('backupstart') . ' DESC')
|
->order($db->quoteName('backupstart') . ' DESC')
|
||||||
->setLimit(50);
|
->setLimit(50);
|
||||||
|
|
||||||
@@ -156,7 +156,7 @@ class BackupStatusHelper
|
|||||||
$streak = 0;
|
$streak = 0;
|
||||||
|
|
||||||
foreach ($statuses as $s) {
|
foreach ($statuses as $s) {
|
||||||
if ($s === 'complete' || $s === 'warning') {
|
if ($s === 'complete') {
|
||||||
$streak++;
|
$streak++;
|
||||||
} else {
|
} else {
|
||||||
break;
|
break;
|
||||||
|
|||||||
@@ -30,7 +30,7 @@ class DashboardModel extends BaseDatabaseModel
|
|||||||
->select('r.*, p.title AS profile_title')
|
->select('r.*, p.title AS profile_title')
|
||||||
->from($db->quoteName('#__mokosuitebackup_records', 'r'))
|
->from($db->quoteName('#__mokosuitebackup_records', 'r'))
|
||||||
->join('LEFT', $db->quoteName('#__mokosuitebackup_profiles', 'p') . ' ON p.id = r.profile_id')
|
->join('LEFT', $db->quoteName('#__mokosuitebackup_profiles', 'p') . ' ON p.id = r.profile_id')
|
||||||
->where($db->quoteName('r.status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning'])) . ')')
|
->where($db->quoteName('r.status') . ' = ' . $db->quote('complete'))
|
||||||
->order($db->quoteName('r.backupend') . ' DESC');
|
->order($db->quoteName('r.backupend') . ' DESC');
|
||||||
$db->setQuery($query, 0, 1);
|
$db->setQuery($query, 0, 1);
|
||||||
|
|
||||||
@@ -75,7 +75,7 @@ class DashboardModel extends BaseDatabaseModel
|
|||||||
->select('COUNT(*) AS total_count')
|
->select('COUNT(*) AS total_count')
|
||||||
->select('COALESCE(SUM(' . $db->quoteName('total_size') . '), 0) AS total_size')
|
->select('COALESCE(SUM(' . $db->quoteName('total_size') . '), 0) AS total_size')
|
||||||
->from($db->quoteName('#__mokosuitebackup_records'))
|
->from($db->quoteName('#__mokosuitebackup_records'))
|
||||||
->where($db->quoteName('status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning'])) . ')');
|
->where($db->quoteName('status') . ' = ' . $db->quote('complete'));
|
||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
$stats = $db->loadObject();
|
$stats = $db->loadObject();
|
||||||
|
|
||||||
@@ -274,7 +274,7 @@ class DashboardModel extends BaseDatabaseModel
|
|||||||
->select('COALESCE(SUM(r.total_size), 0) AS total_size')
|
->select('COALESCE(SUM(r.total_size), 0) AS total_size')
|
||||||
->from($db->quoteName('#__mokosuitebackup_records', 'r'))
|
->from($db->quoteName('#__mokosuitebackup_records', 'r'))
|
||||||
->join('LEFT', $db->quoteName('#__mokosuitebackup_profiles', 'p') . ' ON p.id = r.profile_id')
|
->join('LEFT', $db->quoteName('#__mokosuitebackup_profiles', 'p') . ' ON p.id = r.profile_id')
|
||||||
->where($db->quoteName('r.status') . ' IN (' . implode(',', array_map([$db, 'quote'], ['complete', 'warning'])) . ')')
|
->where($db->quoteName('r.status') . ' = ' . $db->quote('complete'))
|
||||||
->group($db->quoteName('r.profile_id'))
|
->group($db->quoteName('r.profile_id'))
|
||||||
->order('total_size DESC');
|
->order('total_size DESC');
|
||||||
$db->setQuery($query);
|
$db->setQuery($query);
|
||||||
|
|||||||
@@ -41,7 +41,7 @@ class HtmlView extends BaseHtmlView
|
|||||||
|
|
||||||
$user = Factory::getApplication()->getIdentity();
|
$user = Factory::getApplication()->getIdentity();
|
||||||
|
|
||||||
if (\in_array($this->item->status, ['complete', 'warning'], true)
|
if ($this->item->status === 'complete'
|
||||||
&& !empty($this->item->filesexist)
|
&& !empty($this->item->filesexist)
|
||||||
&& $user->authorise('mokosuitebackup.backup.download', 'com_mokosuitebackup')
|
&& $user->authorise('mokosuitebackup.backup.download', 'com_mokosuitebackup')
|
||||||
) {
|
) {
|
||||||
|
|||||||
@@ -113,10 +113,6 @@ class HtmlView extends BaseHtmlView
|
|||||||
ToolbarHelper::custom('backups.compare', 'copy', '', 'COM_MOKOJOOMBACKUP_TOOLBAR_COMPARE', true);
|
ToolbarHelper::custom('backups.compare', 'copy', '', 'COM_MOKOJOOMBACKUP_TOOLBAR_COMPARE', true);
|
||||||
}
|
}
|
||||||
|
|
||||||
if ($user->authorise('mokosuitebackup.backup.cancel', 'com_mokosuitebackup')) {
|
|
||||||
ToolbarHelper::custom('backups.cancelStalled', 'stop-circle', '', 'COM_MOKOJOOMBACKUP_TOOLBAR_CANCEL_STALLED', true);
|
|
||||||
}
|
|
||||||
|
|
||||||
if ($user->authorise('core.delete', 'com_mokosuitebackup')) {
|
if ($user->authorise('core.delete', 'com_mokosuitebackup')) {
|
||||||
ToolbarHelper::deleteList('JGLOBAL_CONFIRM_DELETE', 'backups.delete');
|
ToolbarHelper::deleteList('JGLOBAL_CONFIRM_DELETE', 'backups.delete');
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -30,7 +30,6 @@ $ajaxUrl = Route::_('index.php?option=com_mokosuitebackup&format=json', false)
|
|||||||
<?php
|
<?php
|
||||||
$statusClass = match ($this->item->status) {
|
$statusClass = match ($this->item->status) {
|
||||||
'complete' => 'badge bg-success',
|
'complete' => 'badge bg-success',
|
||||||
'warning' => 'badge bg-warning text-dark',
|
|
||||||
'running' => 'badge bg-info',
|
'running' => 'badge bg-info',
|
||||||
'fail' => 'badge bg-danger',
|
'fail' => 'badge bg-danger',
|
||||||
default => 'badge bg-secondary',
|
default => 'badge bg-secondary',
|
||||||
|
|||||||
@@ -92,7 +92,6 @@ $listDirn = $this->escape($this->state->get('list.direction'));
|
|||||||
<?php
|
<?php
|
||||||
$statusClass = match ($item->status) {
|
$statusClass = match ($item->status) {
|
||||||
'complete' => 'badge bg-success',
|
'complete' => 'badge bg-success',
|
||||||
'warning' => 'badge bg-warning text-dark',
|
|
||||||
'running' => 'badge bg-info',
|
'running' => 'badge bg-info',
|
||||||
'fail' => 'badge bg-danger',
|
'fail' => 'badge bg-danger',
|
||||||
default => 'badge bg-secondary',
|
default => 'badge bg-secondary',
|
||||||
|
|||||||
-1
@@ -11,7 +11,6 @@ MOD_MOKOSUITEBACKUP_CPANEL_NOT_INSTALLED="MokoSuiteBackup is not installed or is
|
|||||||
|
|
||||||
MOD_MOKOSUITEBACKUP_CPANEL_LAST_BACKUP="Last Backup"
|
MOD_MOKOSUITEBACKUP_CPANEL_LAST_BACKUP="Last Backup"
|
||||||
MOD_MOKOSUITEBACKUP_CPANEL_STATUS_OK="Success"
|
MOD_MOKOSUITEBACKUP_CPANEL_STATUS_OK="Success"
|
||||||
MOD_MOKOSUITEBACKUP_CPANEL_STATUS_WARNING="Warning"
|
|
||||||
MOD_MOKOSUITEBACKUP_CPANEL_STATUS_FAIL="Failed"
|
MOD_MOKOSUITEBACKUP_CPANEL_STATUS_FAIL="Failed"
|
||||||
MOD_MOKOSUITEBACKUP_CPANEL_NO_BACKUPS="No backups yet."
|
MOD_MOKOSUITEBACKUP_CPANEL_NO_BACKUPS="No backups yet."
|
||||||
MOD_MOKOSUITEBACKUP_CPANEL_FILES_TABLES="%d files, %d tables"
|
MOD_MOKOSUITEBACKUP_CPANEL_FILES_TABLES="%d files, %d tables"
|
||||||
|
|||||||
@@ -8,7 +8,7 @@
|
|||||||
-->
|
-->
|
||||||
<extension type="module" client="administrator" method="upgrade">
|
<extension type="module" client="administrator" method="upgrade">
|
||||||
<name>mod_mokosuitebackup_cpanel</name>
|
<name>mod_mokosuitebackup_cpanel</name>
|
||||||
<version>02.52.24</version>
|
<version>02.52.19</version>
|
||||||
<creationDate>2026-06-23</creationDate>
|
<creationDate>2026-06-23</creationDate>
|
||||||
<author>Moko Consulting</author>
|
<author>Moko Consulting</author>
|
||||||
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
<authorEmail>hello@mokoconsulting.tech</authorEmail>
|
||||||
|
|||||||
@@ -51,20 +51,10 @@ $moduleId = 'mod-msb-cpanel-' . $displayData['module']->id;
|
|||||||
<?php if ($latest) : ?>
|
<?php if ($latest) : ?>
|
||||||
<div class="d-flex align-items-center justify-content-between">
|
<div class="d-flex align-items-center justify-content-between">
|
||||||
<div>
|
<div>
|
||||||
<?php
|
<span class="badge <?php echo $latest['status'] === 'complete' ? 'bg-success' : 'bg-danger'; ?>">
|
||||||
$cpanelBadge = match ($latest['status']) {
|
<?php echo $latest['status'] === 'complete'
|
||||||
'complete' => 'bg-success',
|
? Text::_('MOD_MOKOSUITEBACKUP_CPANEL_STATUS_OK')
|
||||||
'warning' => 'bg-warning text-dark',
|
: Text::_('MOD_MOKOSUITEBACKUP_CPANEL_STATUS_FAIL'); ?>
|
||||||
default => 'bg-danger',
|
|
||||||
};
|
|
||||||
$cpanelLabel = match ($latest['status']) {
|
|
||||||
'complete' => Text::_('MOD_MOKOSUITEBACKUP_CPANEL_STATUS_OK'),
|
|
||||||
'warning' => Text::_('MOD_MOKOSUITEBACKUP_CPANEL_STATUS_WARNING'),
|
|
||||||
default => Text::_('MOD_MOKOSUITEBACKUP_CPANEL_STATUS_FAIL'),
|
|
||||||
};
|
|
||||||
?>
|
|
||||||
<span class="badge <?php echo $cpanelBadge; ?>">
|
|
||||||
<?php echo $cpanelLabel; ?>
|
|
||||||
</span>
|
</span>
|
||||||
<span class="ms-1 small text-muted">
|
<span class="ms-1 small text-muted">
|
||||||
<?php echo htmlspecialchars($latest['profile'] ?? ''); ?>
|
<?php echo htmlspecialchars($latest['profile'] ?? ''); ?>
|
||||||
|
|||||||
@@ -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>02.52.24</version>
|
<version>02.52.19</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>02.52.24</version>
|
<version>02.52.19</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>02.52.24</version>
|
<version>02.52.19</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>02.52.24</version>
|
<version>02.52.19</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>02.52.24</version>
|
<version>02.52.19</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>
|
||||||
|
|||||||
@@ -28,6 +28,7 @@ final class MokoSuiteBackup extends CMSPlugin implements SubscriberInterface
|
|||||||
return [
|
return [
|
||||||
'onAfterInitialise' => 'onAfterInitialise',
|
'onAfterInitialise' => 'onAfterInitialise',
|
||||||
'onAfterRoute' => 'onAfterRoute',
|
'onAfterRoute' => 'onAfterRoute',
|
||||||
|
'onBeforeRender' => 'onBeforeRender',
|
||||||
'onExtensionBeforeUpdate' => 'onExtensionBeforeUpdate',
|
'onExtensionBeforeUpdate' => 'onExtensionBeforeUpdate',
|
||||||
'onExtensionBeforeUninstall' => 'onExtensionBeforeUninstall',
|
'onExtensionBeforeUninstall' => 'onExtensionBeforeUninstall',
|
||||||
];
|
];
|
||||||
@@ -347,10 +348,52 @@ final class MokoSuiteBackup extends CMSPlugin implements SubscriberInterface
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Run a backup before any extension is updated.
|
* Inject JavaScript on installer/update pages to show a backup progress
|
||||||
|
* modal before extension updates proceed.
|
||||||
|
*/
|
||||||
|
public function onBeforeRender(Event $event): void
|
||||||
|
{
|
||||||
|
$app = $this->getApplication();
|
||||||
|
|
||||||
|
if (!$app->isClient('administrator')) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$option = $app->input->getCmd('option', '');
|
||||||
|
$view = $app->input->getCmd('view', '');
|
||||||
|
|
||||||
|
if ($option !== 'com_installer' && $option !== 'com_joomlaupdate') {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$params = ComponentHelper::getParams('com_mokosuitebackup');
|
||||||
|
|
||||||
|
if (!(int) $params->get('backup_before_update', 0)) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$profileId = (int) $params->get('default_profile', 1);
|
||||||
|
$token = \Joomla\CMS\Session\Session::getFormToken();
|
||||||
|
|
||||||
|
$js = $this->getPreUpdateBackupScript($profileId, $token);
|
||||||
|
|
||||||
|
$app->getDocument()->addScriptDeclaration($js);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Run a backup before any extension is updated (server-side fallback
|
||||||
|
* for CLI/API updates where JavaScript is not available).
|
||||||
*/
|
*/
|
||||||
public function onExtensionBeforeUpdate(Event $event): void
|
public function onExtensionBeforeUpdate(Event $event): void
|
||||||
{
|
{
|
||||||
|
$session = Factory::getSession();
|
||||||
|
|
||||||
|
if ($session->get('mokosuitebackup.preupdate_js_done', false)) {
|
||||||
|
$session->set('mokosuitebackup.preupdate_js_done', false);
|
||||||
|
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
$this->runPreActionBackup('backup_before_update', 'Pre-update backup');
|
$this->runPreActionBackup('backup_before_update', 'Pre-update backup');
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -408,6 +451,161 @@ final class MokoSuiteBackup extends CMSPlugin implements SubscriberInterface
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Build the inline JavaScript that intercepts extension update actions
|
||||||
|
* and runs a stepped backup with a progress modal first.
|
||||||
|
*/
|
||||||
|
private function getPreUpdateBackupScript(int $profileId, string $token): string
|
||||||
|
{
|
||||||
|
$baseUrl = \Joomla\CMS\Uri\Uri::base() . 'index.php?option=com_mokosuitebackup&format=json&' . $token . '=1';
|
||||||
|
|
||||||
|
return <<<JS
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
var msbOrigSubmit = Joomla.submitbutton;
|
||||||
|
var msbBackupRunning = false;
|
||||||
|
var msbPendingTask = null;
|
||||||
|
|
||||||
|
// Create modal
|
||||||
|
var msbModal = document.createElement('div');
|
||||||
|
msbModal.id = 'msbPreUpdateModal';
|
||||||
|
msbModal.className = 'modal fade';
|
||||||
|
msbModal.setAttribute('tabindex', '-1');
|
||||||
|
msbModal.setAttribute('data-bs-backdrop', 'static');
|
||||||
|
msbModal.setAttribute('data-bs-keyboard', 'false');
|
||||||
|
msbModal.innerHTML = '<div class="modal-dialog modal-dialog-centered"><div class="modal-content">'
|
||||||
|
+ '<div class="modal-header"><h5 class="modal-title"><span class="icon-archive me-2"></span>Pre-Update Backup</h5></div>'
|
||||||
|
+ '<div class="modal-body">'
|
||||||
|
+ '<p id="msbStatusText">Creating a backup before updating...</p>'
|
||||||
|
+ '<div class="progress" style="height:24px"><div id="msbProgressBar" class="progress-bar progress-bar-striped progress-bar-animated" style="width:0%">0%</div></div>'
|
||||||
|
+ '<div id="msbLogArea" style="max-height:120px;overflow-y:auto;font-size:0.8rem;color:#64748b;margin-top:12px;font-family:monospace;white-space:pre-wrap"></div>'
|
||||||
|
+ '</div>'
|
||||||
|
+ '<div class="modal-footer" id="msbFooter" style="display:none">'
|
||||||
|
+ '<button type="button" class="btn btn-secondary" id="msbSkipBtn">Skip & Update</button>'
|
||||||
|
+ '<button type="button" class="btn btn-danger" id="msbCancelBtn">Cancel</button>'
|
||||||
|
+ '</div>'
|
||||||
|
+ '</div></div>';
|
||||||
|
document.body.appendChild(msbModal);
|
||||||
|
|
||||||
|
var bsModal = new bootstrap.Modal(msbModal);
|
||||||
|
|
||||||
|
function msbUpdateProgress(pct, msg) {
|
||||||
|
var bar = document.getElementById('msbProgressBar');
|
||||||
|
bar.style.width = pct + '%';
|
||||||
|
bar.textContent = pct + '%';
|
||||||
|
if (msg) document.getElementById('msbStatusText').textContent = msg;
|
||||||
|
}
|
||||||
|
|
||||||
|
function msbLog(msg) {
|
||||||
|
var log = document.getElementById('msbLogArea');
|
||||||
|
log.textContent += msg + '\\n';
|
||||||
|
log.scrollTop = log.scrollHeight;
|
||||||
|
}
|
||||||
|
|
||||||
|
function msbShowFooter() {
|
||||||
|
document.getElementById('msbFooter').style.display = '';
|
||||||
|
}
|
||||||
|
|
||||||
|
function msbFinish(success) {
|
||||||
|
msbBackupRunning = false;
|
||||||
|
if (success && msbPendingTask) {
|
||||||
|
msbUpdateProgress(100, 'Backup complete — proceeding with update...');
|
||||||
|
// Mark JS backup done so server-side handler skips
|
||||||
|
fetch('{$baseUrl}&task=ajax.markPreUpdateDone', {method:'POST', headers:{'X-Requested-With':'XMLHttpRequest'}});
|
||||||
|
setTimeout(function() {
|
||||||
|
bsModal.hide();
|
||||||
|
msbOrigSubmit.call(Joomla, msbPendingTask);
|
||||||
|
msbPendingTask = null;
|
||||||
|
}, 800);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
function msbRunStep(sessionId) {
|
||||||
|
fetch('{$baseUrl}&task=ajax.step&session_id=' + encodeURIComponent(sessionId), {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {'X-Requested-With': 'XMLHttpRequest'}
|
||||||
|
})
|
||||||
|
.then(function(r) { return r.json(); })
|
||||||
|
.then(function(data) {
|
||||||
|
if (data.error) {
|
||||||
|
msbUpdateProgress(data.progress || 0, 'Backup error: ' + data.message);
|
||||||
|
msbLog('ERROR: ' + data.message);
|
||||||
|
msbShowFooter();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
msbUpdateProgress(data.progress || 0, data.message || data.phase || 'Working...');
|
||||||
|
if (data.message) msbLog(data.message);
|
||||||
|
if (data.done) {
|
||||||
|
msbFinish(true);
|
||||||
|
} else {
|
||||||
|
msbRunStep(sessionId);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(function(err) {
|
||||||
|
msbUpdateProgress(0, 'Backup request failed');
|
||||||
|
msbLog('Network error: ' + err.message);
|
||||||
|
msbShowFooter();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function msbStartBackup() {
|
||||||
|
msbBackupRunning = true;
|
||||||
|
msbUpdateProgress(0, 'Initializing backup...');
|
||||||
|
msbLog('Starting pre-update backup (profile {$profileId})...');
|
||||||
|
|
||||||
|
fetch('{$baseUrl}&task=ajax.init&profile_id={$profileId}&description=Pre-update+backup', {
|
||||||
|
method: 'POST',
|
||||||
|
headers: {'X-Requested-With': 'XMLHttpRequest'}
|
||||||
|
})
|
||||||
|
.then(function(r) { return r.json(); })
|
||||||
|
.then(function(data) {
|
||||||
|
if (data.error) {
|
||||||
|
msbUpdateProgress(0, 'Backup init failed: ' + data.message);
|
||||||
|
msbLog('INIT ERROR: ' + data.message);
|
||||||
|
msbShowFooter();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
msbLog('Backup initialized — ' + data.message);
|
||||||
|
msbUpdateProgress(data.progress || 5, data.message || 'Running...');
|
||||||
|
msbRunStep(data.session_id);
|
||||||
|
})
|
||||||
|
.catch(function(err) {
|
||||||
|
msbUpdateProgress(0, 'Could not start backup');
|
||||||
|
msbLog('Network error: ' + err.message);
|
||||||
|
msbShowFooter();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// Intercept Joomla toolbar submit
|
||||||
|
Joomla.submitbutton = function(task) {
|
||||||
|
if ((task === 'update.update' || task === 'update.install') && !msbBackupRunning) {
|
||||||
|
msbPendingTask = task;
|
||||||
|
bsModal.show();
|
||||||
|
msbStartBackup();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
msbOrigSubmit.call(Joomla, task);
|
||||||
|
};
|
||||||
|
|
||||||
|
// Skip button — proceed without backup
|
||||||
|
document.getElementById('msbSkipBtn').addEventListener('click', function() {
|
||||||
|
bsModal.hide();
|
||||||
|
msbBackupRunning = false;
|
||||||
|
if (msbPendingTask) {
|
||||||
|
msbOrigSubmit.call(Joomla, msbPendingTask);
|
||||||
|
msbPendingTask = null;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// Cancel button — abort everything
|
||||||
|
document.getElementById('msbCancelBtn').addEventListener('click', function() {
|
||||||
|
bsModal.hide();
|
||||||
|
msbBackupRunning = false;
|
||||||
|
msbPendingTask = null;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
JS;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Send a JSON response and terminate — used by web cron handler.
|
* Send a JSON response and terminate — used by web cron handler.
|
||||||
*/
|
*/
|
||||||
|
|||||||
@@ -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>02.52.24</version>
|
<version>02.52.19</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>02.52.24</version>
|
<version>02.52.19</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>02.52.24</version>
|
<version>02.52.19</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