You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

195 lines
6.2 KiB

9 years ago
9 years ago
13 years ago
13 years ago
13 years ago
13 years ago
13 years ago
13 years ago
12 years ago
11 years ago
13 years ago
11 years ago
  1. <?php
  2. /**
  3. * @copyright Copyright (c) 2016, ownCloud, Inc.
  4. *
  5. * @author Artem Sidorenko <artem@posteo.de>
  6. * @author Christopher Schäpers <kondou@ts.unde.re>
  7. * @author Christoph Wurst <christoph@winzerhof-wurst.at>
  8. * @author Daniel Kesselberg <mail@danielkesselberg.de>
  9. * @author hoellen <dev@hoellen.eu>
  10. * @author J0WI <J0WI@users.noreply.github.com>
  11. * @author Jakob Sack <mail@jakobsack.de>
  12. * @author Joas Schilling <coding@schilljs.com>
  13. * @author Jörn Friedrich Dreyer <jfd@butonic.de>
  14. * @author Ko- <k.stoffelen@cs.ru.nl>
  15. * @author Michael Kuhn <michael@ikkoku.de>
  16. * @author Morris Jobke <hey@morrisjobke.de>
  17. * @author Oliver Kohl D.Sc. <oliver@kohl.bz>
  18. * @author Robin Appelman <robin@icewind.nl>
  19. * @author Roeland Jago Douma <roeland@famdouma.nl>
  20. * @author Steffen Lindner <mail@steffen-lindner.de>
  21. * @author Thomas Müller <thomas.mueller@tmit.eu>
  22. * @author Vincent Petry <vincent@nextcloud.com>
  23. * @author Stephen Michel <git@smichel.me>
  24. *
  25. * @license AGPL-3.0
  26. *
  27. * This code is free software: you can redistribute it and/or modify
  28. * it under the terms of the GNU Affero General Public License, version 3,
  29. * as published by the Free Software Foundation.
  30. *
  31. * This program is distributed in the hope that it will be useful,
  32. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  33. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  34. * GNU Affero General Public License for more details.
  35. *
  36. * You should have received a copy of the GNU Affero General Public License, version 3,
  37. * along with this program. If not, see <http://www.gnu.org/licenses/>
  38. *
  39. */
  40. require_once __DIR__ . '/lib/versioncheck.php';
  41. try {
  42. require_once __DIR__ . '/lib/base.php';
  43. if (\OCP\Util::needUpgrade()) {
  44. \OC::$server->getLogger()->debug('Update required, skipping cron', ['app' => 'cron']);
  45. exit;
  46. }
  47. if ((bool) \OC::$server->getSystemConfig()->getValue('maintenance', false)) {
  48. \OC::$server->getLogger()->debug('We are in maintenance mode, skipping cron', ['app' => 'cron']);
  49. exit;
  50. }
  51. // load all apps to get all api routes properly setup
  52. OC_App::loadApps();
  53. \OC::$server->getSession()->close();
  54. // initialize a dummy memory session
  55. $session = new \OC\Session\Memory('');
  56. $cryptoWrapper = \OC::$server->getSessionCryptoWrapper();
  57. $session = $cryptoWrapper->wrapSession($session);
  58. \OC::$server->setSession($session);
  59. $logger = \OC::$server->getLogger();
  60. $config = \OC::$server->getConfig();
  61. $tempManager = \OC::$server->getTempManager();
  62. // Don't do anything if Nextcloud has not been installed
  63. if (!$config->getSystemValue('installed', false)) {
  64. exit(0);
  65. }
  66. $tempManager->cleanOld();
  67. // Exit if background jobs are disabled!
  68. $appMode = $config->getAppValue('core', 'backgroundjobs_mode', 'ajax');
  69. if ($appMode === 'none') {
  70. if (OC::$CLI) {
  71. echo 'Background Jobs are disabled!' . PHP_EOL;
  72. } else {
  73. OC_JSON::error(['data' => ['message' => 'Background jobs disabled!']]);
  74. }
  75. exit(1);
  76. }
  77. if (OC::$CLI) {
  78. // set to run indefinitely if needed
  79. if (strpos(@ini_get('disable_functions'), 'set_time_limit') === false) {
  80. @set_time_limit(0);
  81. }
  82. // the cron job must be executed with the right user
  83. if (!function_exists('posix_getuid')) {
  84. echo "The posix extensions are required - see https://www.php.net/manual/en/book.posix.php" . PHP_EOL;
  85. exit(1);
  86. }
  87. $user = posix_getuid();
  88. $configUser = fileowner(OC::$configDir . 'config.php');
  89. if ($user !== $configUser) {
  90. echo "Console has to be executed with the user that owns the file config/config.php" . PHP_EOL;
  91. echo "Current user id: " . $user . PHP_EOL;
  92. echo "Owner id of config.php: " . $configUser . PHP_EOL;
  93. exit(1);
  94. }
  95. // We call Nextcloud from the CLI (aka cron)
  96. if ($appMode !== 'cron') {
  97. $config->setAppValue('core', 'backgroundjobs_mode', 'cron');
  98. }
  99. // Low-load hours
  100. $onlyTimeSensitive = false;
  101. $startHour = $config->getSystemValueInt('maintenance_window_start', 100);
  102. if ($startHour <= 23) {
  103. $date = new \DateTime('now', new \DateTimeZone('UTC'));
  104. $currentHour = (int) $date->format('G');
  105. $endHour = $startHour + 4;
  106. if ($startHour <= 20) {
  107. // Start time: 01:00
  108. // End time: 05:00
  109. // Only run sensitive tasks when it's before the start or after the end
  110. $onlyTimeSensitive = $currentHour < $startHour || $currentHour > $endHour;
  111. } else {
  112. // Start time: 23:00
  113. // End time: 03:00
  114. $endHour -= 24; // Correct the end time from 27:00 to 03:00
  115. // Only run sensitive tasks when it's after the end and before the start
  116. $onlyTimeSensitive = $currentHour > $endHour && $currentHour < $startHour;
  117. }
  118. }
  119. // Work
  120. $jobList = \OC::$server->getJobList();
  121. // We only ask for jobs for 14 minutes, because after 5 minutes the next
  122. // system cron task should spawn and we want to have at most three
  123. // cron jobs running in parallel.
  124. $endTime = time() + 14 * 60;
  125. $executedJobs = [];
  126. while ($job = $jobList->getNext($onlyTimeSensitive)) {
  127. if (isset($executedJobs[$job->getId()])) {
  128. $jobList->unlockJob($job);
  129. break;
  130. }
  131. $logger->debug('CLI cron call has selected job with ID ' . strval($job->getId()), ['app' => 'cron']);
  132. $job->execute($jobList, $logger);
  133. // clean up after unclean jobs
  134. \OC_Util::tearDownFS();
  135. $tempManager->clean();
  136. $jobList->setLastJob($job);
  137. $executedJobs[$job->getId()] = true;
  138. unset($job);
  139. if (time() > $endTime) {
  140. break;
  141. }
  142. }
  143. } else {
  144. // We call cron.php from some website
  145. if ($appMode === 'cron') {
  146. // Cron is cron :-P
  147. OC_JSON::error(['data' => ['message' => 'Backgroundjobs are using system cron!']]);
  148. } else {
  149. // Work and success :-)
  150. $jobList = \OC::$server->getJobList();
  151. $job = $jobList->getNext();
  152. if ($job != null) {
  153. $logger->debug('WebCron call has selected job with ID ' . strval($job->getId()), ['app' => 'cron']);
  154. $job->execute($jobList, $logger);
  155. $jobList->setLastJob($job);
  156. }
  157. OC_JSON::success();
  158. }
  159. }
  160. // Log the successful cron execution
  161. $config->setAppValue('core', 'lastcron', time());
  162. exit();
  163. } catch (Exception $ex) {
  164. \OC::$server->getLogger()->logException($ex, ['app' => 'cron']);
  165. echo $ex . PHP_EOL;
  166. exit(1);
  167. } catch (Error $ex) {
  168. \OC::$server->getLogger()->logException($ex, ['app' => 'cron']);
  169. echo $ex . PHP_EOL;
  170. exit(1);
  171. }