summaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorCôme Chilliet <91878298+come-nc@users.noreply.github.com>2023-04-25 12:10:43 +0200
committerGitHub <noreply@github.com>2023-04-25 12:10:43 +0200
commit7250f54ca32c1c353470fbc4042ee648e57b760f (patch)
tree15be4b1e070978f81ee9481238e27f5d704e6997
parent77bb867ad5c74216364d593c419419412d27594f (diff)
parent06d6cf4ebcc49d97f89b46e127599b97f8fa1800 (diff)
downloadnextcloud-server-7250f54ca32c1c353470fbc4042ee648e57b760f.tar.gz
nextcloud-server-7250f54ca32c1c353470fbc4042ee648e57b760f.zip
Merge pull request #37835 from nextcloud/feat/background-allow-parallel-runs
feat(BackgroundJobs): Allow preventing parallel runs for a job class
-rw-r--r--lib/composer/composer/autoload_classmap.php1
-rw-r--r--lib/composer/composer/autoload_static.php1
-rw-r--r--lib/private/BackgroundJob/JobList.php28
-rw-r--r--lib/private/SpeechToText/TranscriptionJob.php1
-rw-r--r--lib/public/BackgroundJob/IJobList.php9
-rw-r--r--lib/public/BackgroundJob/IParallelAwareJob.php47
-rw-r--r--lib/public/BackgroundJob/Job.php33
-rw-r--r--tests/lib/BackgroundJob/DummyJobList.php13
-rw-r--r--tests/lib/BackgroundJob/JobListTest.php25
-rw-r--r--tests/lib/BackgroundJob/JobTest.php65
-rw-r--r--tests/lib/BackgroundJob/TestJob.php7
11 files changed, 217 insertions, 13 deletions
diff --git a/lib/composer/composer/autoload_classmap.php b/lib/composer/composer/autoload_classmap.php
index a4bab729592..40fe5e431aa 100644
--- a/lib/composer/composer/autoload_classmap.php
+++ b/lib/composer/composer/autoload_classmap.php
@@ -120,6 +120,7 @@ return array(
'OCP\\AutoloadNotAllowedException' => $baseDir . '/lib/public/AutoloadNotAllowedException.php',
'OCP\\BackgroundJob\\IJob' => $baseDir . '/lib/public/BackgroundJob/IJob.php',
'OCP\\BackgroundJob\\IJobList' => $baseDir . '/lib/public/BackgroundJob/IJobList.php',
+ 'OCP\\BackgroundJob\\IParallelAwareJob' => $baseDir . '/lib/public/BackgroundJob/IParallelAwareJob.php',
'OCP\\BackgroundJob\\Job' => $baseDir . '/lib/public/BackgroundJob/Job.php',
'OCP\\BackgroundJob\\QueuedJob' => $baseDir . '/lib/public/BackgroundJob/QueuedJob.php',
'OCP\\BackgroundJob\\TimedJob' => $baseDir . '/lib/public/BackgroundJob/TimedJob.php',
diff --git a/lib/composer/composer/autoload_static.php b/lib/composer/composer/autoload_static.php
index 8a8de63fc4c..93e91bb1641 100644
--- a/lib/composer/composer/autoload_static.php
+++ b/lib/composer/composer/autoload_static.php
@@ -153,6 +153,7 @@ class ComposerStaticInit749170dad3f5e7f9ca158f5a9f04f6a2
'OCP\\AutoloadNotAllowedException' => __DIR__ . '/../../..' . '/lib/public/AutoloadNotAllowedException.php',
'OCP\\BackgroundJob\\IJob' => __DIR__ . '/../../..' . '/lib/public/BackgroundJob/IJob.php',
'OCP\\BackgroundJob\\IJobList' => __DIR__ . '/../../..' . '/lib/public/BackgroundJob/IJobList.php',
+ 'OCP\\BackgroundJob\\IParallelAwareJob' => __DIR__ . '/../../..' . '/lib/public/BackgroundJob/IParallelAwareJob.php',
'OCP\\BackgroundJob\\Job' => __DIR__ . '/../../..' . '/lib/public/BackgroundJob/Job.php',
'OCP\\BackgroundJob\\QueuedJob' => __DIR__ . '/../../..' . '/lib/public/BackgroundJob/QueuedJob.php',
'OCP\\BackgroundJob\\TimedJob' => __DIR__ . '/../../..' . '/lib/public/BackgroundJob/TimedJob.php',
diff --git a/lib/private/BackgroundJob/JobList.php b/lib/private/BackgroundJob/JobList.php
index 67b736b8dd9..fbeee1f56e9 100644
--- a/lib/private/BackgroundJob/JobList.php
+++ b/lib/private/BackgroundJob/JobList.php
@@ -35,19 +35,23 @@ use OCP\AppFramework\Utility\ITimeFactory;
use OCP\AutoloadNotAllowedException;
use OCP\BackgroundJob\IJob;
use OCP\BackgroundJob\IJobList;
+use OCP\DB\Exception;
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\IConfig;
use OCP\IDBConnection;
+use Psr\Log\LoggerInterface;
class JobList implements IJobList {
protected IDBConnection $connection;
protected IConfig $config;
protected ITimeFactory $timeFactory;
+ protected LoggerInterface $logger;
- public function __construct(IDBConnection $connection, IConfig $config, ITimeFactory $timeFactory) {
+ public function __construct(IDBConnection $connection, IConfig $config, ITimeFactory $timeFactory, LoggerInterface $logger) {
$this->connection = $connection;
$this->config = $config;
$this->timeFactory = $timeFactory;
+ $this->logger = $logger;
}
/**
@@ -382,4 +386,26 @@ class JobList implements IJobList {
->where($query->expr()->eq('id', $query->createNamedParameter($job->getId()), IQueryBuilder::PARAM_INT));
$query->executeStatement();
}
+
+ public function hasReservedJob(?string $className = null): bool {
+ $query = $this->connection->getQueryBuilder();
+ $query->select('*')
+ ->from('jobs')
+ ->where($query->expr()->neq('reserved_at', $query->createNamedParameter(0, IQueryBuilder::PARAM_INT)))
+ ->setMaxResults(1);
+
+ if ($className !== null) {
+ $query->andWhere($query->expr()->eq('class', $query->createNamedParameter($className)));
+ }
+
+ try {
+ $result = $query->executeQuery();
+ $hasReservedJobs = $result->fetch() !== false;
+ $result->closeCursor();
+ return $hasReservedJobs;
+ } catch (Exception $e) {
+ $this->logger->debug('Querying reserved jobs failed', ['exception' => $e]);
+ return false;
+ }
+ }
}
diff --git a/lib/private/SpeechToText/TranscriptionJob.php b/lib/private/SpeechToText/TranscriptionJob.php
index d5cc9ed7c38..8921d52ecd1 100644
--- a/lib/private/SpeechToText/TranscriptionJob.php
+++ b/lib/private/SpeechToText/TranscriptionJob.php
@@ -49,6 +49,7 @@ class TranscriptionJob extends QueuedJob {
private LoggerInterface $logger,
) {
parent::__construct($timeFactory);
+ $this->setAllowParallelRuns(false);
}
diff --git a/lib/public/BackgroundJob/IJobList.php b/lib/public/BackgroundJob/IJobList.php
index e8d0380e604..71faefb8825 100644
--- a/lib/public/BackgroundJob/IJobList.php
+++ b/lib/public/BackgroundJob/IJobList.php
@@ -145,4 +145,13 @@ interface IJobList {
* @since 23.0.0
*/
public function resetBackgroundJob(IJob $job): void;
+
+ /**
+ * Checks whether a job of the passed class is reserved to run
+ *
+ * @param string|null $className
+ * @return bool
+ * @since 27.0.0
+ */
+ public function hasReservedJob(?string $className): bool;
}
diff --git a/lib/public/BackgroundJob/IParallelAwareJob.php b/lib/public/BackgroundJob/IParallelAwareJob.php
new file mode 100644
index 00000000000..30c69e16b5f
--- /dev/null
+++ b/lib/public/BackgroundJob/IParallelAwareJob.php
@@ -0,0 +1,47 @@
+<?php
+
+declare(strict_types=1);
+
+
+/**
+ * @copyright Copyright (c) 2023, Marcel Klehr <mklehr@gmx.net>
+ *
+ * @author Marcel Klehr <mklehr@gmx.net>
+ *
+ * @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 <http://www.gnu.org/licenses/>.
+ *
+ */
+namespace OCP\BackgroundJob;
+
+/**
+ * @since 27.0.0
+ */
+interface IParallelAwareJob {
+ /**
+ * Set this to false to prevent two Jobs from the same class from running in parallel
+ *
+ * @param bool $allow
+ * @return void
+ * @since 27.0.0
+ */
+ public function setAllowParallelRuns(bool $allow): void;
+
+ /**
+ * @return bool
+ * @since 27.0.0
+ */
+ public function getAllowParallelRuns(): bool;
+}
diff --git a/lib/public/BackgroundJob/Job.php b/lib/public/BackgroundJob/Job.php
index d60fb5905c9..455fb3d42e7 100644
--- a/lib/public/BackgroundJob/Job.php
+++ b/lib/public/BackgroundJob/Job.php
@@ -38,11 +38,13 @@ use Psr\Log\LoggerInterface;
*
* @since 15.0.0
*/
-abstract class Job implements IJob {
+abstract class Job implements IJob, IParallelAwareJob {
protected int $id = 0;
protected int $lastRun = 0;
protected $argument;
protected ITimeFactory $time;
+ protected bool $allowParallelRuns = true;
+ private ?ILogger $logger = null;
/**
* @since 15.0.0
@@ -61,6 +63,7 @@ abstract class Job implements IJob {
* @since 15.0.0
*/
public function execute(IJobList $jobList, ILogger $logger = null) {
+ $this->logger = $logger;
$this->start($jobList);
}
@@ -70,7 +73,12 @@ abstract class Job implements IJob {
*/
public function start(IJobList $jobList): void {
$jobList->setLastRun($this);
- $logger = \OCP\Server::get(LoggerInterface::class);
+ $logger = $this->logger ?? \OCP\Server::get(LoggerInterface::class);
+
+ if (!$this->getAllowParallelRuns() && $jobList->hasReservedJob(get_class($this))) {
+ $logger->debug('Skipping ' . get_class($this) . ' job with ID ' . $this->getId() . ' because another job with the same class is already running', ['app' => 'cron']);
+ return;
+ }
try {
$jobStartTime = $this->time->getTime();
@@ -80,7 +88,7 @@ abstract class Job implements IJob {
$logger->debug('Finished ' . get_class($this) . ' job with ID ' . $this->getId() . ' in ' . $timeTaken . ' seconds', ['app' => 'cron']);
$jobList->setExecutionTime($this, $timeTaken);
- } catch (\Exception $e) {
+ } catch (\Throwable $e) {
if ($logger) {
$logger->error('Error while running background job (class: ' . get_class($this) . ', arguments: ' . print_r($this->argument, true) . ')', [
'app' => 'core',
@@ -133,6 +141,25 @@ abstract class Job implements IJob {
}
/**
+ * Set this to false to prevent two Jobs from this class from running in parallel
+ *
+ * @param bool $allow
+ * @return void
+ * @since 27.0.0
+ */
+ public function setAllowParallelRuns(bool $allow): void {
+ $this->allowParallelRuns = $allow;
+ }
+
+ /**
+ * @return bool
+ * @since 27.0.0
+ */
+ public function getAllowParallelRuns(): bool {
+ return $this->allowParallelRuns;
+ }
+
+ /**
* The actual function that is called to run the job
*
* @param $argument
diff --git a/tests/lib/BackgroundJob/DummyJobList.php b/tests/lib/BackgroundJob/DummyJobList.php
index be48259789a..42b69cfbe41 100644
--- a/tests/lib/BackgroundJob/DummyJobList.php
+++ b/tests/lib/BackgroundJob/DummyJobList.php
@@ -21,6 +21,11 @@ class DummyJobList extends \OC\BackgroundJob\JobList {
*/
private array $jobs = [];
+ /**
+ * @var bool[]
+ */
+ private array $reserved = [];
+
private int $last = 0;
public function __construct() {
@@ -135,6 +140,14 @@ class DummyJobList extends \OC\BackgroundJob\JobList {
$job->setLastRun(time());
}
+ public function hasReservedJob(?string $className = null): bool {
+ return $this->reserved[$className ?? ''];
+ }
+
+ public function setHasReservedJob(?string $className, bool $hasReserved): void {
+ $this->reserved[$className ?? ''] = $hasReserved;
+ }
+
public function setExecutionTime(IJob $job, $timeTaken): void {
}
diff --git a/tests/lib/BackgroundJob/JobListTest.php b/tests/lib/BackgroundJob/JobListTest.php
index ea02e1cd8b9..c15e556d5f7 100644
--- a/tests/lib/BackgroundJob/JobListTest.php
+++ b/tests/lib/BackgroundJob/JobListTest.php
@@ -12,6 +12,7 @@ use OCP\AppFramework\Utility\ITimeFactory;
use OCP\BackgroundJob\IJob;
use OCP\DB\QueryBuilder\IQueryBuilder;
use OCP\IConfig;
+use Psr\Log\LoggerInterface;
use Test\TestCase;
/**
@@ -32,6 +33,7 @@ class JobListTest extends TestCase {
/** @var \OCP\AppFramework\Utility\ITimeFactory|\PHPUnit\Framework\MockObject\MockObject */
protected $timeFactory;
+ private bool $ran = false;
protected function setUp(): void {
parent::setUp();
@@ -43,7 +45,8 @@ class JobListTest extends TestCase {
$this->instance = new \OC\BackgroundJob\JobList(
$this->connection,
$this->config,
- $this->timeFactory
+ $this->timeFactory,
+ \OC::$server->get(LoggerInterface::class),
);
}
@@ -244,4 +247,24 @@ class JobListTest extends TestCase {
$this->assertGreaterThanOrEqual($timeStart, $addedJob->getLastRun());
$this->assertLessThanOrEqual($timeEnd, $addedJob->getLastRun());
}
+
+ public function testHasReservedJobs() {
+ $this->clearJobsList();
+ $job = new TestJob($this->timeFactory, $this, function () {
+ $this->assertTrue($this->instance->hasReservedJob());
+ $this->assertTrue($this->instance->hasReservedJob(TestJob::class));
+ });
+ $this->instance->add($job);
+
+ $this->assertFalse($this->instance->hasReservedJob());
+ $this->assertFalse($this->instance->hasReservedJob(TestJob::class));
+
+ $job->start($this->instance);
+
+ $this->assertTrue($this->ran);
+ }
+
+ public function markRun() {
+ $this->ran = true;
+ }
}
diff --git a/tests/lib/BackgroundJob/JobTest.php b/tests/lib/BackgroundJob/JobTest.php
index b4048aa1c22..ca9d68f0a2b 100644
--- a/tests/lib/BackgroundJob/JobTest.php
+++ b/tests/lib/BackgroundJob/JobTest.php
@@ -8,20 +8,23 @@
namespace Test\BackgroundJob;
+use OCP\AppFramework\Utility\ITimeFactory;
use OCP\ILogger;
class JobTest extends \Test\TestCase {
private $run = false;
+ private ITimeFactory $timeFactory;
protected function setUp(): void {
parent::setUp();
$this->run = false;
+ $this->timeFactory = \OC::$server->get(ITimeFactory::class);
}
public function testRemoveAfterException() {
$jobList = new DummyJobList();
$e = new \Exception();
- $job = new TestJob($this, function () use ($e) {
+ $job = new TestJob($this->timeFactory, $this, function () use ($e) {
throw $e;
});
$jobList->add($job);
@@ -30,8 +33,7 @@ class JobTest extends \Test\TestCase {
->disableOriginalConstructor()
->getMock();
$logger->expects($this->once())
- ->method('logException')
- ->with($e);
+ ->method('error');
$this->assertCount(1, $jobList->getAll());
$job->execute($jobList, $logger);
@@ -41,7 +43,7 @@ class JobTest extends \Test\TestCase {
public function testRemoveAfterError() {
$jobList = new DummyJobList();
- $job = new TestJob($this, function () {
+ $job = new TestJob($this->timeFactory, $this, function () {
$test = null;
$test->someMethod();
});
@@ -51,8 +53,7 @@ class JobTest extends \Test\TestCase {
->disableOriginalConstructor()
->getMock();
$logger->expects($this->once())
- ->method('logException')
- ->with($this->isInstanceOf(\Throwable::class));
+ ->method('error');
$this->assertCount(1, $jobList->getAll());
$job->execute($jobList, $logger);
@@ -60,6 +61,58 @@ class JobTest extends \Test\TestCase {
$this->assertCount(1, $jobList->getAll());
}
+ public function testDisallowParallelRunsWithNoOtherJobs() {
+ $jobList = new DummyJobList();
+ $job = new TestJob($this->timeFactory, $this, function () {
+ });
+ $job->setAllowParallelRuns(false);
+ $jobList->add($job);
+
+ $jobList->setHasReservedJob(null, false);
+ $jobList->setHasReservedJob(TestJob::class, false);
+ $job->start($jobList);
+ $this->assertTrue($this->run);
+ }
+
+ public function testAllowParallelRunsWithNoOtherJobs() {
+ $jobList = new DummyJobList();
+ $job = new TestJob($this->timeFactory, $this, function () {
+ });
+ $job->setAllowParallelRuns(true);
+ $jobList->add($job);
+
+ $jobList->setHasReservedJob(null, false);
+ $jobList->setHasReservedJob(TestJob::class, false);
+ $job->start($jobList);
+ $this->assertTrue($this->run);
+ }
+
+ public function testAllowParallelRunsWithOtherJobs() {
+ $jobList = new DummyJobList();
+ $job = new TestJob($this->timeFactory, $this, function () {
+ });
+ $job->setAllowParallelRuns(true);
+ $jobList->add($job);
+
+ $jobList->setHasReservedJob(null, true);
+ $jobList->setHasReservedJob(TestJob::class, true);
+ $job->start($jobList);
+ $this->assertTrue($this->run);
+ }
+
+ public function testDisallowParallelRunsWithOtherJobs() {
+ $jobList = new DummyJobList();
+ $job = new TestJob($this->timeFactory, $this, function () {
+ });
+ $job->setAllowParallelRuns(false);
+ $jobList->add($job);
+
+ $jobList->setHasReservedJob(null, true);
+ $jobList->setHasReservedJob(TestJob::class, true);
+ $job->start($jobList);
+ $this->assertFalse($this->run);
+ }
+
public function markRun() {
$this->run = true;
}
diff --git a/tests/lib/BackgroundJob/TestJob.php b/tests/lib/BackgroundJob/TestJob.php
index e15c7e86c99..cc7a4651c4b 100644
--- a/tests/lib/BackgroundJob/TestJob.php
+++ b/tests/lib/BackgroundJob/TestJob.php
@@ -8,7 +8,9 @@
namespace Test\BackgroundJob;
-class TestJob extends \OC\BackgroundJob\Job {
+use OCP\AppFramework\Utility\ITimeFactory;
+
+class TestJob extends \OCP\BackgroundJob\Job {
private $testCase;
/**
@@ -20,7 +22,8 @@ class TestJob extends \OC\BackgroundJob\Job {
* @param JobTest $testCase
* @param callable $callback
*/
- public function __construct($testCase = null, $callback = null) {
+ public function __construct(ITimeFactory $time = null, $testCase = null, $callback = null) {
+ parent::__construct($time ?? \OC::$server->get(ITimeFactory::class));
$this->testCase = $testCase;
$this->callback = $callback;
}