aboutsummaryrefslogtreecommitdiffstats
path: root/lib/private/TaskProcessing/SynchronousBackgroundJob.php
blob: 093882d4c1e5369b2f7f882faa586a26ff2bd73e (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
<?php

/**
 * SPDX-FileCopyrightText: 2024 Nextcloud GmbH and Nextcloud contributors
 * SPDX-License-Identifier: AGPL-3.0-or-later
 */
namespace OC\TaskProcessing;

use OCP\AppFramework\Utility\ITimeFactory;
use OCP\BackgroundJob\IJobList;
use OCP\BackgroundJob\QueuedJob;
use OCP\Files\GenericFileException;
use OCP\Files\NotPermittedException;
use OCP\Lock\LockedException;
use OCP\TaskProcessing\Exception\Exception;
use OCP\TaskProcessing\Exception\NotFoundException;
use OCP\TaskProcessing\Exception\ProcessingException;
use OCP\TaskProcessing\Exception\UnauthorizedException;
use OCP\TaskProcessing\Exception\ValidationException;
use OCP\TaskProcessing\IManager;
use OCP\TaskProcessing\ISynchronousProvider;
use OCP\TaskProcessing\Task;
use Psr\Log\LoggerInterface;

class SynchronousBackgroundJob extends QueuedJob {
	public function __construct(
		ITimeFactory $timeFactory,
		private readonly IManager $taskProcessingManager,
		private readonly IJobList $jobList,
		private readonly LoggerInterface $logger,
	) {
		parent::__construct($timeFactory);
	}


	/**
	 * @inheritDoc
	 */
	protected function run($argument) {
		$providers = $this->taskProcessingManager->getProviders();

		foreach ($providers as $provider) {
			if (!$provider instanceof ISynchronousProvider) {
				continue;
			}
			$taskType = $provider->getTaskTypeId();
			try {
				$task = $this->taskProcessingManager->getNextScheduledTask([$taskType]);
			} catch (NotFoundException $e) {
				continue;
			} catch (Exception $e) {
				$this->logger->error('Unknown error while retrieving scheduled TaskProcessing tasks', ['exception' => $e]);
				continue;
			}
			try {
				try {
					$input = $this->taskProcessingManager->prepareInputData($task);
				} catch (GenericFileException|NotPermittedException|LockedException|ValidationException|UnauthorizedException $e) {
					$this->logger->warning('Failed to prepare input data for a TaskProcessing task with synchronous provider ' . $provider->getId(), ['exception' => $e]);
					$this->taskProcessingManager->setTaskResult($task->getId(), $e->getMessage(), null);
					// Schedule again
					$this->jobList->add(self::class, $argument);
					return;
				}
				try {
					$this->taskProcessingManager->setTaskStatus($task, Task::STATUS_RUNNING);
					$output = $provider->process($task->getUserId(), $input, fn (float $progress) => $this->taskProcessingManager->setTaskProgress($task->getId(), $progress));
				} catch (ProcessingException $e) {
					$this->logger->warning('Failed to process a TaskProcessing task with synchronous provider ' . $provider->getId(), ['exception' => $e]);
					$this->taskProcessingManager->setTaskResult($task->getId(), $e->getMessage(), null);
					// Schedule again
					$this->jobList->add(self::class, $argument);
					return;
				} catch (\Throwable $e) {
					$this->logger->error('Unknown error while processing TaskProcessing task', ['exception' => $e]);
					$this->taskProcessingManager->setTaskResult($task->getId(), $e->getMessage(), null);
					// Schedule again
					$this->jobList->add(self::class, $argument);
					return;
				}
				$this->taskProcessingManager->setTaskResult($task->getId(), null, $output);
			} catch (NotFoundException $e) {
				$this->logger->info('Could not find task anymore after execution. Moving on.', ['exception' => $e]);
			} catch (Exception $e) {
				$this->logger->error('Failed to report result of TaskProcessing task', ['exception' => $e]);
			}
		}

		$synchronousProviders = array_filter($providers, fn ($provider) =>
			$provider instanceof ISynchronousProvider);
		$taskTypes = array_values(array_map(fn ($provider) =>
			$provider->getTaskTypeId(),
			$synchronousProviders
		));
		$taskTypesWithTasks = array_filter($taskTypes, function ($taskType) {
			try {
				$this->taskProcessingManager->getNextScheduledTask([$taskType]);
				return true;
			} catch (NotFoundException|Exception $e) {
				return false;
			}
		});

		if (count($taskTypesWithTasks) > 0) {
			// Schedule again
			$this->jobList->add(self::class, $argument);
		}
	}
}