xref: /template/strap/ComboStrap/Lock.php (revision 0360a8489dc44e35b88aa6f64336e9107f9a1a0f)
1a6d63b89Sgerardnico<?php
2a6d63b89Sgerardnico
3a6d63b89Sgerardniconamespace ComboStrap;
4a6d63b89Sgerardnico
5a6d63b89Sgerardnico
6a6d63b89Sgerardnicouse dokuwiki\Search\Indexer;
7a6d63b89Sgerardnico
8a6d63b89Sgerardnico/**
9a6d63b89Sgerardnico * Adapted from the {@link Indexer::lock()}
10a6d63b89Sgerardnico * because the TaskRunner does not run serially
11a6d63b89Sgerardnico * Only the indexer does
12a6d63b89Sgerardnico * https://forum.dokuwiki.org/d/21044-taskrunner-running-multiple-times-eating-the-memory-lock
13a6d63b89Sgerardnico */
14a6d63b89Sgerardnicoclass Lock
15a6d63b89Sgerardnico{
16a6d63b89Sgerardnico    private string $lockName;
17a6d63b89Sgerardnico    private string $lockFile;
18a6d63b89Sgerardnico    /**
19a6d63b89Sgerardnico     * @var mixed|null
20a6d63b89Sgerardnico     */
21a6d63b89Sgerardnico    private $perm;
22a6d63b89Sgerardnico
23a6d63b89Sgerardnico
24a6d63b89Sgerardnico    /**
25a6d63b89Sgerardnico     * @param string $name
26a6d63b89Sgerardnico     */
27a6d63b89Sgerardnico    public function __construct(string $name)
28a6d63b89Sgerardnico    {
29a6d63b89Sgerardnico        $this->lockName = $name;
30a6d63b89Sgerardnico        global $conf;
31a6d63b89Sgerardnico        $this->lockFile = $conf['lockdir'] . "/_{$this->lockName}.lock";
32a6d63b89Sgerardnico        $this->perm = $conf['dperm'] ?? null;
33a6d63b89Sgerardnico    }
34a6d63b89Sgerardnico
35a6d63b89Sgerardnico    public static function create(string $name): Lock
36a6d63b89Sgerardnico    {
37a6d63b89Sgerardnico        return new Lock($name);
38a6d63b89Sgerardnico    }
39a6d63b89Sgerardnico
40a6d63b89Sgerardnico    /**
41a6d63b89Sgerardnico     * @throws ExceptionTimeOut - with the timeout
42a6d63b89Sgerardnico     */
43a6d63b89Sgerardnico    function acquire()
44a6d63b89Sgerardnico    {
45a6d63b89Sgerardnico        $run = 0;
46a6d63b89Sgerardnico        while (!@mkdir($this->lockFile)) {
47a6d63b89Sgerardnico            usleep(1000);
48a6d63b89Sgerardnico            /**
49a6d63b89Sgerardnico             * Old lock ? More than 5 minutes run
50a6d63b89Sgerardnico             */
51a6d63b89Sgerardnico            if (is_dir($this->lockFile) && (time() - @filemtime($this->lockFile)) > 60 * 5) {
52a6d63b89Sgerardnico                if (!@rmdir($this->lockFile)) {
53a6d63b89Sgerardnico                    throw new ExceptionRuntimeInternal("Removing the lock failed ($this->lockFile)");
54a6d63b89Sgerardnico                }
55a6d63b89Sgerardnico            }
56a6d63b89Sgerardnico            if ($run++ == 5) {
57a6d63b89Sgerardnico                // we waited 5 seconds for that lock
58a6d63b89Sgerardnico                throw new ExceptionTimeOut("Unable to get the lock ($this->lockFile)");
59a6d63b89Sgerardnico            }
60a6d63b89Sgerardnico        }
61a6d63b89Sgerardnico        if ($this->perm) {
62a6d63b89Sgerardnico            chmod($this->lockFile, $this->perm);
63a6d63b89Sgerardnico        }
64a6d63b89Sgerardnico
65a6d63b89Sgerardnico    }
66a6d63b89Sgerardnico
67a6d63b89Sgerardnico    /**
68a6d63b89Sgerardnico     * Release the indexer lock.
69a6d63b89Sgerardnico     *
70a6d63b89Sgerardnico     */
71a6d63b89Sgerardnico    function release()
72a6d63b89Sgerardnico    {
73a6d63b89Sgerardnico        @rmdir($this->lockFile);
74a6d63b89Sgerardnico    }
75a6d63b89Sgerardnico
76*0360a848Sgerardnico    public function isReleased(): bool
77*0360a848Sgerardnico    {
78*0360a848Sgerardnico        return !is_dir($this->lockFile);
79*0360a848Sgerardnico    }
80*0360a848Sgerardnico
81a6d63b89Sgerardnico}
82