friendica/src/Console/Lock.php

203 lines
4.9 KiB
PHP
Raw Normal View History

2019-08-13 15:20:41 -04:00
<?php
2020-02-09 09:45:36 -05:00
/**
2022-01-02 02:27:47 -05:00
* @copyright Copyright (C) 2010-2022, the Friendica project
2020-02-09 09:45:36 -05:00
*
* @license GNU AGPL version 3 or any later version
*
* 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.
*
* 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 Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with this program. If not, see <https://www.gnu.org/licenses/>.
*
*/
2019-08-13 15:20:41 -04:00
namespace Friendica\Console;
use Asika\SimpleConsole\CommandArgsException;
use Friendica\App;
2021-10-26 15:44:29 -04:00
use Friendica\Core\Lock\Capability\ICanLock;
2019-08-13 15:20:41 -04:00
use RuntimeException;
/**
2020-01-19 01:05:23 -05:00
* tool to access the locks from the CLI
2019-08-13 15:20:41 -04:00
*
* With this script you can access the locks of your node from the CLI.
* You can read current locks and set/remove locks.
*/
class Lock extends \Asika\SimpleConsole\Console
{
protected $helpOptions = ['h', 'help', '?'];
/**
* @var App\Mode
*/
private $appMode;
/**
2021-10-26 16:09:11 -04:00
* @var ICanLock
2019-08-13 15:20:41 -04:00
*/
private $lock;
protected function getHelp()
{
$help = <<<HELP
2019-08-15 08:22:29 -04:00
console lock - Manage node locks
2019-08-13 15:20:41 -04:00
Synopsis
bin/console lock list [<prefix>] [-h|--help|-?] [-v]
bin/console lock set <lock> [<timeout> [<ttl>]] [-h|--help|-?] [-v]
bin/console lock del <lock> [-h|--help|-?] [-v]
bin/console lock clear [-h|--help|-?] [-v]
Description
bin/console lock list [<prefix>]
List all locks, optionally filtered by a prefix
bin/console lock set <lock> [<timeout> [<ttl>]]
Sets manually a lock, optionally with the provided TTL (time to live) with a default of five minutes.
bin/console lock del <lock>
Deletes a lock.
bin/console lock clear
Clears all locks
Options
-h|--help|-? Show help information
-v Show more debug information.
HELP;
return $help;
}
2021-10-26 15:44:29 -04:00
public function __construct(App\Mode $appMode, ICanLock $lock, array $argv = null)
2019-08-13 15:20:41 -04:00
{
parent::__construct($argv);
$this->appMode = $appMode;
$this->lock = $lock;
}
2022-07-13 15:09:49 -04:00
protected function doExecute(): int
2019-08-13 15:20:41 -04:00
{
if ($this->getOption('v')) {
$this->out('Executable: ' . $this->executable);
$this->out('Class: ' . __CLASS__);
$this->out('Arguments: ' . var_export($this->args, true));
$this->out('Options: ' . var_export($this->options, true));
}
if (!$this->appMode->has(App\Mode::DBCONFIGAVAILABLE)) {
$this->out('Database isn\'t ready or populated yet, database cache won\'t be available');
}
if ($this->getOption('v')) {
$this->out('Lock Driver Name: ' . $this->lock->getName());
$this->out('Lock Driver Class: ' . get_class($this->lock));
}
switch ($this->getArgument(0)) {
case 'list':
$this->executeList();
break;
case 'set':
$this->executeSet();
break;
case 'del':
$this->executeDel();
break;
case 'clear':
$this->executeClear();
break;
}
if (count($this->args) == 0) {
$this->out($this->getHelp());
return 0;
}
return 0;
}
private function executeList()
{
$prefix = $this->getArgument(1, '');
$keys = $this->lock->getLocks($prefix);
if (empty($prefix)) {
$this->out('Listing all Locks:');
} else {
$this->out('Listing all Locks starting with "' . $prefix . '":');
}
$count = 0;
foreach ($keys as $key) {
$this->out($key);
$count++;
}
$this->out($count . ' locks found');
}
private function executeDel()
{
if (count($this->args) >= 2) {
2019-08-15 08:22:29 -04:00
$lock = $this->getArgument(1);
2019-08-13 15:20:41 -04:00
if ($this->lock->release($lock, true)) {
2019-08-13 15:20:41 -04:00
$this->out(sprintf('Lock \'%s\' released.', $lock));
} else {
$this->out(sprintf('Couldn\'t release Lock \'%s\'', $lock));
}
} else {
throw new CommandArgsException('Too few arguments for del.');
}
}
private function executeSet()
{
if (count($this->args) >= 2) {
2019-08-15 08:22:29 -04:00
$lock = $this->getArgument(1);
2019-08-13 15:20:41 -04:00
$timeout = intval($this->getArgument(2, false));
2019-08-15 08:22:29 -04:00
$ttl = intval($this->getArgument(3, false));
2019-08-13 15:20:41 -04:00
2019-08-15 08:22:29 -04:00
if ($this->lock->isLocked($lock)) {
2019-08-13 15:20:41 -04:00
throw new RuntimeException(sprintf('\'%s\' is already set.', $lock));
}
if (!empty($ttl) && !empty($timeout)) {
$result = $this->lock->acquire($lock, $timeout, $ttl);
2019-08-13 15:20:41 -04:00
} elseif (!empty($timeout)) {
$result = $this->lock->acquire($lock, $timeout);
2019-08-13 15:20:41 -04:00
} else {
$result = $this->lock->acquire($lock);
2019-08-13 15:20:41 -04:00
}
if ($result) {
$this->out(sprintf('Lock \'%s\' acquired.', $lock));
} else {
2019-08-15 08:22:29 -04:00
throw new RuntimeException(sprintf('Unable to lock \'%s\'.', $lock));
2019-08-13 15:20:41 -04:00
}
} else {
throw new CommandArgsException('Too few arguments for set.');
}
}
private function executeClear()
{
$result = $this->lock->releaseAll(true);
if ($result) {
2019-08-15 08:22:29 -04:00
$this->out('Locks successfully cleared.');
2019-08-13 15:20:41 -04:00
} else {
2019-08-15 08:22:29 -04:00
throw new RuntimeException('Unable to clear the locks.');
2019-08-13 15:20:41 -04:00
}
}
}