2018-10-05 04:08:03 +03:00
|
|
|
<?php
|
|
|
|
/**
|
|
|
|
* @copyright Copyright (c) 2018, Patrik Kernstock <info@pkern.at>
|
|
|
|
*
|
2020-04-29 12:57:22 +03:00
|
|
|
* @author Christoph Wurst <christoph@winzerhof-wurst.at>
|
2018-10-05 04:08:03 +03:00
|
|
|
* @author Patrik Kernstock <info@pkern.at>
|
2019-12-03 21:57:53 +03:00
|
|
|
* @author Roeland Jago Douma <roeland@famdouma.nl>
|
2018-10-05 04:08:03 +03:00
|
|
|
*
|
2019-12-03 21:57:53 +03:00
|
|
|
* @license GNU AGPL version 3 or any later version
|
2018-10-05 04:08:03 +03:00
|
|
|
*
|
2019-12-03 21:57:53 +03:00
|
|
|
* This program is free software: you can redistribute it and/or modify
|
|
|
|
* it under the terms of the GNU Affero General Public License as
|
|
|
|
* published by the Free Software Foundation, either version 3 of the
|
|
|
|
* License, or (at your option) any later version.
|
2018-10-05 04:08:03 +03:00
|
|
|
*
|
|
|
|
* This program is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
2019-12-03 21:57:53 +03:00
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
2018-10-05 04:08:03 +03:00
|
|
|
* GNU Affero General Public License for more details.
|
|
|
|
*
|
2019-12-03 21:57:53 +03:00
|
|
|
* You should have received a copy of the GNU Affero General Public License
|
|
|
|
* along with this program. If not, see <http://www.gnu.org/licenses/>.
|
2018-10-05 04:08:03 +03:00
|
|
|
*
|
|
|
|
*/
|
|
|
|
|
|
|
|
namespace OC\Core\Command\App;
|
|
|
|
|
|
|
|
use OC\Installer;
|
2018-10-05 20:51:50 +03:00
|
|
|
use OCP\App\IAppManager;
|
2018-10-05 22:09:13 +03:00
|
|
|
use OCP\ILogger;
|
2018-10-05 04:08:03 +03:00
|
|
|
use Stecman\Component\Symfony\Console\BashCompletion\Completion\CompletionAwareInterface;
|
|
|
|
use Stecman\Component\Symfony\Console\BashCompletion\CompletionContext;
|
|
|
|
use Symfony\Component\Console\Command\Command;
|
|
|
|
use Symfony\Component\Console\Input\InputArgument;
|
|
|
|
use Symfony\Component\Console\Input\InputInterface;
|
2019-11-22 22:52:10 +03:00
|
|
|
use Symfony\Component\Console\Input\InputOption;
|
2018-10-05 04:08:03 +03:00
|
|
|
use Symfony\Component\Console\Output\OutputInterface;
|
2019-11-22 22:52:10 +03:00
|
|
|
use Throwable;
|
2018-10-05 04:08:03 +03:00
|
|
|
|
|
|
|
class Remove extends Command implements CompletionAwareInterface {
|
|
|
|
|
2018-10-05 20:51:50 +03:00
|
|
|
/** @var IAppManager */
|
|
|
|
protected $manager;
|
|
|
|
/** @var Installer */
|
|
|
|
private $installer;
|
2018-10-05 22:09:13 +03:00
|
|
|
/** @var ILogger */
|
|
|
|
private $logger;
|
2018-10-05 20:51:50 +03:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param IAppManager $manager
|
|
|
|
* @param Installer $installer
|
2018-10-05 22:09:13 +03:00
|
|
|
* @param ILogger $logger
|
2018-10-05 20:51:50 +03:00
|
|
|
*/
|
2018-10-05 22:09:13 +03:00
|
|
|
public function __construct(IAppManager $manager, Installer $installer, ILogger $logger) {
|
2018-10-05 20:51:50 +03:00
|
|
|
parent::__construct();
|
|
|
|
$this->manager = $manager;
|
|
|
|
$this->installer = $installer;
|
2018-10-05 22:09:13 +03:00
|
|
|
$this->logger = $logger;
|
2018-10-05 20:51:50 +03:00
|
|
|
}
|
|
|
|
|
2018-10-05 04:08:03 +03:00
|
|
|
protected function configure() {
|
|
|
|
$this
|
|
|
|
->setName('app:remove')
|
|
|
|
->setDescription('remove an app')
|
|
|
|
->addArgument(
|
|
|
|
'app-id',
|
|
|
|
InputArgument::REQUIRED,
|
|
|
|
'remove the specified app'
|
2018-10-05 13:49:34 +03:00
|
|
|
)
|
|
|
|
->addOption(
|
|
|
|
'keep-data',
|
|
|
|
null,
|
|
|
|
InputOption::VALUE_NONE,
|
|
|
|
'keep app data and do not remove them'
|
2018-10-05 04:08:03 +03:00
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2020-06-26 15:54:51 +03:00
|
|
|
protected function execute(InputInterface $input, OutputInterface $output): int {
|
2018-10-05 04:08:03 +03:00
|
|
|
$appId = $input->getArgument('app-id');
|
|
|
|
|
2018-10-05 20:51:50 +03:00
|
|
|
// Check if the app is installed
|
2018-10-05 04:08:03 +03:00
|
|
|
if (!\OC_App::getAppPath($appId)) {
|
|
|
|
$output->writeln($appId . ' is not installed');
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2018-10-05 20:51:50 +03:00
|
|
|
// Removing shipped apps is not possible, therefore we pre-check that
|
|
|
|
// before trying to remove it
|
|
|
|
if ($this->manager->isShipped($appId)) {
|
|
|
|
$output->writeln($appId . ' could not be removed as it is a shipped app');
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
// If we want to keep the data of the app, we simply don't disable it here.
|
|
|
|
// App uninstall tasks are being executed when disabled. More info: PR #11627.
|
2018-10-05 13:49:34 +03:00
|
|
|
if (!$input->getOption('keep-data')) {
|
|
|
|
try {
|
2018-10-05 20:51:50 +03:00
|
|
|
$this->manager->disableApp($appId);
|
2018-10-05 13:49:34 +03:00
|
|
|
$output->writeln($appId . ' disabled');
|
2020-04-10 15:19:56 +03:00
|
|
|
} catch (Throwable $e) {
|
2018-10-10 19:28:26 +03:00
|
|
|
$output->writeln('<error>Error: ' . $e->getMessage() . '</error>');
|
2018-10-05 22:09:13 +03:00
|
|
|
$this->logger->logException($e, [
|
|
|
|
'app' => 'CLI',
|
|
|
|
'level' => ILogger::ERROR
|
|
|
|
]);
|
2018-10-05 13:49:34 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-05 20:51:50 +03:00
|
|
|
// Let's try to remove the app...
|
2018-10-05 04:08:03 +03:00
|
|
|
try {
|
2018-10-05 20:51:50 +03:00
|
|
|
$result = $this->installer->removeApp($appId);
|
2020-04-10 15:19:56 +03:00
|
|
|
} catch (Throwable $e) {
|
2018-10-10 19:28:26 +03:00
|
|
|
$output->writeln('<error>Error: ' . $e->getMessage() . '</error>');
|
2018-10-05 22:09:13 +03:00
|
|
|
$this->logger->logException($e, [
|
|
|
|
'app' => 'CLI',
|
|
|
|
'level' => ILogger::ERROR
|
|
|
|
]);
|
2018-10-05 04:08:03 +03:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-04-10 15:19:56 +03:00
|
|
|
if ($result === false) {
|
2018-10-05 04:08:03 +03:00
|
|
|
$output->writeln($appId . ' could not be removed');
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2020-04-30 10:43:33 +03:00
|
|
|
$appVersion = \OC_App::getAppVersion($appId);
|
|
|
|
$output->writeln($appId . ' ' . $appVersion . ' removed');
|
2018-10-05 04:08:03 +03:00
|
|
|
|
|
|
|
return 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $optionName
|
|
|
|
* @param CompletionContext $context
|
|
|
|
* @return string[]
|
|
|
|
*/
|
|
|
|
public function completeOptionValues($optionName, CompletionContext $context) {
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @param string $argumentName
|
|
|
|
* @param CompletionContext $context
|
|
|
|
* @return string[]
|
|
|
|
*/
|
|
|
|
public function completeArgumentValues($argumentName, CompletionContext $context) {
|
|
|
|
if ($argumentName === 'app-id') {
|
|
|
|
return \OC_App::getAllApps();
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
}
|
|
|
|
}
|