add update cli-command; add update-channel setting (stable|testing)
Signed-off-by: Michael Kaufmann <d00p@froxlor.org>
This commit is contained in:
@@ -108,6 +108,19 @@ return [
|
|||||||
'default' => false,
|
'default' => false,
|
||||||
'save_method' => 'storeSettingField'
|
'save_method' => 'storeSettingField'
|
||||||
],
|
],
|
||||||
|
'update_channel' => [
|
||||||
|
'label' => lng('serversettings.update_channel'),
|
||||||
|
'settinggroup' => 'system',
|
||||||
|
'varname' => 'update_channel',
|
||||||
|
'type' => 'select',
|
||||||
|
'default' => 'stable',
|
||||||
|
'select_var' => [
|
||||||
|
'stable' => lng('serversettings.uc_stable'),
|
||||||
|
'testing' => lng('serversettings.uc_testing')
|
||||||
|
],
|
||||||
|
'save_method' => 'storeSettingField',
|
||||||
|
'advanced_mode' => true
|
||||||
|
],
|
||||||
'system_validatedomain' => [
|
'system_validatedomain' => [
|
||||||
'label' => lng('serversettings.validate_domain'),
|
'label' => lng('serversettings.validate_domain'),
|
||||||
'settinggroup' => 'system',
|
'settinggroup' => 'system',
|
||||||
|
|||||||
@@ -29,32 +29,12 @@ require __DIR__ . '/lib/init.php';
|
|||||||
use Froxlor\Froxlor;
|
use Froxlor\Froxlor;
|
||||||
use Froxlor\FroxlorLogger;
|
use Froxlor\FroxlorLogger;
|
||||||
use Froxlor\Http\HttpClient;
|
use Froxlor\Http\HttpClient;
|
||||||
|
use Froxlor\Install\AutoUpdate;
|
||||||
use Froxlor\Settings;
|
use Froxlor\Settings;
|
||||||
use Froxlor\UI\Panel\UI;
|
use Froxlor\UI\Panel\UI;
|
||||||
use Froxlor\UI\Response;
|
use Froxlor\UI\Response;
|
||||||
|
|
||||||
// define update-uri
|
|
||||||
define('UPDATE_URI', "https://version.froxlor.org/Froxlor/api/" . Froxlor::VERSION);
|
|
||||||
define('RELEASE_URI', "https://autoupdate.froxlor.org/froxlor-{version}.zip");
|
|
||||||
define('CHECKSUM_URI', "https://autoupdate.froxlor.org/froxlor-{version}.zip.sha256");
|
|
||||||
|
|
||||||
if ($page != 'error') {
|
if ($page != 'error') {
|
||||||
// check for archive-stuff
|
|
||||||
if (!extension_loaded('zip')) {
|
|
||||||
Response::redirectTo($filename, [
|
|
||||||
'page' => 'error',
|
|
||||||
'errno' => 2
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// 0.11.x requires 7.4 at least
|
|
||||||
if (version_compare("7.4.0", PHP_VERSION, ">=")) {
|
|
||||||
Response::redirectTo($filename, [
|
|
||||||
'page' => 'error',
|
|
||||||
'errno' => 10
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// check for webupdate to be enabled
|
// check for webupdate to be enabled
|
||||||
if (Settings::Config('enable_webupdate') != true) {
|
if (Settings::Config('enable_webupdate') != true) {
|
||||||
Response::redirectTo($filename, [
|
Response::redirectTo($filename, [
|
||||||
@@ -71,170 +51,98 @@ if ($page == 'overview') {
|
|||||||
|
|
||||||
// check for new version
|
// check for new version
|
||||||
try {
|
try {
|
||||||
$latestversion = HttpClient::urlGet(UPDATE_URI, true, 3);
|
$result = AutoUpdate::checkVersion();
|
||||||
} catch (Exception $e) {
|
} catch (Exception $e) {
|
||||||
Response::dynamicError("Version-check currently unavailable, please try again later");
|
Response::dynamicError($e->getMessage());
|
||||||
}
|
}
|
||||||
$latestversion = explode('|', $latestversion);
|
|
||||||
|
|
||||||
if (is_array($latestversion) && count($latestversion) >= 1) {
|
if ($result == 1) {
|
||||||
$_version = $latestversion[0];
|
|
||||||
$_message = isset($latestversion[1]) ? $latestversion[1] : '';
|
|
||||||
$_link = isset($latestversion[2]) ? $latestversion[2] : htmlspecialchars($filename . '?page=' . urlencode($page) . '&lookfornewversion=yes');
|
|
||||||
|
|
||||||
// add the branding so debian guys are not gettings confused
|
|
||||||
// about their version-number
|
|
||||||
$version_label = $_version . Froxlor::BRANDING;
|
|
||||||
$version_link = $_link;
|
|
||||||
$message_addinfo = $_message;
|
|
||||||
|
|
||||||
// not numeric -> error-message
|
|
||||||
if (!preg_match('/^((\d+\\.)(\d+\\.)(\d+\\.)?(\d+)?(\-(svn|dev|rc)(\d+))?)$/', $_version)) {
|
|
||||||
// check for customized version to not output
|
|
||||||
// "There is a newer version of froxlor" besides the error-message
|
|
||||||
Response::redirectTo($filename, [
|
|
||||||
'page' => 'error',
|
|
||||||
'errno' => 3
|
|
||||||
]);
|
|
||||||
} elseif (Froxlor::versionCompare2(Froxlor::VERSION, $_version) == -1) {
|
|
||||||
// there is a newer version - yay
|
|
||||||
$isnewerversion = 1;
|
|
||||||
} else {
|
|
||||||
// nothing new
|
|
||||||
$isnewerversion = 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
// anzeige über version-status mit ggfls. formular
|
// anzeige über version-status mit ggfls. formular
|
||||||
// zum update schritt #1 -> download
|
// zum update schritt #1 -> download
|
||||||
if ($isnewerversion == 1) {
|
$text = lng('admin.newerversionavailable') . ' ' . lng('admin.newerversiondetails', [AutoUpdate::getFromResult('version'), Froxlor::VERSION]);
|
||||||
$text = 'There is a newer version available. Update to version <b>' . $_version . '</b> now?<br/>(Your current version is: ' . Froxlor::VERSION . ')';
|
|
||||||
|
|
||||||
$upd_formfield = [
|
$upd_formfield = [
|
||||||
'updates' => [
|
'updates' => [
|
||||||
'title' => lng('update.update'),
|
'title' => lng('update.update'),
|
||||||
'image' => 'fa-solid fa-download',
|
'image' => 'fa-solid fa-download',
|
||||||
'sections' => [
|
'sections' => [
|
||||||
'section_autoupd' => [
|
'section_autoupd' => [
|
||||||
'fields' => [
|
'fields' => [
|
||||||
'newversion' => ['type' => 'hidden', 'value' => $_version]
|
'newversion' => ['type' => 'hidden', 'value' => AutoUpdate::getFromResult('version')]
|
||||||
]
|
|
||||||
]
|
|
||||||
],
|
|
||||||
'buttons' => [
|
|
||||||
[
|
|
||||||
'class' => 'btn-outline-secondary',
|
|
||||||
'label' => lng('panel.cancel'),
|
|
||||||
'type' => 'reset'
|
|
||||||
],
|
|
||||||
[
|
|
||||||
'label' => lng('update.proceed')
|
|
||||||
]
|
]
|
||||||
]
|
]
|
||||||
|
],
|
||||||
|
'buttons' => [
|
||||||
|
[
|
||||||
|
'class' => 'btn-outline-secondary',
|
||||||
|
'label' => lng('panel.cancel'),
|
||||||
|
'type' => 'reset'
|
||||||
|
],
|
||||||
|
[
|
||||||
|
'label' => lng('update.proceed')
|
||||||
|
]
|
||||||
]
|
]
|
||||||
];
|
]
|
||||||
|
];
|
||||||
|
|
||||||
UI::view('user/form-note.html.twig', [
|
UI::view('user/form-note.html.twig', [
|
||||||
'formaction' => $linker->getLink(['section' => 'autoupdate', 'page' => 'getdownload']),
|
'formaction' => $linker->getLink(['section' => 'autoupdate', 'page' => 'getdownload']),
|
||||||
'formdata' => $upd_formfield['updates'],
|
'formdata' => $upd_formfield['updates'],
|
||||||
// alert
|
// alert
|
||||||
'type' => 'warning',
|
'type' => 'warning',
|
||||||
'alert_msg' => $text
|
'alert_msg' => $text
|
||||||
]);
|
]);
|
||||||
} elseif ($isnewerversion == 0) {
|
} else if ($result < 0 || $result > 1) {
|
||||||
// all good
|
// remote errors
|
||||||
Response::standardSuccess('noupdatesavail');
|
if ($result < 0) {
|
||||||
|
Response::dynamicError(AutoUpdate::getLastError());
|
||||||
} else {
|
} else {
|
||||||
Response::standardError('customized_version');
|
Response::redirectTo($filename, [
|
||||||
|
'page' => 'error',
|
||||||
|
'errno' => $result
|
||||||
|
]);
|
||||||
}
|
}
|
||||||
|
} else {
|
||||||
|
// no new version
|
||||||
|
Response::standardSuccess('update.noupdatesavail');
|
||||||
}
|
}
|
||||||
} // download the new archive
|
} // download the new archive
|
||||||
elseif ($page == 'getdownload') {
|
elseif ($page == 'getdownload') {
|
||||||
// retrieve the new version from the form
|
// retrieve the new version from the form
|
||||||
$newversion = isset($_POST['newversion']) ? $_POST['newversion'] : null;
|
$newversion = isset($_POST['newversion']) ? $_POST['newversion'] : null;
|
||||||
|
|
||||||
|
$result = 6;
|
||||||
// valid?
|
// valid?
|
||||||
if ($newversion !== null) {
|
if ($newversion !== null) {
|
||||||
// define files to get
|
$result = AutoUpdate::downloadZip($newversion);
|
||||||
$toLoad = str_replace('{version}', $newversion, RELEASE_URI);
|
if (!is_numeric($result)) {
|
||||||
$toCheck = str_replace('{version}', $newversion, CHECKSUM_URI);
|
// to the next step
|
||||||
|
|
||||||
// check for local destination folder
|
|
||||||
if (!is_dir(Froxlor::getInstallDir() . '/updates/')) {
|
|
||||||
mkdir(Froxlor::getInstallDir() . '/updates/');
|
|
||||||
}
|
|
||||||
|
|
||||||
// name archive
|
|
||||||
$localArchive = Froxlor::getInstallDir() . '/updates/' . basename($toLoad);
|
|
||||||
|
|
||||||
$log->logAction(FroxlorLogger::ADM_ACTION, LOG_NOTICE, "Downloading " . $toLoad . " to " . $localArchive);
|
|
||||||
|
|
||||||
// remove old archive
|
|
||||||
if (file_exists($localArchive)) {
|
|
||||||
@unlink($localArchive);
|
|
||||||
}
|
|
||||||
|
|
||||||
// get archive data
|
|
||||||
try {
|
|
||||||
HttpClient::fileGet($toLoad, $localArchive);
|
|
||||||
} catch (Exception $e) {
|
|
||||||
Response::redirectTo($filename, [
|
Response::redirectTo($filename, [
|
||||||
'page' => 'error',
|
'page' => 'extract',
|
||||||
'errno' => 4
|
'archive' => $result
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
// validate the integrity of the downloaded file
|
|
||||||
$_shouldsum = HttpClient::urlGet($toCheck);
|
|
||||||
if (!empty($_shouldsum)) {
|
|
||||||
$_t = explode(" ", $_shouldsum);
|
|
||||||
$shouldsum = $_t[0];
|
|
||||||
} else {
|
|
||||||
$shouldsum = null;
|
|
||||||
}
|
|
||||||
$filesum = hash_file('sha256', $localArchive);
|
|
||||||
|
|
||||||
if ($filesum != $shouldsum) {
|
|
||||||
Response::redirectTo($filename, [
|
|
||||||
'page' => 'error',
|
|
||||||
'errno' => 9
|
|
||||||
]);
|
|
||||||
}
|
|
||||||
|
|
||||||
// to the next step
|
|
||||||
Response::redirectTo($filename, [
|
|
||||||
'page' => 'extract',
|
|
||||||
'archive' => basename($localArchive)
|
|
||||||
]);
|
|
||||||
}
|
}
|
||||||
Response::redirectTo($filename, [
|
Response::redirectTo($filename, [
|
||||||
'page' => 'error',
|
'page' => 'error',
|
||||||
'errno' => 6
|
'errno' => $result
|
||||||
]);
|
]);
|
||||||
} // extract and install new version
|
} // extract and install new version
|
||||||
elseif ($page == 'extract') {
|
elseif ($page == 'extract') {
|
||||||
if (isset($_POST['send']) && $_POST['send'] == 'send') {
|
if (isset($_POST['send']) && $_POST['send'] == 'send') {
|
||||||
$toExtract = isset($_POST['archive']) ? $_POST['archive'] : null;
|
$toExtract = isset($_POST['archive']) ? $_POST['archive'] : null;
|
||||||
$localArchive = Froxlor::getInstallDir() . '/updates/' . $toExtract;
|
$localArchive = Froxlor::getInstallDir() . '/updates/' . $toExtract;
|
||||||
// decompress from zip
|
$log->logAction(FroxlorLogger::ADM_ACTION, LOG_NOTICE, "Extracting " . $localArchive . " to " . Froxlor::getInstallDir());
|
||||||
$zip = new ZipArchive();
|
$result = AutoUpdate::extractZip($localArchive);
|
||||||
$res = $zip->open($localArchive);
|
if ($result > 0) {
|
||||||
if ($res === true) {
|
|
||||||
$log->logAction(FroxlorLogger::ADM_ACTION, LOG_NOTICE, "Extracting " . $localArchive . " to " . Froxlor::getInstallDir());
|
|
||||||
$zip->extractTo(Froxlor::getInstallDir());
|
|
||||||
$zip->close();
|
|
||||||
// success - remove unused archive
|
|
||||||
@unlink($localArchive);
|
|
||||||
// wait a bit before we redirect to be sure
|
|
||||||
sleep(2);
|
|
||||||
} else {
|
|
||||||
// error
|
// error
|
||||||
Response::redirectTo($filename, [
|
Response::redirectTo($filename, [
|
||||||
'page' => 'error',
|
'page' => 'error',
|
||||||
'errno' => 8
|
'errno' => $result
|
||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
// redirect to update-page
|
||||||
// redirect to update-page?
|
|
||||||
Response::redirectTo('admin_updates.php');
|
Response::redirectTo('admin_updates.php');
|
||||||
} else {
|
} else {
|
||||||
$toExtract = isset($_GET['archive']) ? $_GET['archive'] : null;
|
$toExtract = isset($_GET['archive']) ? $_GET['archive'] : null;
|
||||||
@@ -248,7 +156,7 @@ elseif ($page == 'extract') {
|
|||||||
]);
|
]);
|
||||||
}
|
}
|
||||||
|
|
||||||
$text = 'Extract downloaded archive "' . $toExtract . '"?';
|
$text = lng('admin.extractdownloadedzip', [$toExtract]);
|
||||||
|
|
||||||
$upd_formfield = [
|
$upd_formfield = [
|
||||||
'updates' => [
|
'updates' => [
|
||||||
|
|||||||
@@ -31,6 +31,7 @@ use Froxlor\Cli\RunApiCommand;
|
|||||||
use Froxlor\Cli\ConfigServices;
|
use Froxlor\Cli\ConfigServices;
|
||||||
use Froxlor\Cli\PhpSessionclean;
|
use Froxlor\Cli\PhpSessionclean;
|
||||||
use Froxlor\Cli\SwitchServerIp;
|
use Froxlor\Cli\SwitchServerIp;
|
||||||
|
use Froxlor\Cli\UpdateCommand;
|
||||||
use Froxlor\Froxlor;
|
use Froxlor\Froxlor;
|
||||||
|
|
||||||
// validate correct php version
|
// validate correct php version
|
||||||
@@ -51,4 +52,5 @@ $application->add(new RunApiCommand());
|
|||||||
$application->add(new ConfigServices());
|
$application->add(new ConfigServices());
|
||||||
$application->add(new PhpSessionclean());
|
$application->add(new PhpSessionclean());
|
||||||
$application->add(new SwitchServerIp());
|
$application->add(new SwitchServerIp());
|
||||||
|
$application->add(new UpdateCommand());
|
||||||
$application->run();
|
$application->run();
|
||||||
|
|||||||
@@ -699,6 +699,7 @@ opcache.validate_timestamps'),
|
|||||||
('system', 'froxlorusergroup_gid', ''),
|
('system', 'froxlorusergroup_gid', ''),
|
||||||
('system', 'acmeshpath', '/root/.acme.sh/acme.sh'),
|
('system', 'acmeshpath', '/root/.acme.sh/acme.sh'),
|
||||||
('system', 'distribution', ''),
|
('system', 'distribution', ''),
|
||||||
|
('system', 'update_channel', 'stable'),
|
||||||
('api', 'enabled', '0'),
|
('api', 'enabled', '0'),
|
||||||
('2fa', 'enabled', '1'),
|
('2fa', 'enabled', '1'),
|
||||||
('panel', 'decimal_places', '4'),
|
('panel', 'decimal_places', '4'),
|
||||||
|
|||||||
@@ -133,6 +133,7 @@ if (Froxlor::isFroxlorVersion('0.10.99')) {
|
|||||||
Settings::AddNew("panel.settings_mode", $panel_settings_mode);
|
Settings::AddNew("panel.settings_mode", $panel_settings_mode);
|
||||||
$system_distribution = isset($_POST['system_distribution']) ? $_POST['system_distribution'] : '';
|
$system_distribution = isset($_POST['system_distribution']) ? $_POST['system_distribution'] : '';
|
||||||
Settings::AddNew("system.distribution", $system_distribution);
|
Settings::AddNew("system.distribution", $system_distribution);
|
||||||
|
Settings::AddNew("system.update_channel", 'stable');
|
||||||
Update::lastStepStatus(0);
|
Update::lastStepStatus(0);
|
||||||
|
|
||||||
Update::showUpdateStep("Adjusting existing settings");
|
Update::showUpdateStep("Adjusting existing settings");
|
||||||
|
|||||||
176
lib/Froxlor/Cli/UpdateCommand.php
Normal file
176
lib/Froxlor/Cli/UpdateCommand.php
Normal file
@@ -0,0 +1,176 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This file is part of the Froxlor project.
|
||||||
|
* Copyright (c) 2010 the Froxlor Team (see authors).
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, you can also view it online at
|
||||||
|
* https://files.froxlor.org/misc/COPYING.txt
|
||||||
|
*
|
||||||
|
* @copyright the authors
|
||||||
|
* @author Froxlor team <team@froxlor.org>
|
||||||
|
* @license https://files.froxlor.org/misc/COPYING.txt GPLv2
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Froxlor\Cli;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
use Froxlor\Froxlor;
|
||||||
|
use Froxlor\Install\AutoUpdate;
|
||||||
|
use Symfony\Component\Console\Input\InputInterface;
|
||||||
|
use Symfony\Component\Console\Input\InputOption;
|
||||||
|
use Symfony\Component\Console\Output\OutputInterface;
|
||||||
|
use Symfony\Component\Console\Question\ConfirmationQuestion;
|
||||||
|
|
||||||
|
final class UpdateCommand extends CliCommand
|
||||||
|
{
|
||||||
|
|
||||||
|
protected function configure()
|
||||||
|
{
|
||||||
|
$this->setName('froxlor:update');
|
||||||
|
$this->setDescription('Check for newer version and update froxlor');
|
||||||
|
$this->addOption('check-only', 'c', InputOption::VALUE_NONE, 'Only check for newer version and exit')
|
||||||
|
->addOption('yes-to-all', 'A', InputOption::VALUE_NONE, 'Do not ask for download, extract and database-update, just do it (if not --check-only is set)')
|
||||||
|
->addOption('integer-return', 'i', InputOption::VALUE_NONE, 'Return integer whether a new version is available or not (implies --check-only). Useful for programmatic use.');
|
||||||
|
}
|
||||||
|
|
||||||
|
protected function execute(InputInterface $input, OutputInterface $output)
|
||||||
|
{
|
||||||
|
$result = self::SUCCESS;
|
||||||
|
|
||||||
|
$result = $this->validateRequirements($input, $output);
|
||||||
|
|
||||||
|
require Froxlor::getInstallDir() . '/lib/functions.php';
|
||||||
|
|
||||||
|
// version check
|
||||||
|
$newversionavail = false;
|
||||||
|
if ($result == self::SUCCESS) {
|
||||||
|
try {
|
||||||
|
$aucheck = AutoUpdate::checkVersion();
|
||||||
|
|
||||||
|
if ($aucheck == 1) {
|
||||||
|
if ($input->getOption('integer-return')) {
|
||||||
|
$output->write(1);
|
||||||
|
return self::SUCCESS;
|
||||||
|
}
|
||||||
|
// there is a new version
|
||||||
|
$text = lng('admin.newerversionavailable') . ' ' . lng('admin.newerversiondetails', [AutoUpdate::getFromResult('version'), Froxlor::VERSION]);
|
||||||
|
$text = str_replace("<br/>", " ", $text);
|
||||||
|
$text = str_replace("<b>", "<info>", $text);
|
||||||
|
$text = str_replace("</b>", "</info>", $text);
|
||||||
|
$newversionavail = true;
|
||||||
|
$output->writeln('<comment>' . $text . '</>');
|
||||||
|
$result = self::SUCCESS;
|
||||||
|
} else if ($aucheck < 0 || $aucheck > 1) {
|
||||||
|
if ($input->getOption('integer-return')) {
|
||||||
|
$output->write(-1);
|
||||||
|
return self::INVALID;
|
||||||
|
}
|
||||||
|
// errors
|
||||||
|
if ($aucheck < 0) {
|
||||||
|
$output->writeln('<error>' . AutoUpdate::getLastError() . '</>');
|
||||||
|
} else {
|
||||||
|
$errmsg = 'error.autoupdate_' . $aucheck;
|
||||||
|
if ($aucheck == 3) {
|
||||||
|
$errmsg = 'error.customized_version';
|
||||||
|
}
|
||||||
|
$output->writeln('<error>' . lng($errmsg) . '</>');
|
||||||
|
}
|
||||||
|
$result = self::INVALID;
|
||||||
|
} else {
|
||||||
|
if ($input->getOption('integer-return')) {
|
||||||
|
$output->write(0);
|
||||||
|
return self::SUCCESS;
|
||||||
|
}
|
||||||
|
// no new version
|
||||||
|
$output->writeln('<info>' . lng('update.noupdatesavail') . '</>');
|
||||||
|
$result = self::SUCCESS;
|
||||||
|
}
|
||||||
|
} catch (Exception $e) {
|
||||||
|
if ($input->getOption('integer-return')) {
|
||||||
|
$output->write(-1);
|
||||||
|
return self::FAILURE;
|
||||||
|
}
|
||||||
|
$output->writeln('<error>' . $e->getMessage() . '</>');
|
||||||
|
$result = self::FAILURE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// if there's a newer version, proceed
|
||||||
|
if ($result == self::SUCCESS && $newversionavail) {
|
||||||
|
|
||||||
|
// check whether we only wanted to check
|
||||||
|
if ($input->getOption('check-only')) {
|
||||||
|
//$output->writeln('<comment>Not proceeding as "check-only" is specified</>');
|
||||||
|
return $result;
|
||||||
|
} else {
|
||||||
|
$yestoall = $input->getOption('yes-to-all') !== false;
|
||||||
|
|
||||||
|
$helper = $this->getHelper('question');
|
||||||
|
|
||||||
|
// ask download
|
||||||
|
$question = new ConfirmationQuestion('Download newer version? [no] ', false, '/^(y|j)/i');
|
||||||
|
if ($yestoall || $helper->ask($input, $output, $question)) {
|
||||||
|
// do download
|
||||||
|
$output->writeln('Downloading...');
|
||||||
|
$audl = AutoUpdate::downloadZip(AutoUpdate::getFromResult('version'));
|
||||||
|
if (!is_numeric($audl)) {
|
||||||
|
// ask extract
|
||||||
|
$question = new ConfirmationQuestion('Extract downloaded archive? [no] ', false, '/^(y|j)/i');
|
||||||
|
if ($yestoall || $helper->ask($input, $output, $question)) {
|
||||||
|
// do extract
|
||||||
|
$output->writeln('Extracting...');
|
||||||
|
$auex = AutoUpdate::extractZip($audl);
|
||||||
|
if ($auex == 0) {
|
||||||
|
$output->writeln("<info>Froxlor files updated successfully.</>");
|
||||||
|
$result = self::SUCCCESS;
|
||||||
|
$question = new ConfirmationQuestion('Update database? [no] ', false, '/^(y|j)/i');
|
||||||
|
if ($yestoall || $helper->ask($input, $output, $question)) {
|
||||||
|
$result = $this->updateDatabase();
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$errmsg = 'error.autoupdate_' . $auex;
|
||||||
|
$output->writeln('<error>' . lng($errmsg) . '</>');
|
||||||
|
$result = self::FAILURE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$errmsg = 'error.autoupdate_' . $audl;
|
||||||
|
$output->writeln('<error>' . lng($errmsg) . '</>');
|
||||||
|
$result = self::FAILURE;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function updateDatabase()
|
||||||
|
{
|
||||||
|
include_once Froxlor::getInstallDir() . '/lib/tables.inc.php';
|
||||||
|
define('_CRON_UPDATE', 1);
|
||||||
|
ob_start([
|
||||||
|
'this',
|
||||||
|
'cleanUpdateOutput'
|
||||||
|
]);
|
||||||
|
include_once Froxlor::getInstallDir() . '/install/updatesql.php';
|
||||||
|
ob_end_flush();
|
||||||
|
return self::SUCCCESS;
|
||||||
|
}
|
||||||
|
|
||||||
|
private function cleanUpdateOutput($buffer)
|
||||||
|
{
|
||||||
|
return strip_tags(preg_replace("/<br\W*?\/>/", "\n", $buffer));
|
||||||
|
}
|
||||||
|
}
|
||||||
173
lib/Froxlor/Install/AutoUpdate.php
Normal file
173
lib/Froxlor/Install/AutoUpdate.php
Normal file
@@ -0,0 +1,173 @@
|
|||||||
|
<?php
|
||||||
|
|
||||||
|
/**
|
||||||
|
* This file is part of the Froxlor project.
|
||||||
|
* Copyright (c) 2010 the Froxlor Team (see authors).
|
||||||
|
*
|
||||||
|
* This program is free software; you can redistribute it and/or
|
||||||
|
* modify it under the terms of the GNU General Public License
|
||||||
|
* as published by the Free Software Foundation; either version 2
|
||||||
|
* of the License, or (at your option) any later version.
|
||||||
|
*
|
||||||
|
* This program is distributed in the hope that it will be useful,
|
||||||
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||||
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||||
|
* GNU General Public License for more details.
|
||||||
|
*
|
||||||
|
* You should have received a copy of the GNU General Public License
|
||||||
|
* along with this program; if not, you can also view it online at
|
||||||
|
* https://files.froxlor.org/misc/COPYING.txt
|
||||||
|
*
|
||||||
|
* @copyright the authors
|
||||||
|
* @author Froxlor team <team@froxlor.org>
|
||||||
|
* @license https://files.froxlor.org/misc/COPYING.txt GPLv2
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace Froxlor\Install;
|
||||||
|
|
||||||
|
use Exception;
|
||||||
|
use Froxlor\Froxlor;
|
||||||
|
use Froxlor\Settings;
|
||||||
|
use Froxlor\Http\HttpClient;
|
||||||
|
|
||||||
|
class AutoUpdate
|
||||||
|
{
|
||||||
|
// define update-uri
|
||||||
|
const UPDATE_URI = "https://version.froxlor.org/froxlor/api2/";
|
||||||
|
const RELEASE_URI = "https://autoupdate.froxlor.org/froxlor-{version}.zip";
|
||||||
|
const CHECKSUM_URI = "https://autoupdate.froxlor.org/froxlor-{version}.zip.sha256";
|
||||||
|
|
||||||
|
const ERR_NOZIPEXT = 2;
|
||||||
|
const ERR_COULDNOTSTORE = 4;
|
||||||
|
const ERR_ZIPNOTFOUND = 7;
|
||||||
|
const ERR_COULDNOTEXTRACT = 8;
|
||||||
|
const ERR_CHKSUM_MISMATCH = 9;
|
||||||
|
const ERR_MINPHP = 10;
|
||||||
|
|
||||||
|
private static $latestversion = "";
|
||||||
|
|
||||||
|
private static $lasterror = "";
|
||||||
|
|
||||||
|
/**
|
||||||
|
* returns status about whether there is a newer version
|
||||||
|
*
|
||||||
|
* 0 = no new version available
|
||||||
|
* 1 = new version available
|
||||||
|
* -1 = remote error message
|
||||||
|
* >1 = local error message
|
||||||
|
*
|
||||||
|
* @return int
|
||||||
|
*/
|
||||||
|
public static function checkVersion(): int
|
||||||
|
{
|
||||||
|
$result = self::checkPrerequisites();
|
||||||
|
|
||||||
|
if ($result == 0) {
|
||||||
|
try {
|
||||||
|
$channel = '';
|
||||||
|
if (Settings::Get('system.update_channel') == 'testing') {
|
||||||
|
$channel = '/testing';
|
||||||
|
}
|
||||||
|
$latestversion = HttpClient::urlGet(self::UPDATE_URI . Froxlor::VERSION . $channel, true, 3);
|
||||||
|
} catch (Exception $e) {
|
||||||
|
throw new Exception("Version-check currently unavailable, please try again later", 504);
|
||||||
|
}
|
||||||
|
|
||||||
|
self::$latestversion = json_decode($latestversion, true);
|
||||||
|
|
||||||
|
if (self::$latestversion) {
|
||||||
|
if (!empty(self::$latestversion['error']) && self::$latestversion['error']) {
|
||||||
|
$result = -1;
|
||||||
|
self::$lasterror = self::$latestversion['message'];
|
||||||
|
} else if (isset(self::$latestversion['has_latest']) && self::$latestversion['has_latest'] == false) {
|
||||||
|
$result = 1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function downloadZip(string $newversion)
|
||||||
|
{
|
||||||
|
// define files to get
|
||||||
|
$toLoad = str_replace('{version}', $newversion, self::RELEASE_URI);
|
||||||
|
$toCheck = str_replace('{version}', $newversion, self::CHECKSUM_URI);
|
||||||
|
|
||||||
|
// check for local destination folder
|
||||||
|
if (!is_dir(Froxlor::getInstallDir() . '/updates/')) {
|
||||||
|
mkdir(Froxlor::getInstallDir() . '/updates/');
|
||||||
|
}
|
||||||
|
|
||||||
|
// name archive
|
||||||
|
$localArchive = Froxlor::getInstallDir() . '/updates/' . basename($toLoad);
|
||||||
|
|
||||||
|
// remove old archive
|
||||||
|
if (file_exists($localArchive)) {
|
||||||
|
@unlink($localArchive);
|
||||||
|
}
|
||||||
|
|
||||||
|
// get archive data
|
||||||
|
try {
|
||||||
|
HttpClient::fileGet($toLoad, $localArchive);
|
||||||
|
} catch (Exception $e) {
|
||||||
|
return self::ERR_COULDNOTSTORE;
|
||||||
|
}
|
||||||
|
|
||||||
|
// validate the integrity of the downloaded file
|
||||||
|
$_shouldsum = HttpClient::urlGet($toCheck);
|
||||||
|
if (!empty($_shouldsum)) {
|
||||||
|
$_t = explode(" ", $_shouldsum);
|
||||||
|
$shouldsum = $_t[0];
|
||||||
|
} else {
|
||||||
|
$shouldsum = null;
|
||||||
|
}
|
||||||
|
$filesum = hash_file('sha256', $localArchive);
|
||||||
|
|
||||||
|
if ($filesum != $shouldsum) {
|
||||||
|
return self::ERR_CHKSUM_MISMATCH;
|
||||||
|
}
|
||||||
|
|
||||||
|
return basename($localArchive);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function extractZip(string $localArchive): int
|
||||||
|
{
|
||||||
|
if (!file_exists($localArchive)) {
|
||||||
|
return self::ERR_ZIPNOTFOUND;
|
||||||
|
}
|
||||||
|
// decompress from zip
|
||||||
|
$zip = new ZipArchive();
|
||||||
|
$res = $zip->open($localArchive);
|
||||||
|
if ($res === true) {
|
||||||
|
$zip->extractTo(Froxlor::getInstallDir());
|
||||||
|
$zip->close();
|
||||||
|
// success - remove unused archive
|
||||||
|
@unlink($localArchive);
|
||||||
|
// wait a bit before we redirect to be sure
|
||||||
|
sleep(3);
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
return self::ERR_COULDNOTEXTRACT;
|
||||||
|
}
|
||||||
|
|
||||||
|
private static function checkPrerequisites(): int
|
||||||
|
{
|
||||||
|
if (!extension_loaded('zip')) {
|
||||||
|
return self::ERR_NOZIPEXT;
|
||||||
|
}
|
||||||
|
if (version_compare("7.4.0", PHP_VERSION, ">=")) {
|
||||||
|
return self::ERR_MINPHP;
|
||||||
|
}
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getLastError(): string
|
||||||
|
{
|
||||||
|
return self::$lasterror ?? "";
|
||||||
|
}
|
||||||
|
|
||||||
|
public static function getFromResult(string $index)
|
||||||
|
{
|
||||||
|
return self::$latestversion[$index] ?? "";
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -329,7 +329,9 @@ return [
|
|||||||
'accountdata' => 'Benutzerdaten',
|
'accountdata' => 'Benutzerdaten',
|
||||||
'contactdata' => 'Kontaktdaten',
|
'contactdata' => 'Kontaktdaten',
|
||||||
'servicedata' => 'Dienstleistungsdaten',
|
'servicedata' => 'Dienstleistungsdaten',
|
||||||
'newerversionavailable' => 'Eine neuere Version von Froxlor wurde veröffentlicht',
|
'newerversionavailable' => 'Eine neuere Version von Froxlor wurde veröffentlicht.',
|
||||||
|
'newerversiondetails' => 'Jetzt auf Version <b>%s</b> aktualisieren?<br/>(Aktuelle Version ist: %s)',
|
||||||
|
'extractdownloadedzip' => 'Heruntergeladenes Archiv "%s" entpacken?',
|
||||||
'cron' => [
|
'cron' => [
|
||||||
'cronsettings' => 'Cronjob-Einstellungen',
|
'cronsettings' => 'Cronjob-Einstellungen',
|
||||||
'add' => 'Cronjob hinzufügen',
|
'add' => 'Cronjob hinzufügen',
|
||||||
@@ -2005,6 +2007,10 @@ Vielen Dank, Ihr Administrator',
|
|||||||
'title' => 'Pfad zu acme.sh',
|
'title' => 'Pfad zu acme.sh',
|
||||||
'description' => 'Installationspfad zu acme.sh, inklusive acme.sh Script<br>Standard ist <b>/root/.acme.sh/acme.sh</b>',
|
'description' => 'Installationspfad zu acme.sh, inklusive acme.sh Script<br>Standard ist <b>/root/.acme.sh/acme.sh</b>',
|
||||||
],
|
],
|
||||||
|
'update_channel' => [
|
||||||
|
'title' => 'froxlor Update Kanal',
|
||||||
|
'description' => 'Wähle den bevorzugten Update Kanal. Standard ist "stable"',
|
||||||
|
],
|
||||||
],
|
],
|
||||||
'spf' => [
|
'spf' => [
|
||||||
'use_spf' => 'Aktiviere SPF für Domains?',
|
'use_spf' => 'Aktiviere SPF für Domains?',
|
||||||
|
|||||||
@@ -331,7 +331,9 @@ return [
|
|||||||
'accountdata' => 'Account Data',
|
'accountdata' => 'Account Data',
|
||||||
'contactdata' => 'Contact Data',
|
'contactdata' => 'Contact Data',
|
||||||
'servicedata' => 'Service Data',
|
'servicedata' => 'Service Data',
|
||||||
'newerversionavailable' => 'There is a newer version of Froxlor available',
|
'newerversionavailable' => 'There is a newer version of Froxlor available.',
|
||||||
|
'newerversiondetails' => 'Update to version <b>%s</b> now?<br/>(Your current version is: %s)',
|
||||||
|
'extractdownloadedzip' => 'Extract downloaded archive "%s"?',
|
||||||
'cron' => [
|
'cron' => [
|
||||||
'cronsettings' => 'Cronjob settings',
|
'cronsettings' => 'Cronjob settings',
|
||||||
'add' => 'Add cronjob',
|
'add' => 'Add cronjob',
|
||||||
@@ -2377,6 +2379,12 @@ Yours sincerely, your administrator',
|
|||||||
'title' => 'Path to acme.sh',
|
'title' => 'Path to acme.sh',
|
||||||
'description' => 'Set this to where acme.sh is installed to, including the acme.sh script<br>Default is <b>/root/.acme.sh/acme.sh</b>',
|
'description' => 'Set this to where acme.sh is installed to, including the acme.sh script<br>Default is <b>/root/.acme.sh/acme.sh</b>',
|
||||||
],
|
],
|
||||||
|
'update_channel' => [
|
||||||
|
'title' => 'froxlor update-channel',
|
||||||
|
'description' => 'Select the update channel of froxlor. Default is "stable"',
|
||||||
|
],
|
||||||
|
'uc_stable' => 'stable',
|
||||||
|
'uc_testing' => 'testing',
|
||||||
],
|
],
|
||||||
'spf' => [
|
'spf' => [
|
||||||
'use_spf' => 'Activate SPF for domains?',
|
'use_spf' => 'Activate SPF for domains?',
|
||||||
|
|||||||
Reference in New Issue
Block a user