Компонент Symfony Lock не блокируется - как исправить?

Я недавно обновился до Symfony 3.4.x, реорганизовал LockHandler из-за предупреждения об устаревании и впал в странное поведение.

Код в команде перед рефакторингом:

class FooCommand
{
    protected function configure() { /* ... does not matter ... */ }
    protected function lock() : bool
    {
        $resource = $this->getName();
        $lock     = new \Symfony\Component\Filesystem\LockHandler($resource);

        return $lock->lock();
    }
    protected function execute()
    {
        if (!$this->lock()) return 0;

        // Execute some task
    }
}

И это мешает запускать две команды одновременно - вторая просто заканчивается без выполнения работы. Это хорошо.

Но после предложенного рефакторинга он позволяет запускать много команд одновременно. Это ФАЙЛ. Как предотвратить казнь? Новый код:

class FooCommand
{
    protected function configure() { /* ... does not matter ... */ }
    protected function lock() : bool
    {
        $resource = $this->getName();
        $store    = new \Symfony\Component\Lock\FlockStore(sys_get_temp_dir());
        $factory  = new \Symfony\Component\Lock\Factory($store);
        $lock     = $factory->createLock($resource);

        return $lock->acquire();
    }
    protected function execute()
    {
        if (!$this->lock()) return 0;

        // Execute some task
    }
}

NB # 1: Меня не волнует много серверов или около того, только один экземпляр приложения.

NB # 2: Если процесс был убит, новая команда должна разблокироваться и работать.

1 ответ

Решение

Вы должны использовать черту LockableTrait

    use Symfony\Component\Console\Command\LockableTrait;
    use Symfony\Component\Console\Command\Command

    class FooCommand  extends Command
    {
        use LockableTrait;
.....
protected function execute(InputInterface $input, OutputInterface $output)
    {
        if (!$this->lock()) {
            $output->writeln('The command is already running in another process.');

            return 0;
        }
// If you prefer to wait until the lock is released, use this:
        // $this->lock(null, true);

        // ...

        // if not released explicitly, Symfony releases the lock
        // automatically when the execution of the command ends
        $this->release();

}

Добавляя к ответу Мохамеда, важно назначить идентификатор для шкафчика команд.

В противном случае у блокировки будут проблемы с параллелизмом с другими командами, особенно если у вас разные среды (тестовая, производственная, стадия...). Вы увидите, что команда не выполняется с ожидаемой периодичностью.

Этот идентификатор можно присвоить lock() само заявление.

    use Symfony\Component\Console\Command\LockableTrait;
    use Symfony\Component\Console\Command\Command

    class FooCommand  extends Command
    {
       use LockableTrait;
       .....
       protected function execute(InputInterface $input, OutputInterface $output)
       {
         if (!$this->lock('FooCommand'.getenv('APP_ENV'))) {
            $output->writeln('The command is already running in another process.');
            return 0;
         }

         $this->release();
       }
    }
Другие вопросы по тегам