Completed
Push — master ( e2ea6d...b17745 )
by
unknown
30:23
created
lib/private/PreviewManager.php 2 patches
Spacing   +3 added lines, -3 removed lines patch added patch discarded remove patch
@@ -326,7 +326,7 @@  discard block
 block discarded – undo
326 326
 	 */
327 327
 	protected function registerCoreProvider($class, $mimeType, $options = []) {
328 328
 		if (in_array(trim($class, '\\'), $this->getEnabledDefaultProvider())) {
329
-			$this->registerProvider($mimeType, function () use ($class, $options) {
329
+			$this->registerProvider($mimeType, function() use ($class, $options) {
330 330
 				return new $class($options);
331 331
 			});
332 332
 		}
@@ -425,14 +425,14 @@  discard block
 block discarded – undo
425 425
 
426 426
 		$providers = $context->getPreviewProviders();
427 427
 		foreach ($providers as $provider) {
428
-			$key = $provider->getMimeTypeRegex() . '-' . $provider->getService();
428
+			$key = $provider->getMimeTypeRegex().'-'.$provider->getService();
429 429
 			if (array_key_exists($key, $this->loadedBootstrapProviders)) {
430 430
 				// Do not load the provider more than once
431 431
 				continue;
432 432
 			}
433 433
 			$this->loadedBootstrapProviders[$key] = null;
434 434
 
435
-			$this->registerProvider($provider->getMimeTypeRegex(), function () use ($provider) {
435
+			$this->registerProvider($provider->getMimeTypeRegex(), function() use ($provider) {
436 436
 				try {
437 437
 					return $this->container->get($provider->getService());
438 438
 				} catch (QueryException $e) {
Please login to merge, or discard this patch.
Indentation   +431 added lines, -431 removed lines patch added patch discarded remove patch
@@ -28,435 +28,435 @@
 block discarded – undo
28 28
 use function array_key_exists;
29 29
 
30 30
 class PreviewManager implements IPreview {
31
-	protected IConfig $config;
32
-	protected IRootFolder $rootFolder;
33
-	protected IAppData $appData;
34
-	protected IEventDispatcher $eventDispatcher;
35
-	private ?Generator $generator = null;
36
-	private GeneratorHelper $helper;
37
-	protected bool $providerListDirty = false;
38
-	protected bool $registeredCoreProviders = false;
39
-	protected array $providers = [];
40
-
41
-	/** @var array mime type => support status */
42
-	protected array $mimeTypeSupportMap = [];
43
-	protected ?array $defaultProviders = null;
44
-	protected ?string $userId;
45
-	private Coordinator $bootstrapCoordinator;
46
-
47
-	/**
48
-	 * Hash map (without value) of loaded bootstrap providers
49
-	 * @psalm-var array<string, null>
50
-	 */
51
-	private array $loadedBootstrapProviders = [];
52
-	private ContainerInterface $container;
53
-	private IBinaryFinder $binaryFinder;
54
-	private IMagickSupport $imagickSupport;
55
-	private bool $enablePreviews;
56
-
57
-	public function __construct(
58
-		IConfig $config,
59
-		IRootFolder $rootFolder,
60
-		IAppData $appData,
61
-		IEventDispatcher $eventDispatcher,
62
-		GeneratorHelper $helper,
63
-		?string $userId,
64
-		Coordinator $bootstrapCoordinator,
65
-		ContainerInterface $container,
66
-		IBinaryFinder $binaryFinder,
67
-		IMagickSupport $imagickSupport,
68
-	) {
69
-		$this->config = $config;
70
-		$this->rootFolder = $rootFolder;
71
-		$this->appData = $appData;
72
-		$this->eventDispatcher = $eventDispatcher;
73
-		$this->helper = $helper;
74
-		$this->userId = $userId;
75
-		$this->bootstrapCoordinator = $bootstrapCoordinator;
76
-		$this->container = $container;
77
-		$this->binaryFinder = $binaryFinder;
78
-		$this->imagickSupport = $imagickSupport;
79
-		$this->enablePreviews = $config->getSystemValueBool('enable_previews', true);
80
-	}
81
-
82
-	/**
83
-	 * In order to improve lazy loading a closure can be registered which will be
84
-	 * called in case preview providers are actually requested
85
-	 *
86
-	 * $callable has to return an instance of \OCP\Preview\IProvider or \OCP\Preview\IProviderV2
87
-	 *
88
-	 * @param string $mimeTypeRegex Regex with the mime types that are supported by this provider
89
-	 * @param \Closure $callable
90
-	 * @return void
91
-	 */
92
-	public function registerProvider($mimeTypeRegex, \Closure $callable): void {
93
-		if (!$this->enablePreviews) {
94
-			return;
95
-		}
96
-
97
-		if (!isset($this->providers[$mimeTypeRegex])) {
98
-			$this->providers[$mimeTypeRegex] = [];
99
-		}
100
-		$this->providers[$mimeTypeRegex][] = $callable;
101
-		$this->providerListDirty = true;
102
-	}
103
-
104
-	/**
105
-	 * Get all providers
106
-	 */
107
-	public function getProviders(): array {
108
-		if (!$this->enablePreviews) {
109
-			return [];
110
-		}
111
-
112
-		$this->registerCoreProviders();
113
-		$this->registerBootstrapProviders();
114
-		if ($this->providerListDirty) {
115
-			$keys = array_map('strlen', array_keys($this->providers));
116
-			array_multisort($keys, SORT_DESC, $this->providers);
117
-			$this->providerListDirty = false;
118
-		}
119
-
120
-		return $this->providers;
121
-	}
122
-
123
-	/**
124
-	 * Does the manager have any providers
125
-	 */
126
-	public function hasProviders(): bool {
127
-		$this->registerCoreProviders();
128
-		return !empty($this->providers);
129
-	}
130
-
131
-	private function getGenerator(): Generator {
132
-		if ($this->generator === null) {
133
-			$this->generator = new Generator(
134
-				$this->config,
135
-				$this,
136
-				$this->appData,
137
-				new GeneratorHelper(
138
-					$this->rootFolder,
139
-					$this->config
140
-				),
141
-				$this->eventDispatcher,
142
-				$this->container->get(LoggerInterface::class),
143
-			);
144
-		}
145
-		return $this->generator;
146
-	}
147
-
148
-	public function getPreview(
149
-		File $file,
150
-		$width = -1,
151
-		$height = -1,
152
-		$crop = false,
153
-		$mode = IPreview::MODE_FILL,
154
-		$mimeType = null,
155
-		bool $cacheResult = true,
156
-	): ISimpleFile {
157
-		$this->throwIfPreviewsDisabled($file, $mimeType);
158
-		$previewConcurrency = $this->getGenerator()->getNumConcurrentPreviews('preview_concurrency_all');
159
-		$sem = Generator::guardWithSemaphore(Generator::SEMAPHORE_ID_ALL, $previewConcurrency);
160
-		try {
161
-			$preview = $this->getGenerator()->getPreview($file, $width, $height, $crop, $mode, $mimeType, $cacheResult);
162
-		} finally {
163
-			Generator::unguardWithSemaphore($sem);
164
-		}
165
-
166
-		return $preview;
167
-	}
168
-
169
-	/**
170
-	 * Generates previews of a file
171
-	 *
172
-	 * @param File $file
173
-	 * @param array $specifications
174
-	 * @param string $mimeType
175
-	 * @return ISimpleFile the last preview that was generated
176
-	 * @throws NotFoundException
177
-	 * @throws \InvalidArgumentException if the preview would be invalid (in case the original image is invalid)
178
-	 * @since 19.0.0
179
-	 */
180
-	public function generatePreviews(File $file, array $specifications, $mimeType = null) {
181
-		$this->throwIfPreviewsDisabled($file, $mimeType);
182
-		return $this->getGenerator()->generatePreviews($file, $specifications, $mimeType);
183
-	}
184
-
185
-	/**
186
-	 * returns true if the passed mime type is supported
187
-	 *
188
-	 * @param string $mimeType
189
-	 * @return boolean
190
-	 */
191
-	public function isMimeSupported($mimeType = '*') {
192
-		if (!$this->enablePreviews) {
193
-			return false;
194
-		}
195
-
196
-		if (isset($this->mimeTypeSupportMap[$mimeType])) {
197
-			return $this->mimeTypeSupportMap[$mimeType];
198
-		}
199
-
200
-		$this->registerCoreProviders();
201
-		$this->registerBootstrapProviders();
202
-		$providerMimeTypes = array_keys($this->providers);
203
-		foreach ($providerMimeTypes as $supportedMimeType) {
204
-			if (preg_match($supportedMimeType, $mimeType)) {
205
-				$this->mimeTypeSupportMap[$mimeType] = true;
206
-				return true;
207
-			}
208
-		}
209
-		$this->mimeTypeSupportMap[$mimeType] = false;
210
-		return false;
211
-	}
212
-
213
-	/**
214
-	 * Check if a preview can be generated for a file
215
-	 */
216
-	public function isAvailable(\OCP\Files\FileInfo $file, ?string $mimeType = null): bool {
217
-		if (!$this->enablePreviews) {
218
-			return false;
219
-		}
220
-
221
-		$fileMimeType = $mimeType ?? $file->getMimeType();
222
-
223
-		$this->registerCoreProviders();
224
-		if (!$this->isMimeSupported($fileMimeType)) {
225
-			return false;
226
-		}
227
-
228
-		$mount = $file->getMountPoint();
229
-		if ($mount and !$mount->getOption('previews', true)) {
230
-			return false;
231
-		}
232
-
233
-		foreach ($this->providers as $supportedMimeType => $providers) {
234
-			if (preg_match($supportedMimeType, $fileMimeType)) {
235
-				foreach ($providers as $providerClosure) {
236
-					$provider = $this->helper->getProvider($providerClosure);
237
-					if (!($provider instanceof IProviderV2)) {
238
-						continue;
239
-					}
240
-
241
-					if ($provider->isAvailable($file)) {
242
-						return true;
243
-					}
244
-				}
245
-			}
246
-		}
247
-		return false;
248
-	}
249
-
250
-	/**
251
-	 * List of enabled default providers
252
-	 *
253
-	 * The following providers are enabled by default:
254
-	 *  - OC\Preview\PNG
255
-	 *  - OC\Preview\JPEG
256
-	 *  - OC\Preview\GIF
257
-	 *  - OC\Preview\BMP
258
-	 *  - OC\Preview\XBitmap
259
-	 *  - OC\Preview\MarkDown
260
-	 *  - OC\Preview\MP3
261
-	 *  - OC\Preview\TXT
262
-	 *
263
-	 * The following providers are disabled by default due to performance or privacy concerns:
264
-	 *  - OC\Preview\Font
265
-	 *  - OC\Preview\HEIC
266
-	 *  - OC\Preview\Illustrator
267
-	 *  - OC\Preview\Movie
268
-	 *  - OC\Preview\MSOfficeDoc
269
-	 *  - OC\Preview\MSOffice2003
270
-	 *  - OC\Preview\MSOffice2007
271
-	 *  - OC\Preview\OpenDocument
272
-	 *  - OC\Preview\PDF
273
-	 *  - OC\Preview\Photoshop
274
-	 *  - OC\Preview\Postscript
275
-	 *  - OC\Preview\StarOffice
276
-	 *  - OC\Preview\SVG
277
-	 *  - OC\Preview\TIFF
278
-	 *
279
-	 * @return array
280
-	 */
281
-	protected function getEnabledDefaultProvider() {
282
-		if ($this->defaultProviders !== null) {
283
-			return $this->defaultProviders;
284
-		}
285
-
286
-		$imageProviders = [
287
-			Preview\PNG::class,
288
-			Preview\JPEG::class,
289
-			Preview\GIF::class,
290
-			Preview\BMP::class,
291
-			Preview\XBitmap::class,
292
-			Preview\Krita::class,
293
-			Preview\WebP::class,
294
-		];
295
-
296
-		$this->defaultProviders = $this->config->getSystemValue('enabledPreviewProviders', array_merge([
297
-			Preview\MarkDown::class,
298
-			Preview\MP3::class,
299
-			Preview\TXT::class,
300
-			Preview\OpenDocument::class,
301
-		], $imageProviders));
302
-
303
-		if (in_array(Preview\Image::class, $this->defaultProviders)) {
304
-			$this->defaultProviders = array_merge($this->defaultProviders, $imageProviders);
305
-		}
306
-		$this->defaultProviders = array_unique($this->defaultProviders);
307
-		return $this->defaultProviders;
308
-	}
309
-
310
-	/**
311
-	 * Register the default providers (if enabled)
312
-	 *
313
-	 * @param string $class
314
-	 * @param string $mimeType
315
-	 */
316
-	protected function registerCoreProvider($class, $mimeType, $options = []) {
317
-		if (in_array(trim($class, '\\'), $this->getEnabledDefaultProvider())) {
318
-			$this->registerProvider($mimeType, function () use ($class, $options) {
319
-				return new $class($options);
320
-			});
321
-		}
322
-	}
323
-
324
-	/**
325
-	 * Register the default providers (if enabled)
326
-	 */
327
-	protected function registerCoreProviders() {
328
-		if ($this->registeredCoreProviders) {
329
-			return;
330
-		}
331
-		$this->registeredCoreProviders = true;
332
-
333
-		$this->registerCoreProvider(Preview\TXT::class, '/text\/plain/');
334
-		$this->registerCoreProvider(Preview\MarkDown::class, '/text\/(x-)?markdown/');
335
-		$this->registerCoreProvider(Preview\PNG::class, '/image\/png/');
336
-		$this->registerCoreProvider(Preview\JPEG::class, '/image\/jpeg/');
337
-		$this->registerCoreProvider(Preview\GIF::class, '/image\/gif/');
338
-		$this->registerCoreProvider(Preview\BMP::class, '/image\/bmp/');
339
-		$this->registerCoreProvider(Preview\XBitmap::class, '/image\/x-xbitmap/');
340
-		$this->registerCoreProvider(Preview\WebP::class, '/image\/webp/');
341
-		$this->registerCoreProvider(Preview\Krita::class, '/application\/x-krita/');
342
-		$this->registerCoreProvider(Preview\MP3::class, '/audio\/mpeg$/');
343
-		$this->registerCoreProvider(Preview\OpenDocument::class, '/application\/vnd.oasis.opendocument.*/');
344
-		$this->registerCoreProvider(Preview\Imaginary::class, Preview\Imaginary::supportedMimeTypes());
345
-		$this->registerCoreProvider(Preview\ImaginaryPDF::class, Preview\ImaginaryPDF::supportedMimeTypes());
346
-
347
-		// SVG and Bitmap require imagick
348
-		if ($this->imagickSupport->hasExtension()) {
349
-			$imagickProviders = [
350
-				'SVG' => ['mimetype' => '/image\/svg\+xml/', 'class' => Preview\SVG::class],
351
-				'TIFF' => ['mimetype' => '/image\/tiff/', 'class' => Preview\TIFF::class],
352
-				'PDF' => ['mimetype' => '/application\/pdf/', 'class' => Preview\PDF::class],
353
-				'AI' => ['mimetype' => '/application\/illustrator/', 'class' => Preview\Illustrator::class],
354
-				'PSD' => ['mimetype' => '/application\/x-photoshop/', 'class' => Preview\Photoshop::class],
355
-				'EPS' => ['mimetype' => '/application\/postscript/', 'class' => Preview\Postscript::class],
356
-				'TTF' => ['mimetype' => '/application\/(?:font-sfnt|x-font$)/', 'class' => Preview\Font::class],
357
-				'HEIC' => ['mimetype' => '/image\/(x-)?hei(f|c)/', 'class' => Preview\HEIC::class],
358
-				'TGA' => ['mimetype' => '/image\/(x-)?t(ar)?ga/', 'class' => Preview\TGA::class],
359
-				'SGI' => ['mimetype' => '/image\/(x-)?sgi/', 'class' => Preview\SGI::class],
360
-			];
361
-
362
-			foreach ($imagickProviders as $queryFormat => $provider) {
363
-				$class = $provider['class'];
364
-				if (!in_array(trim($class, '\\'), $this->getEnabledDefaultProvider())) {
365
-					continue;
366
-				}
367
-
368
-				if ($this->imagickSupport->supportsFormat($queryFormat)) {
369
-					$this->registerCoreProvider($class, $provider['mimetype']);
370
-				}
371
-			}
372
-		}
373
-
374
-		$this->registerCoreProvidersOffice();
375
-
376
-		// Video requires ffmpeg
377
-		if (in_array(Preview\Movie::class, $this->getEnabledDefaultProvider())) {
378
-			$movieBinary = $this->config->getSystemValue('preview_ffmpeg_path', null);
379
-			if (!is_string($movieBinary)) {
380
-				$movieBinary = $this->binaryFinder->findBinaryPath('ffmpeg');
381
-			}
382
-
383
-
384
-			if (is_string($movieBinary)) {
385
-				$this->registerCoreProvider(Preview\Movie::class, '/video\/.*/', ['movieBinary' => $movieBinary]);
386
-			}
387
-		}
388
-	}
389
-
390
-	private function registerCoreProvidersOffice(): void {
391
-		$officeProviders = [
392
-			['mimetype' => '/application\/msword/', 'class' => Preview\MSOfficeDoc::class],
393
-			['mimetype' => '/application\/vnd.ms-.*/', 'class' => Preview\MSOffice2003::class],
394
-			['mimetype' => '/application\/vnd.openxmlformats-officedocument.*/', 'class' => Preview\MSOffice2007::class],
395
-			['mimetype' => '/application\/vnd.oasis.opendocument.*/', 'class' => Preview\OpenDocument::class],
396
-			['mimetype' => '/application\/vnd.sun.xml.*/', 'class' => Preview\StarOffice::class],
397
-			['mimetype' => '/image\/emf/', 'class' => Preview\EMF::class],
398
-		];
399
-
400
-		$findBinary = true;
401
-		$officeBinary = false;
402
-
403
-		foreach ($officeProviders as $provider) {
404
-			$class = $provider['class'];
405
-			if (!in_array(trim($class, '\\'), $this->getEnabledDefaultProvider())) {
406
-				continue;
407
-			}
408
-
409
-			if ($findBinary) {
410
-				// Office requires openoffice or libreoffice
411
-				$officeBinary = $this->config->getSystemValue('preview_libreoffice_path', false);
412
-				if ($officeBinary === false) {
413
-					$officeBinary = $this->binaryFinder->findBinaryPath('libreoffice');
414
-				}
415
-				if ($officeBinary === false) {
416
-					$officeBinary = $this->binaryFinder->findBinaryPath('openoffice');
417
-				}
418
-				$findBinary = false;
419
-			}
420
-
421
-			if ($officeBinary) {
422
-				$this->registerCoreProvider($class, $provider['mimetype'], ['officeBinary' => $officeBinary]);
423
-			}
424
-		}
425
-	}
426
-
427
-	private function registerBootstrapProviders(): void {
428
-		$context = $this->bootstrapCoordinator->getRegistrationContext();
429
-
430
-		if ($context === null) {
431
-			// Just ignore for now
432
-			return;
433
-		}
434
-
435
-		$providers = $context->getPreviewProviders();
436
-		foreach ($providers as $provider) {
437
-			$key = $provider->getMimeTypeRegex() . '-' . $provider->getService();
438
-			if (array_key_exists($key, $this->loadedBootstrapProviders)) {
439
-				// Do not load the provider more than once
440
-				continue;
441
-			}
442
-			$this->loadedBootstrapProviders[$key] = null;
443
-
444
-			$this->registerProvider($provider->getMimeTypeRegex(), function () use ($provider) {
445
-				try {
446
-					return $this->container->get($provider->getService());
447
-				} catch (QueryException $e) {
448
-					return null;
449
-				}
450
-			});
451
-		}
452
-	}
453
-
454
-	/**
455
-	 * @throws NotFoundException if preview generation is disabled
456
-	 */
457
-	private function throwIfPreviewsDisabled(File $file, ?string $mimeType = null): void {
458
-		if (!$this->isAvailable($file, $mimeType)) {
459
-			throw new NotFoundException('Previews disabled');
460
-		}
461
-	}
31
+    protected IConfig $config;
32
+    protected IRootFolder $rootFolder;
33
+    protected IAppData $appData;
34
+    protected IEventDispatcher $eventDispatcher;
35
+    private ?Generator $generator = null;
36
+    private GeneratorHelper $helper;
37
+    protected bool $providerListDirty = false;
38
+    protected bool $registeredCoreProviders = false;
39
+    protected array $providers = [];
40
+
41
+    /** @var array mime type => support status */
42
+    protected array $mimeTypeSupportMap = [];
43
+    protected ?array $defaultProviders = null;
44
+    protected ?string $userId;
45
+    private Coordinator $bootstrapCoordinator;
46
+
47
+    /**
48
+     * Hash map (without value) of loaded bootstrap providers
49
+     * @psalm-var array<string, null>
50
+     */
51
+    private array $loadedBootstrapProviders = [];
52
+    private ContainerInterface $container;
53
+    private IBinaryFinder $binaryFinder;
54
+    private IMagickSupport $imagickSupport;
55
+    private bool $enablePreviews;
56
+
57
+    public function __construct(
58
+        IConfig $config,
59
+        IRootFolder $rootFolder,
60
+        IAppData $appData,
61
+        IEventDispatcher $eventDispatcher,
62
+        GeneratorHelper $helper,
63
+        ?string $userId,
64
+        Coordinator $bootstrapCoordinator,
65
+        ContainerInterface $container,
66
+        IBinaryFinder $binaryFinder,
67
+        IMagickSupport $imagickSupport,
68
+    ) {
69
+        $this->config = $config;
70
+        $this->rootFolder = $rootFolder;
71
+        $this->appData = $appData;
72
+        $this->eventDispatcher = $eventDispatcher;
73
+        $this->helper = $helper;
74
+        $this->userId = $userId;
75
+        $this->bootstrapCoordinator = $bootstrapCoordinator;
76
+        $this->container = $container;
77
+        $this->binaryFinder = $binaryFinder;
78
+        $this->imagickSupport = $imagickSupport;
79
+        $this->enablePreviews = $config->getSystemValueBool('enable_previews', true);
80
+    }
81
+
82
+    /**
83
+     * In order to improve lazy loading a closure can be registered which will be
84
+     * called in case preview providers are actually requested
85
+     *
86
+     * $callable has to return an instance of \OCP\Preview\IProvider or \OCP\Preview\IProviderV2
87
+     *
88
+     * @param string $mimeTypeRegex Regex with the mime types that are supported by this provider
89
+     * @param \Closure $callable
90
+     * @return void
91
+     */
92
+    public function registerProvider($mimeTypeRegex, \Closure $callable): void {
93
+        if (!$this->enablePreviews) {
94
+            return;
95
+        }
96
+
97
+        if (!isset($this->providers[$mimeTypeRegex])) {
98
+            $this->providers[$mimeTypeRegex] = [];
99
+        }
100
+        $this->providers[$mimeTypeRegex][] = $callable;
101
+        $this->providerListDirty = true;
102
+    }
103
+
104
+    /**
105
+     * Get all providers
106
+     */
107
+    public function getProviders(): array {
108
+        if (!$this->enablePreviews) {
109
+            return [];
110
+        }
111
+
112
+        $this->registerCoreProviders();
113
+        $this->registerBootstrapProviders();
114
+        if ($this->providerListDirty) {
115
+            $keys = array_map('strlen', array_keys($this->providers));
116
+            array_multisort($keys, SORT_DESC, $this->providers);
117
+            $this->providerListDirty = false;
118
+        }
119
+
120
+        return $this->providers;
121
+    }
122
+
123
+    /**
124
+     * Does the manager have any providers
125
+     */
126
+    public function hasProviders(): bool {
127
+        $this->registerCoreProviders();
128
+        return !empty($this->providers);
129
+    }
130
+
131
+    private function getGenerator(): Generator {
132
+        if ($this->generator === null) {
133
+            $this->generator = new Generator(
134
+                $this->config,
135
+                $this,
136
+                $this->appData,
137
+                new GeneratorHelper(
138
+                    $this->rootFolder,
139
+                    $this->config
140
+                ),
141
+                $this->eventDispatcher,
142
+                $this->container->get(LoggerInterface::class),
143
+            );
144
+        }
145
+        return $this->generator;
146
+    }
147
+
148
+    public function getPreview(
149
+        File $file,
150
+        $width = -1,
151
+        $height = -1,
152
+        $crop = false,
153
+        $mode = IPreview::MODE_FILL,
154
+        $mimeType = null,
155
+        bool $cacheResult = true,
156
+    ): ISimpleFile {
157
+        $this->throwIfPreviewsDisabled($file, $mimeType);
158
+        $previewConcurrency = $this->getGenerator()->getNumConcurrentPreviews('preview_concurrency_all');
159
+        $sem = Generator::guardWithSemaphore(Generator::SEMAPHORE_ID_ALL, $previewConcurrency);
160
+        try {
161
+            $preview = $this->getGenerator()->getPreview($file, $width, $height, $crop, $mode, $mimeType, $cacheResult);
162
+        } finally {
163
+            Generator::unguardWithSemaphore($sem);
164
+        }
165
+
166
+        return $preview;
167
+    }
168
+
169
+    /**
170
+     * Generates previews of a file
171
+     *
172
+     * @param File $file
173
+     * @param array $specifications
174
+     * @param string $mimeType
175
+     * @return ISimpleFile the last preview that was generated
176
+     * @throws NotFoundException
177
+     * @throws \InvalidArgumentException if the preview would be invalid (in case the original image is invalid)
178
+     * @since 19.0.0
179
+     */
180
+    public function generatePreviews(File $file, array $specifications, $mimeType = null) {
181
+        $this->throwIfPreviewsDisabled($file, $mimeType);
182
+        return $this->getGenerator()->generatePreviews($file, $specifications, $mimeType);
183
+    }
184
+
185
+    /**
186
+     * returns true if the passed mime type is supported
187
+     *
188
+     * @param string $mimeType
189
+     * @return boolean
190
+     */
191
+    public function isMimeSupported($mimeType = '*') {
192
+        if (!$this->enablePreviews) {
193
+            return false;
194
+        }
195
+
196
+        if (isset($this->mimeTypeSupportMap[$mimeType])) {
197
+            return $this->mimeTypeSupportMap[$mimeType];
198
+        }
199
+
200
+        $this->registerCoreProviders();
201
+        $this->registerBootstrapProviders();
202
+        $providerMimeTypes = array_keys($this->providers);
203
+        foreach ($providerMimeTypes as $supportedMimeType) {
204
+            if (preg_match($supportedMimeType, $mimeType)) {
205
+                $this->mimeTypeSupportMap[$mimeType] = true;
206
+                return true;
207
+            }
208
+        }
209
+        $this->mimeTypeSupportMap[$mimeType] = false;
210
+        return false;
211
+    }
212
+
213
+    /**
214
+     * Check if a preview can be generated for a file
215
+     */
216
+    public function isAvailable(\OCP\Files\FileInfo $file, ?string $mimeType = null): bool {
217
+        if (!$this->enablePreviews) {
218
+            return false;
219
+        }
220
+
221
+        $fileMimeType = $mimeType ?? $file->getMimeType();
222
+
223
+        $this->registerCoreProviders();
224
+        if (!$this->isMimeSupported($fileMimeType)) {
225
+            return false;
226
+        }
227
+
228
+        $mount = $file->getMountPoint();
229
+        if ($mount and !$mount->getOption('previews', true)) {
230
+            return false;
231
+        }
232
+
233
+        foreach ($this->providers as $supportedMimeType => $providers) {
234
+            if (preg_match($supportedMimeType, $fileMimeType)) {
235
+                foreach ($providers as $providerClosure) {
236
+                    $provider = $this->helper->getProvider($providerClosure);
237
+                    if (!($provider instanceof IProviderV2)) {
238
+                        continue;
239
+                    }
240
+
241
+                    if ($provider->isAvailable($file)) {
242
+                        return true;
243
+                    }
244
+                }
245
+            }
246
+        }
247
+        return false;
248
+    }
249
+
250
+    /**
251
+     * List of enabled default providers
252
+     *
253
+     * The following providers are enabled by default:
254
+     *  - OC\Preview\PNG
255
+     *  - OC\Preview\JPEG
256
+     *  - OC\Preview\GIF
257
+     *  - OC\Preview\BMP
258
+     *  - OC\Preview\XBitmap
259
+     *  - OC\Preview\MarkDown
260
+     *  - OC\Preview\MP3
261
+     *  - OC\Preview\TXT
262
+     *
263
+     * The following providers are disabled by default due to performance or privacy concerns:
264
+     *  - OC\Preview\Font
265
+     *  - OC\Preview\HEIC
266
+     *  - OC\Preview\Illustrator
267
+     *  - OC\Preview\Movie
268
+     *  - OC\Preview\MSOfficeDoc
269
+     *  - OC\Preview\MSOffice2003
270
+     *  - OC\Preview\MSOffice2007
271
+     *  - OC\Preview\OpenDocument
272
+     *  - OC\Preview\PDF
273
+     *  - OC\Preview\Photoshop
274
+     *  - OC\Preview\Postscript
275
+     *  - OC\Preview\StarOffice
276
+     *  - OC\Preview\SVG
277
+     *  - OC\Preview\TIFF
278
+     *
279
+     * @return array
280
+     */
281
+    protected function getEnabledDefaultProvider() {
282
+        if ($this->defaultProviders !== null) {
283
+            return $this->defaultProviders;
284
+        }
285
+
286
+        $imageProviders = [
287
+            Preview\PNG::class,
288
+            Preview\JPEG::class,
289
+            Preview\GIF::class,
290
+            Preview\BMP::class,
291
+            Preview\XBitmap::class,
292
+            Preview\Krita::class,
293
+            Preview\WebP::class,
294
+        ];
295
+
296
+        $this->defaultProviders = $this->config->getSystemValue('enabledPreviewProviders', array_merge([
297
+            Preview\MarkDown::class,
298
+            Preview\MP3::class,
299
+            Preview\TXT::class,
300
+            Preview\OpenDocument::class,
301
+        ], $imageProviders));
302
+
303
+        if (in_array(Preview\Image::class, $this->defaultProviders)) {
304
+            $this->defaultProviders = array_merge($this->defaultProviders, $imageProviders);
305
+        }
306
+        $this->defaultProviders = array_unique($this->defaultProviders);
307
+        return $this->defaultProviders;
308
+    }
309
+
310
+    /**
311
+     * Register the default providers (if enabled)
312
+     *
313
+     * @param string $class
314
+     * @param string $mimeType
315
+     */
316
+    protected function registerCoreProvider($class, $mimeType, $options = []) {
317
+        if (in_array(trim($class, '\\'), $this->getEnabledDefaultProvider())) {
318
+            $this->registerProvider($mimeType, function () use ($class, $options) {
319
+                return new $class($options);
320
+            });
321
+        }
322
+    }
323
+
324
+    /**
325
+     * Register the default providers (if enabled)
326
+     */
327
+    protected function registerCoreProviders() {
328
+        if ($this->registeredCoreProviders) {
329
+            return;
330
+        }
331
+        $this->registeredCoreProviders = true;
332
+
333
+        $this->registerCoreProvider(Preview\TXT::class, '/text\/plain/');
334
+        $this->registerCoreProvider(Preview\MarkDown::class, '/text\/(x-)?markdown/');
335
+        $this->registerCoreProvider(Preview\PNG::class, '/image\/png/');
336
+        $this->registerCoreProvider(Preview\JPEG::class, '/image\/jpeg/');
337
+        $this->registerCoreProvider(Preview\GIF::class, '/image\/gif/');
338
+        $this->registerCoreProvider(Preview\BMP::class, '/image\/bmp/');
339
+        $this->registerCoreProvider(Preview\XBitmap::class, '/image\/x-xbitmap/');
340
+        $this->registerCoreProvider(Preview\WebP::class, '/image\/webp/');
341
+        $this->registerCoreProvider(Preview\Krita::class, '/application\/x-krita/');
342
+        $this->registerCoreProvider(Preview\MP3::class, '/audio\/mpeg$/');
343
+        $this->registerCoreProvider(Preview\OpenDocument::class, '/application\/vnd.oasis.opendocument.*/');
344
+        $this->registerCoreProvider(Preview\Imaginary::class, Preview\Imaginary::supportedMimeTypes());
345
+        $this->registerCoreProvider(Preview\ImaginaryPDF::class, Preview\ImaginaryPDF::supportedMimeTypes());
346
+
347
+        // SVG and Bitmap require imagick
348
+        if ($this->imagickSupport->hasExtension()) {
349
+            $imagickProviders = [
350
+                'SVG' => ['mimetype' => '/image\/svg\+xml/', 'class' => Preview\SVG::class],
351
+                'TIFF' => ['mimetype' => '/image\/tiff/', 'class' => Preview\TIFF::class],
352
+                'PDF' => ['mimetype' => '/application\/pdf/', 'class' => Preview\PDF::class],
353
+                'AI' => ['mimetype' => '/application\/illustrator/', 'class' => Preview\Illustrator::class],
354
+                'PSD' => ['mimetype' => '/application\/x-photoshop/', 'class' => Preview\Photoshop::class],
355
+                'EPS' => ['mimetype' => '/application\/postscript/', 'class' => Preview\Postscript::class],
356
+                'TTF' => ['mimetype' => '/application\/(?:font-sfnt|x-font$)/', 'class' => Preview\Font::class],
357
+                'HEIC' => ['mimetype' => '/image\/(x-)?hei(f|c)/', 'class' => Preview\HEIC::class],
358
+                'TGA' => ['mimetype' => '/image\/(x-)?t(ar)?ga/', 'class' => Preview\TGA::class],
359
+                'SGI' => ['mimetype' => '/image\/(x-)?sgi/', 'class' => Preview\SGI::class],
360
+            ];
361
+
362
+            foreach ($imagickProviders as $queryFormat => $provider) {
363
+                $class = $provider['class'];
364
+                if (!in_array(trim($class, '\\'), $this->getEnabledDefaultProvider())) {
365
+                    continue;
366
+                }
367
+
368
+                if ($this->imagickSupport->supportsFormat($queryFormat)) {
369
+                    $this->registerCoreProvider($class, $provider['mimetype']);
370
+                }
371
+            }
372
+        }
373
+
374
+        $this->registerCoreProvidersOffice();
375
+
376
+        // Video requires ffmpeg
377
+        if (in_array(Preview\Movie::class, $this->getEnabledDefaultProvider())) {
378
+            $movieBinary = $this->config->getSystemValue('preview_ffmpeg_path', null);
379
+            if (!is_string($movieBinary)) {
380
+                $movieBinary = $this->binaryFinder->findBinaryPath('ffmpeg');
381
+            }
382
+
383
+
384
+            if (is_string($movieBinary)) {
385
+                $this->registerCoreProvider(Preview\Movie::class, '/video\/.*/', ['movieBinary' => $movieBinary]);
386
+            }
387
+        }
388
+    }
389
+
390
+    private function registerCoreProvidersOffice(): void {
391
+        $officeProviders = [
392
+            ['mimetype' => '/application\/msword/', 'class' => Preview\MSOfficeDoc::class],
393
+            ['mimetype' => '/application\/vnd.ms-.*/', 'class' => Preview\MSOffice2003::class],
394
+            ['mimetype' => '/application\/vnd.openxmlformats-officedocument.*/', 'class' => Preview\MSOffice2007::class],
395
+            ['mimetype' => '/application\/vnd.oasis.opendocument.*/', 'class' => Preview\OpenDocument::class],
396
+            ['mimetype' => '/application\/vnd.sun.xml.*/', 'class' => Preview\StarOffice::class],
397
+            ['mimetype' => '/image\/emf/', 'class' => Preview\EMF::class],
398
+        ];
399
+
400
+        $findBinary = true;
401
+        $officeBinary = false;
402
+
403
+        foreach ($officeProviders as $provider) {
404
+            $class = $provider['class'];
405
+            if (!in_array(trim($class, '\\'), $this->getEnabledDefaultProvider())) {
406
+                continue;
407
+            }
408
+
409
+            if ($findBinary) {
410
+                // Office requires openoffice or libreoffice
411
+                $officeBinary = $this->config->getSystemValue('preview_libreoffice_path', false);
412
+                if ($officeBinary === false) {
413
+                    $officeBinary = $this->binaryFinder->findBinaryPath('libreoffice');
414
+                }
415
+                if ($officeBinary === false) {
416
+                    $officeBinary = $this->binaryFinder->findBinaryPath('openoffice');
417
+                }
418
+                $findBinary = false;
419
+            }
420
+
421
+            if ($officeBinary) {
422
+                $this->registerCoreProvider($class, $provider['mimetype'], ['officeBinary' => $officeBinary]);
423
+            }
424
+        }
425
+    }
426
+
427
+    private function registerBootstrapProviders(): void {
428
+        $context = $this->bootstrapCoordinator->getRegistrationContext();
429
+
430
+        if ($context === null) {
431
+            // Just ignore for now
432
+            return;
433
+        }
434
+
435
+        $providers = $context->getPreviewProviders();
436
+        foreach ($providers as $provider) {
437
+            $key = $provider->getMimeTypeRegex() . '-' . $provider->getService();
438
+            if (array_key_exists($key, $this->loadedBootstrapProviders)) {
439
+                // Do not load the provider more than once
440
+                continue;
441
+            }
442
+            $this->loadedBootstrapProviders[$key] = null;
443
+
444
+            $this->registerProvider($provider->getMimeTypeRegex(), function () use ($provider) {
445
+                try {
446
+                    return $this->container->get($provider->getService());
447
+                } catch (QueryException $e) {
448
+                    return null;
449
+                }
450
+            });
451
+        }
452
+    }
453
+
454
+    /**
455
+     * @throws NotFoundException if preview generation is disabled
456
+     */
457
+    private function throwIfPreviewsDisabled(File $file, ?string $mimeType = null): void {
458
+        if (!$this->isAvailable($file, $mimeType)) {
459
+            throw new NotFoundException('Previews disabled');
460
+        }
461
+    }
462 462
 }
Please login to merge, or discard this patch.
apps/settings/lib/Controller/HelpController.php 2 patches
Spacing   +1 added lines, -1 removed lines patch added patch discarded remove patch
@@ -79,7 +79,7 @@
 block discarded – undo
79 79
 		}
80 80
 
81 81
 		$documentationUrl = $this->urlGenerator->getAbsoluteURL(
82
-			$this->urlGenerator->linkTo('', 'core/doc/' . $mode . '/index.html')
82
+			$this->urlGenerator->linkTo('', 'core/doc/'.$mode.'/index.html')
83 83
 		);
84 84
 
85 85
 		$urlUserDocs = $this->urlGenerator->linkToRoute('settings.Help.help', ['mode' => 'user']);
Please login to merge, or discard this patch.
Indentation   +57 added lines, -57 removed lines patch added patch discarded remove patch
@@ -25,67 +25,67 @@
 block discarded – undo
25 25
 #[OpenAPI(scope: OpenAPI::SCOPE_IGNORE)]
26 26
 class HelpController extends Controller {
27 27
 
28
-	public function __construct(
29
-		string $appName,
30
-		IRequest $request,
31
-		private INavigationManager $navigationManager,
32
-		private IURLGenerator $urlGenerator,
33
-		/** @var string */
34
-		private ?string $userId,
35
-		private IGroupManager $groupManager,
36
-		private IL10N $l10n,
37
-		private IConfig $config,
38
-		private IAppConfig $appConfig,
39
-	) {
40
-		parent::__construct($appName, $request);
41
-	}
28
+    public function __construct(
29
+        string $appName,
30
+        IRequest $request,
31
+        private INavigationManager $navigationManager,
32
+        private IURLGenerator $urlGenerator,
33
+        /** @var string */
34
+        private ?string $userId,
35
+        private IGroupManager $groupManager,
36
+        private IL10N $l10n,
37
+        private IConfig $config,
38
+        private IAppConfig $appConfig,
39
+    ) {
40
+        parent::__construct($appName, $request);
41
+    }
42 42
 
43
-	/**
44
-	 * @return TemplateResponse
45
-	 *
46
-	 * @NoSubAdminRequired
47
-	 */
48
-	#[NoCSRFRequired]
49
-	#[NoAdminRequired]
50
-	public function help(string $mode = 'user'): TemplateResponse {
51
-		$this->navigationManager->setActiveEntry('help');
52
-		$pageTitle = $this->l10n->t('Administrator documentation');
53
-		if ($mode !== 'admin') {
54
-			$pageTitle = $this->l10n->t('User documentation');
55
-			$mode = 'user';
56
-		}
43
+    /**
44
+     * @return TemplateResponse
45
+     *
46
+     * @NoSubAdminRequired
47
+     */
48
+    #[NoCSRFRequired]
49
+    #[NoAdminRequired]
50
+    public function help(string $mode = 'user'): TemplateResponse {
51
+        $this->navigationManager->setActiveEntry('help');
52
+        $pageTitle = $this->l10n->t('Administrator documentation');
53
+        if ($mode !== 'admin') {
54
+            $pageTitle = $this->l10n->t('User documentation');
55
+            $mode = 'user';
56
+        }
57 57
 
58
-		$documentationUrl = $this->urlGenerator->getAbsoluteURL(
59
-			$this->urlGenerator->linkTo('', 'core/doc/' . $mode . '/index.html')
60
-		);
58
+        $documentationUrl = $this->urlGenerator->getAbsoluteURL(
59
+            $this->urlGenerator->linkTo('', 'core/doc/' . $mode . '/index.html')
60
+        );
61 61
 
62
-		$urlUserDocs = $this->urlGenerator->linkToRoute('settings.Help.help', ['mode' => 'user']);
63
-		$urlAdminDocs = $this->urlGenerator->linkToRoute('settings.Help.help', ['mode' => 'admin']);
62
+        $urlUserDocs = $this->urlGenerator->linkToRoute('settings.Help.help', ['mode' => 'user']);
63
+        $urlAdminDocs = $this->urlGenerator->linkToRoute('settings.Help.help', ['mode' => 'admin']);
64 64
 
65
-		$knowledgebaseEmbedded = $this->config->getSystemValueBool('knowledgebase.embedded', false);
66
-		if (!$knowledgebaseEmbedded) {
67
-			$pageTitle = $this->l10n->t('Nextcloud help overview');
68
-			$urlUserDocs = $this->urlGenerator->linkToDocs('user');
69
-			$urlAdminDocs = $this->urlGenerator->linkToDocs('admin');
70
-		}
65
+        $knowledgebaseEmbedded = $this->config->getSystemValueBool('knowledgebase.embedded', false);
66
+        if (!$knowledgebaseEmbedded) {
67
+            $pageTitle = $this->l10n->t('Nextcloud help overview');
68
+            $urlUserDocs = $this->urlGenerator->linkToDocs('user');
69
+            $urlAdminDocs = $this->urlGenerator->linkToDocs('admin');
70
+        }
71 71
 
72
-		$legalNoticeUrl = $this->appConfig->getValueString('theming', 'imprintUrl');
73
-		$privacyUrl = $this->appConfig->getValueString('theming', 'privacyUrl');
72
+        $legalNoticeUrl = $this->appConfig->getValueString('theming', 'imprintUrl');
73
+        $privacyUrl = $this->appConfig->getValueString('theming', 'privacyUrl');
74 74
 
75
-		$response = new TemplateResponse('settings', 'help', [
76
-			'admin' => $this->groupManager->isAdmin($this->userId),
77
-			'url' => $documentationUrl,
78
-			'urlUserDocs' => $urlUserDocs,
79
-			'urlAdminDocs' => $urlAdminDocs,
80
-			'mode' => $mode,
81
-			'pageTitle' => $pageTitle,
82
-			'knowledgebaseEmbedded' => $knowledgebaseEmbedded,
83
-			'legalNoticeUrl' => $legalNoticeUrl,
84
-			'privacyUrl' => $privacyUrl,
85
-		]);
86
-		$policy = new ContentSecurityPolicy();
87
-		$policy->addAllowedFrameDomain('\'self\'');
88
-		$response->setContentSecurityPolicy($policy);
89
-		return $response;
90
-	}
75
+        $response = new TemplateResponse('settings', 'help', [
76
+            'admin' => $this->groupManager->isAdmin($this->userId),
77
+            'url' => $documentationUrl,
78
+            'urlUserDocs' => $urlUserDocs,
79
+            'urlAdminDocs' => $urlAdminDocs,
80
+            'mode' => $mode,
81
+            'pageTitle' => $pageTitle,
82
+            'knowledgebaseEmbedded' => $knowledgebaseEmbedded,
83
+            'legalNoticeUrl' => $legalNoticeUrl,
84
+            'privacyUrl' => $privacyUrl,
85
+        ]);
86
+        $policy = new ContentSecurityPolicy();
87
+        $policy->addAllowedFrameDomain('\'self\'');
88
+        $response->setContentSecurityPolicy($policy);
89
+        return $response;
90
+    }
91 91
 }
Please login to merge, or discard this patch.
lib/private/Authentication/Token/TokenCleanupJob.php 1 patch
Indentation   +11 added lines, -11 removed lines patch added patch discarded remove patch
@@ -25,17 +25,17 @@
 block discarded – undo
25 25
 use OCP\BackgroundJob\TimedJob;
26 26
 
27 27
 class TokenCleanupJob extends TimedJob {
28
-	private IProvider $provider;
28
+    private IProvider $provider;
29 29
 
30
-	public function __construct(ITimeFactory $time, IProvider $provider) {
31
-		parent::__construct($time);
32
-		$this->provider = $provider;
33
-		// Run once a day at off-peak time
34
-		$this->setInterval(24 * 60 * 60);
35
-		$this->setTimeSensitivity(self::TIME_INSENSITIVE);
36
-	}
30
+    public function __construct(ITimeFactory $time, IProvider $provider) {
31
+        parent::__construct($time);
32
+        $this->provider = $provider;
33
+        // Run once a day at off-peak time
34
+        $this->setInterval(24 * 60 * 60);
35
+        $this->setTimeSensitivity(self::TIME_INSENSITIVE);
36
+    }
37 37
 
38
-	protected function run($argument) {
39
-		$this->provider->invalidateOldTokens();
40
-	}
38
+    protected function run($argument) {
39
+        $this->provider->invalidateOldTokens();
40
+    }
41 41
 }
Please login to merge, or discard this patch.
lib/private/Repair/NC24/AddTokenCleanupJob.php 1 patch
Indentation   +10 added lines, -10 removed lines patch added patch discarded remove patch
@@ -31,17 +31,17 @@
 block discarded – undo
31 31
 use OCP\Migration\IRepairStep;
32 32
 
33 33
 class AddTokenCleanupJob implements IRepairStep {
34
-	private IJobList $jobList;
34
+    private IJobList $jobList;
35 35
 
36
-	public function __construct(IJobList $jobList) {
37
-		$this->jobList = $jobList;
38
-	}
36
+    public function __construct(IJobList $jobList) {
37
+        $this->jobList = $jobList;
38
+    }
39 39
 
40
-	public function getName(): string {
41
-		return 'Add token cleanup job';
42
-	}
40
+    public function getName(): string {
41
+        return 'Add token cleanup job';
42
+    }
43 43
 
44
-	public function run(IOutput $output) {
45
-		$this->jobList->add(TokenCleanupJob::class);
46
-	}
44
+    public function run(IOutput $output) {
45
+        $this->jobList->add(TokenCleanupJob::class);
46
+    }
47 47
 }
Please login to merge, or discard this patch.
lib/public/Files/DavUtil.php 2 patches
Spacing   +1 added lines, -1 removed lines patch added patch discarded remove patch
@@ -47,7 +47,7 @@
 block discarded – undo
47 47
 	public static function getDavFileId(int $id): string {
48 48
 		$instanceId = \OC_Util::getInstanceId();
49 49
 		$id = sprintf('%08d', $id);
50
-		return $id . $instanceId;
50
+		return $id.$instanceId;
51 51
 	}
52 52
 
53 53
 	/**
Please login to merge, or discard this patch.
Indentation   +57 added lines, -57 removed lines patch added patch discarded remove patch
@@ -15,64 +15,64 @@
 block discarded – undo
15 15
  * @since 25.0.0
16 16
  */
17 17
 class DavUtil {
18
-	/**
19
-	 * Compute the fileId to use for dav responses
20
-	 *
21
-	 * @param int $id Id of the file returned by FileInfo::getId
22
-	 * @since 25.0.0
23
-	 */
24
-	public static function getDavFileId(int $id): string {
25
-		$instanceId = \OC_Util::getInstanceId();
26
-		$id = sprintf('%08d', $id);
27
-		return $id . $instanceId;
28
-	}
18
+    /**
19
+     * Compute the fileId to use for dav responses
20
+     *
21
+     * @param int $id Id of the file returned by FileInfo::getId
22
+     * @since 25.0.0
23
+     */
24
+    public static function getDavFileId(int $id): string {
25
+        $instanceId = \OC_Util::getInstanceId();
26
+        $id = sprintf('%08d', $id);
27
+        return $id . $instanceId;
28
+    }
29 29
 
30
-	/**
31
-	 * Compute the format needed for returning permissions for dav
32
-	 *
33
-	 * @since 25.0.0
34
-	 */
35
-	public static function getDavPermissions(FileInfo $info): string {
36
-		$permissions = $info->getPermissions();
37
-		$p = '';
38
-		if ($info->isShared()) {
39
-			$p .= 'S';
40
-		}
41
-		if ($permissions & Constants::PERMISSION_SHARE) {
42
-			$p .= 'R';
43
-		}
44
-		if ($info->isMounted()) {
45
-			$p .= 'M';
46
-		}
47
-		if ($permissions & Constants::PERMISSION_READ) {
48
-			$p .= 'G';
49
-		}
50
-		if ($permissions & Constants::PERMISSION_DELETE) {
51
-			$p .= 'D';
52
-		}
53
-		if ($permissions & Constants::PERMISSION_UPDATE) {
54
-			$p .= 'NV'; // Renameable, Movable
55
-		}
30
+    /**
31
+     * Compute the format needed for returning permissions for dav
32
+     *
33
+     * @since 25.0.0
34
+     */
35
+    public static function getDavPermissions(FileInfo $info): string {
36
+        $permissions = $info->getPermissions();
37
+        $p = '';
38
+        if ($info->isShared()) {
39
+            $p .= 'S';
40
+        }
41
+        if ($permissions & Constants::PERMISSION_SHARE) {
42
+            $p .= 'R';
43
+        }
44
+        if ($info->isMounted()) {
45
+            $p .= 'M';
46
+        }
47
+        if ($permissions & Constants::PERMISSION_READ) {
48
+            $p .= 'G';
49
+        }
50
+        if ($permissions & Constants::PERMISSION_DELETE) {
51
+            $p .= 'D';
52
+        }
53
+        if ($permissions & Constants::PERMISSION_UPDATE) {
54
+            $p .= 'NV'; // Renameable, Movable
55
+        }
56 56
 
57
-		// since we always add update permissions for the root of movable mounts
58
-		// we need to check the shared cache item directly to determine if it's writable
59
-		$storage = $info->getStorage();
60
-		if ($info->getInternalPath() === '' && $info->getMountPoint() instanceof IMovableMount) {
61
-			$rootEntry = $storage->getCache()->get('');
62
-			$isWritable = $rootEntry->getPermissions() & Constants::PERMISSION_UPDATE;
63
-		} else {
64
-			$isWritable = $permissions & Constants::PERMISSION_UPDATE;
65
-		}
57
+        // since we always add update permissions for the root of movable mounts
58
+        // we need to check the shared cache item directly to determine if it's writable
59
+        $storage = $info->getStorage();
60
+        if ($info->getInternalPath() === '' && $info->getMountPoint() instanceof IMovableMount) {
61
+            $rootEntry = $storage->getCache()->get('');
62
+            $isWritable = $rootEntry->getPermissions() & Constants::PERMISSION_UPDATE;
63
+        } else {
64
+            $isWritable = $permissions & Constants::PERMISSION_UPDATE;
65
+        }
66 66
 
67
-		if ($info->getType() === FileInfo::TYPE_FILE) {
68
-			if ($isWritable) {
69
-				$p .= 'W';
70
-			}
71
-		} else {
72
-			if ($permissions & Constants::PERMISSION_CREATE) {
73
-				$p .= 'CK';
74
-			}
75
-		}
76
-		return $p;
77
-	}
67
+        if ($info->getType() === FileInfo::TYPE_FILE) {
68
+            if ($isWritable) {
69
+                $p .= 'W';
70
+            }
71
+        } else {
72
+            if ($permissions & Constants::PERMISSION_CREATE) {
73
+                $p .= 'CK';
74
+            }
75
+        }
76
+        return $p;
77
+    }
78 78
 }
Please login to merge, or discard this patch.
lib/public/Collaboration/Reference/IReferenceProvider.php 1 patch
Indentation   +29 added lines, -29 removed lines patch added patch discarded remove patch
@@ -28,36 +28,36 @@
 block discarded – undo
28 28
  * @since 25.0.0
29 29
  */
30 30
 interface IReferenceProvider {
31
-	/**
32
-	 * Validate that a given reference identifier matches the current provider
33
-	 *
34
-	 * @since 25.0.0
35
-	 */
36
-	public function matchReference(string $referenceText): bool;
31
+    /**
32
+     * Validate that a given reference identifier matches the current provider
33
+     *
34
+     * @since 25.0.0
35
+     */
36
+    public function matchReference(string $referenceText): bool;
37 37
 
38
-	/**
39
-	 * Return a reference with its metadata for a given reference identifier
40
-	 *
41
-	 * @since 25.0.0
42
-	 */
43
-	public function resolveReference(string $referenceText): ?IReference;
38
+    /**
39
+     * Return a reference with its metadata for a given reference identifier
40
+     *
41
+     * @since 25.0.0
42
+     */
43
+    public function resolveReference(string $referenceText): ?IReference;
44 44
 
45
-	/**
46
-	 * Return true if the reference metadata can be globally cached
47
-	 *
48
-	 * @since 25.0.0
49
-	 */
50
-	public function getCachePrefix(string $referenceId): string;
45
+    /**
46
+     * Return true if the reference metadata can be globally cached
47
+     *
48
+     * @since 25.0.0
49
+     */
50
+    public function getCachePrefix(string $referenceId): string;
51 51
 
52
-	/**
53
-	 * Return a custom cache key to be used for caching the metadata
54
-	 * This could be for example the current user id if the reference
55
-	 * access permissions are different for each user
56
-	 *
57
-	 * Should return null, if the cache is only related to the
58
-	 * reference id and has no further dependency
59
-	 *
60
-	 * @since 25.0.0
61
-	 */
62
-	public function getCacheKey(string $referenceId): ?string;
52
+    /**
53
+     * Return a custom cache key to be used for caching the metadata
54
+     * This could be for example the current user id if the reference
55
+     * access permissions are different for each user
56
+     *
57
+     * Should return null, if the cache is only related to the
58
+     * reference id and has no further dependency
59
+     *
60
+     * @since 25.0.0
61
+     */
62
+    public function getCacheKey(string $referenceId): ?string;
63 63
 }
Please login to merge, or discard this patch.
apps/settings/templates/settings/personal/security/password.php 1 patch
Indentation   +1 added lines, -1 removed lines patch added patch discarded remove patch
@@ -23,7 +23,7 @@
 block discarded – undo
23 23
  */
24 24
 
25 25
 if ($_['passwordChangeSupported']) {
26
-	\OCP\Util::addScript('settings', 'vue-settings-personal-password');
26
+    \OCP\Util::addScript('settings', 'vue-settings-personal-password');
27 27
 }
28 28
 ?>
29 29
 <div id="security-password"></div>
Please login to merge, or discard this patch.
lib/private/Files/Stream/HashWrapper.php 1 patch
Indentation   +47 added lines, -47 removed lines patch added patch discarded remove patch
@@ -28,58 +28,58 @@
 block discarded – undo
28 28
 use Icewind\Streams\Wrapper;
29 29
 
30 30
 class HashWrapper extends Wrapper {
31
-	protected $callback;
32
-	protected $hash;
31
+    protected $callback;
32
+    protected $hash;
33 33
 
34
-	public static function wrap($source, string $algo, callable $callback) {
35
-		$hash = hash_init($algo);
36
-		$context = stream_context_create([
37
-			'hash' => [
38
-				'source' => $source,
39
-				'callback' => $callback,
40
-				'hash' => $hash,
41
-			],
42
-		]);
43
-		return Wrapper::wrapSource($source, $context, 'hash', self::class);
44
-	}
34
+    public static function wrap($source, string $algo, callable $callback) {
35
+        $hash = hash_init($algo);
36
+        $context = stream_context_create([
37
+            'hash' => [
38
+                'source' => $source,
39
+                'callback' => $callback,
40
+                'hash' => $hash,
41
+            ],
42
+        ]);
43
+        return Wrapper::wrapSource($source, $context, 'hash', self::class);
44
+    }
45 45
 
46
-	protected function open() {
47
-		$context = $this->loadContext('hash');
46
+    protected function open() {
47
+        $context = $this->loadContext('hash');
48 48
 
49
-		$this->callback = $context['callback'];
50
-		$this->hash = $context['hash'];
51
-		return true;
52
-	}
49
+        $this->callback = $context['callback'];
50
+        $this->hash = $context['hash'];
51
+        return true;
52
+    }
53 53
 
54
-	public function dir_opendir($path, $options) {
55
-		return $this->open();
56
-	}
54
+    public function dir_opendir($path, $options) {
55
+        return $this->open();
56
+    }
57 57
 
58
-	public function stream_open($path, $mode, $options, &$opened_path) {
59
-		return $this->open();
60
-	}
58
+    public function stream_open($path, $mode, $options, &$opened_path) {
59
+        return $this->open();
60
+    }
61 61
 
62
-	public function stream_read($count) {
63
-		$result = parent::stream_read($count);
64
-		hash_update($this->hash, $result);
65
-		return $result;
66
-	}
62
+    public function stream_read($count) {
63
+        $result = parent::stream_read($count);
64
+        hash_update($this->hash, $result);
65
+        return $result;
66
+    }
67 67
 
68
-	public function stream_close() {
69
-		if (is_callable($this->callback)) {
70
-			// if the stream is closed as a result of the end-of-request GC, the hash context might be cleaned up before this stream
71
-			if ($this->hash instanceof \HashContext) {
72
-				try {
73
-					$hash = @hash_final($this->hash);
74
-					if ($hash) {
75
-						call_user_func($this->callback, $hash);
76
-					}
77
-				} catch (\Throwable $e) {
78
-				}
79
-			}
80
-			// prevent further calls by potential PHP 7 GC ghosts
81
-			$this->callback = null;
82
-		}
83
-		return parent::stream_close();
84
-	}
68
+    public function stream_close() {
69
+        if (is_callable($this->callback)) {
70
+            // if the stream is closed as a result of the end-of-request GC, the hash context might be cleaned up before this stream
71
+            if ($this->hash instanceof \HashContext) {
72
+                try {
73
+                    $hash = @hash_final($this->hash);
74
+                    if ($hash) {
75
+                        call_user_func($this->callback, $hash);
76
+                    }
77
+                } catch (\Throwable $e) {
78
+                }
79
+            }
80
+            // prevent further calls by potential PHP 7 GC ghosts
81
+            $this->callback = null;
82
+        }
83
+        return parent::stream_close();
84
+    }
85 85
 }
Please login to merge, or discard this patch.
lib/public/Color.php 2 patches
Spacing   +3 added lines, -3 removed lines patch added patch discarded remove patch
@@ -133,9 +133,9 @@
 block discarded – undo
133 133
 	 */
134 134
 	public function alphaBlending(float $opacity, Color $source): Color {
135 135
 		return new Color(
136
-			(int)((1 - $opacity) * $source->red() + $opacity * $this->red()),
137
-			(int)((1 - $opacity) * $source->green() + $opacity * $this->green()),
138
-			(int)((1 - $opacity) * $source->blue() + $opacity * $this->blue())
136
+			(int) ((1 - $opacity) * $source->red() + $opacity * $this->red()),
137
+			(int) ((1 - $opacity) * $source->green() + $opacity * $this->green()),
138
+			(int) ((1 - $opacity) * $source->blue() + $opacity * $this->blue())
139 139
 		);
140 140
 	}
141 141
 
Please login to merge, or discard this patch.
Indentation   +114 added lines, -114 removed lines patch added patch discarded remove patch
@@ -10,129 +10,129 @@
 block discarded – undo
10 10
  * @since 25.0.0
11 11
  */
12 12
 class Color {
13
-	private int $r;
14
-	private int $g;
15
-	private int $b;
13
+    private int $r;
14
+    private int $g;
15
+    private int $b;
16 16
 
17
-	/**
18
-	 * @since 25.0.0
19
-	 */
20
-	public function __construct($r, $g, $b) {
21
-		$this->r = $r;
22
-		$this->g = $g;
23
-		$this->b = $b;
24
-	}
17
+    /**
18
+     * @since 25.0.0
19
+     */
20
+    public function __construct($r, $g, $b) {
21
+        $this->r = $r;
22
+        $this->g = $g;
23
+        $this->b = $b;
24
+    }
25 25
 
26
-	/**
27
-	 * Returns the red color component of this color as an int from 0 to 255
28
-	 *
29
-	 * @since 25.0.0
30
-	 */
31
-	public function red(): int {
32
-		return $this->r;
33
-	}
26
+    /**
27
+     * Returns the red color component of this color as an int from 0 to 255
28
+     *
29
+     * @since 25.0.0
30
+     */
31
+    public function red(): int {
32
+        return $this->r;
33
+    }
34 34
 
35
-	/**
36
-	 * Returns the red color component of this color as a float from 0 to 1
37
-	 *
38
-	 * @since 25.0.0
39
-	 */
40
-	public function redF(): float {
41
-		return $this->r / 255;
42
-	}
35
+    /**
36
+     * Returns the red color component of this color as a float from 0 to 1
37
+     *
38
+     * @since 25.0.0
39
+     */
40
+    public function redF(): float {
41
+        return $this->r / 255;
42
+    }
43 43
 
44
-	/**
45
-	 * Returns the green color component of this color as an int from 0 to 255
46
-	 *
47
-	 * @since 25.0.0
48
-	 */
49
-	public function green(): int {
50
-		return $this->g;
51
-	}
44
+    /**
45
+     * Returns the green color component of this color as an int from 0 to 255
46
+     *
47
+     * @since 25.0.0
48
+     */
49
+    public function green(): int {
50
+        return $this->g;
51
+    }
52 52
 
53
-	/**
54
-	 * Returns the green color component of this color as a float from 0 to 1
55
-	 *
56
-	 * @since 25.0.0
57
-	 */
58
-	public function greenF(): float {
59
-		return $this->g / 255;
60
-	}
53
+    /**
54
+     * Returns the green color component of this color as a float from 0 to 1
55
+     *
56
+     * @since 25.0.0
57
+     */
58
+    public function greenF(): float {
59
+        return $this->g / 255;
60
+    }
61 61
 
62
-	/**
63
-	 * Returns the green blue component of this color as an int from 0 to 255
64
-	 *
65
-	 * @since 25.0.0
66
-	 */
67
-	public function blue(): int {
68
-		return $this->b;
69
-	}
62
+    /**
63
+     * Returns the green blue component of this color as an int from 0 to 255
64
+     *
65
+     * @since 25.0.0
66
+     */
67
+    public function blue(): int {
68
+        return $this->b;
69
+    }
70 70
 
71
-	/**
72
-	 * Returns the blue color component of this color as a float from 0 to 1
73
-	 *
74
-	 * @since 25.0.0
75
-	 */
76
-	public function blueF(): float {
77
-		return $this->g / 255;
78
-	}
71
+    /**
72
+     * Returns the blue color component of this color as a float from 0 to 1
73
+     *
74
+     * @since 25.0.0
75
+     */
76
+    public function blueF(): float {
77
+        return $this->g / 255;
78
+    }
79 79
 
80
-	/**
81
-	 * Returns the name of the color in the format "#RRGGBB"; i.e. a "#" character followed by three two-digit hexadecimal numbers.
82
-	 *
83
-	 * @since 25.0.0
84
-	 */
85
-	public function name(): string {
86
-		return sprintf('#%02x%02x%02x', $this->r, $this->g, $this->b);
87
-	}
80
+    /**
81
+     * Returns the name of the color in the format "#RRGGBB"; i.e. a "#" character followed by three two-digit hexadecimal numbers.
82
+     *
83
+     * @since 25.0.0
84
+     */
85
+    public function name(): string {
86
+        return sprintf('#%02x%02x%02x', $this->r, $this->g, $this->b);
87
+    }
88 88
 
89
-	/**
90
-	 * Mix two colors
91
-	 *
92
-	 * @param int $steps the number of intermediate colors that should be generated for the palette
93
-	 * @param Color $color1 the first color
94
-	 * @param Color $color2 the second color
95
-	 * @return list<Color>
96
-	 * @since 25.0.0
97
-	 */
98
-	public static function mixPalette(int $steps, Color $color1, Color $color2): array {
99
-		$palette = [$color1];
100
-		$step = self::stepCalc($steps, [$color1, $color2]);
101
-		for ($i = 1; $i < $steps; $i++) {
102
-			$r = intval($color1->red() + ($step[0] * $i));
103
-			$g = intval($color1->green() + ($step[1] * $i));
104
-			$b = intval($color1->blue() + ($step[2] * $i));
105
-			$palette[] = new Color($r, $g, $b);
106
-		}
107
-		return $palette;
108
-	}
89
+    /**
90
+     * Mix two colors
91
+     *
92
+     * @param int $steps the number of intermediate colors that should be generated for the palette
93
+     * @param Color $color1 the first color
94
+     * @param Color $color2 the second color
95
+     * @return list<Color>
96
+     * @since 25.0.0
97
+     */
98
+    public static function mixPalette(int $steps, Color $color1, Color $color2): array {
99
+        $palette = [$color1];
100
+        $step = self::stepCalc($steps, [$color1, $color2]);
101
+        for ($i = 1; $i < $steps; $i++) {
102
+            $r = intval($color1->red() + ($step[0] * $i));
103
+            $g = intval($color1->green() + ($step[1] * $i));
104
+            $b = intval($color1->blue() + ($step[2] * $i));
105
+            $palette[] = new Color($r, $g, $b);
106
+        }
107
+        return $palette;
108
+    }
109 109
 
110
-	/**
111
-	 * Alpha blend another color with a given opacity to this color
112
-	 *
113
-	 * @return Color The new color
114
-	 * @since 25.0.0
115
-	 */
116
-	public function alphaBlending(float $opacity, Color $source): Color {
117
-		return new Color(
118
-			(int)((1 - $opacity) * $source->red() + $opacity * $this->red()),
119
-			(int)((1 - $opacity) * $source->green() + $opacity * $this->green()),
120
-			(int)((1 - $opacity) * $source->blue() + $opacity * $this->blue())
121
-		);
122
-	}
110
+    /**
111
+     * Alpha blend another color with a given opacity to this color
112
+     *
113
+     * @return Color The new color
114
+     * @since 25.0.0
115
+     */
116
+    public function alphaBlending(float $opacity, Color $source): Color {
117
+        return new Color(
118
+            (int)((1 - $opacity) * $source->red() + $opacity * $this->red()),
119
+            (int)((1 - $opacity) * $source->green() + $opacity * $this->green()),
120
+            (int)((1 - $opacity) * $source->blue() + $opacity * $this->blue())
121
+        );
122
+    }
123 123
 
124
-	/**
125
-	 * Calculate steps between two Colors
126
-	 * @param int $steps start color
127
-	 * @param Color[] $ends end color
128
-	 * @return array{0: float, 1: float, 2: float} [r,g,b] steps for each color to go from $steps to $ends
129
-	 * @since 25.0.0
130
-	 */
131
-	private static function stepCalc(int $steps, array $ends): array {
132
-		$step = [];
133
-		$step[0] = ($ends[1]->red() - $ends[0]->red()) / $steps;
134
-		$step[1] = ($ends[1]->green() - $ends[0]->green()) / $steps;
135
-		$step[2] = ($ends[1]->blue() - $ends[0]->blue()) / $steps;
136
-		return $step;
137
-	}
124
+    /**
125
+     * Calculate steps between two Colors
126
+     * @param int $steps start color
127
+     * @param Color[] $ends end color
128
+     * @return array{0: float, 1: float, 2: float} [r,g,b] steps for each color to go from $steps to $ends
129
+     * @since 25.0.0
130
+     */
131
+    private static function stepCalc(int $steps, array $ends): array {
132
+        $step = [];
133
+        $step[0] = ($ends[1]->red() - $ends[0]->red()) / $steps;
134
+        $step[1] = ($ends[1]->green() - $ends[0]->green()) / $steps;
135
+        $step[2] = ($ends[1]->blue() - $ends[0]->blue()) / $steps;
136
+        return $step;
137
+    }
138 138
 }
Please login to merge, or discard this patch.