aboutsummaryrefslogtreecommitdiffstats
path: root/lib/private/TaskProcessing/SynchronousBackgroundJob.php
blob: ee6064aa4c6d91fe04daacf45668c623c0f55e13 (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
<?php

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\ValidationException;
use OCP\TaskProcessing\IManager;
use OCP\TaskProcessing\ISynchronousProvider;
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 $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 {
					$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]);
			}
		}

		// Schedule again
		$this->jobList->add(self::class, $argument);
	}
}