123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151 |
- <?php
- // This file is part of GNU social - https://www.gnu.org/software/social
- //
- // GNU social 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.
- //
- // GNU social 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 GNU social. If not, see <http://www.gnu.org/licenses/>.
- /**
- * GNU social queue-manager-on-visit class
- *
- * Will run events for a certain time, or until finished.
- *
- * Configure remote key if wanted with $config['opportunisticqm']['qmkey'] and
- * use with /main/runqueue?qmkey=abc123
- *
- * @category Cron
- * @package GNUsocial
- * @author Mikael Nordfeldth <mmn@hethane.se>
- * @copyright 2013 Free Software Foundation, Inc http://www.fsf.org
- * @license https://www.gnu.org/licenses/agpl.html GNU AGPL v3 or later
- */
- defined('GNUSOCIAL') || die();
- class OpportunisticQueueManager extends QueueManager
- {
- protected $qmkey = false;
- protected $max_execution_time = null;
- protected $max_execution_margin = null; // margin to PHP's max_execution_time
- protected $max_queue_items = null;
- protected $start_cpu_time = null;
- protected $handled_items = 0;
- protected $verbosity = null;
- // typically just used for the /main/cron action, only used if php.ini max_execution_time is 0
- const MAXEXECTIME = 20;
- public function __construct(array $args = [])
- {
- foreach (get_class_vars(get_class($this)) as $key=>$val) {
- if (array_key_exists($key, $args)) {
- $this->$key = $args[$key];
- }
- }
- $this->verifyKey();
- if (is_null($this->start_cpu_time)) {
- $this->start_cpu_time = hrtime(true);
- }
- if (is_null($this->max_execution_time)) {
- $this->max_execution_time = ini_get('max_execution_time') ?: self::MAXEXECTIME;
- }
- if (is_null($this->max_execution_margin)) {
- // think PHP's max exec time, minus this value to have time for timeouts etc.
- $this->max_execution_margin = common_config('http', 'connect_timeout') + 1;
- }
- return parent::__construct();
- }
- protected function verifyKey()
- {
- if ($this->qmkey !== common_config('opportunisticqm', 'qmkey')) {
- throw new RunQueueBadKeyException($this->qmkey);
- }
- }
- public function enqueue($object, $key)
- {
- // Nothing to do, should never get called
- throw new ServerException('OpportunisticQueueManager::enqueue should never be called');
- }
- public function canContinue()
- {
- $time_passed = (hrtime(true) - $this->start_cpu_time) / 1000000000;
- // Only continue if limit values are sane
- if ($time_passed <= 0 && (!is_null($this->max_queue_items) && $this->max_queue_items <= 0)) {
- return false;
- }
- // If too much time has passed, stop
- if ($time_passed >= $this->max_execution_time ||
- $time_passed > ini_get('max_execution_time') - $this->max_execution_margin) {
- return false;
- }
- // If we have a max-item-limit, check if it has been passed
- if (!is_null($this->max_queue_items) && $this->handled_items >= $this->max_queue_items) {
- return false;
- }
- return true;
- }
- public function poll()
- {
- $qm = $this->get();
- if ($qm->pollInterval() <= 0 && ! $qm instanceof UnQueueManager) {
- // Special case for UnQueueManager as it is a default plugin
- // and does not require queues, since it does everything immediately
- throw new ServerException('OpportunisticQM cannot work together' .
- 'with queues that do not implement polling');
- }
- ++$this->handled_items;
- return $qm->poll();
- }
- /**
- * Takes care of running through the queue items, returning when
- * the limits setup in __construct are met.
- *
- * @return true on workqueue finished, false if there are still items in the queue
- */
- public function runQueue()
- {
- while ($this->canContinue()) {
- if (!$this->poll()) {
- // Out of work
- return true;
- }
- }
- if ($this->handled_items > 0) {
- common_debug('Opportunistic queue manager passed execution time/item ' .
- 'handling limit without being out of work.');
- return true;
- } elseif ($this->verbosity > 1) {
- $time_passed = (hrtime(true) - $this->start_cpu_time) / 1000000000;
- common_debug('Opportunistic queue manager did not have time to start ' .
- "on this action (max: {$this->max_execution_time}" .
- " exceeded: {$time_passed}).");
- }
- return false;
- }
- }
|