* * @author Christoph Wurst * @author Joas Schilling * @author Julius Härtl * @author Roeland Jago Douma * * @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 . * */ namespace OC\AppFramework\Bootstrap; use Closure; use OCP\Calendar\Resource\IBackend as IResourceBackend; use OCP\Calendar\Room\IBackend as IRoomBackend; use OCP\Collaboration\Reference\IReferenceProvider; use OCP\Talk\ITalkBackend; use RuntimeException; use function array_shift; use OC\Support\CrashReport\Registry; use OCP\AppFramework\App; use OCP\AppFramework\Bootstrap\IRegistrationContext; use OCP\AppFramework\Middleware; use OCP\AppFramework\Services\InitialStateProvider; use OCP\Authentication\IAlternativeLogin; use OCP\Calendar\ICalendarProvider; use OCP\Capabilities\ICapability; use OCP\Dashboard\IManager; use OCP\Dashboard\IWidget; use OCP\EventDispatcher\IEventDispatcher; use OCP\Files\Template\ICustomTemplateProvider; use OCP\Http\WellKnown\IHandler; use OCP\Notification\INotifier; use OCP\Profile\ILinkAction; use OCP\Search\IProvider; use OCP\Support\CrashReport\IReporter; use OCP\UserMigration\IMigrator as IUserMigrator; use Psr\Log\LoggerInterface; use Throwable; class RegistrationContext { /** @var ServiceRegistration[] */ private $capabilities = []; /** @var ServiceRegistration[] */ private $crashReporters = []; /** @var ServiceRegistration[] */ private $dashboardPanels = []; /** @var ServiceRegistration[] */ private $profileLinkActions = []; /** @var null|ServiceRegistration */ private $talkBackendRegistration = null; /** @var ServiceRegistration[] */ private $calendarResourceBackendRegistrations = []; /** @var ServiceRegistration[] */ private $calendarRoomBackendRegistrations = []; /** @var ServiceRegistration[] */ private $userMigrators = []; /** @var ServiceFactoryRegistration[] */ private $services = []; /** @var ServiceAliasRegistration[] */ private $aliases = []; /** @var ParameterRegistration[] */ private $parameters = []; /** @var EventListenerRegistration[] */ private $eventListeners = []; /** @var MiddlewareRegistration[] */ private $middlewares = []; /** @var ServiceRegistration[] */ private $searchProviders = []; /** @var ServiceRegistration[] */ private $alternativeLogins = []; /** @var ServiceRegistration[] */ private $initialStates = []; /** @var ServiceRegistration[] */ private $wellKnownHandlers = []; /** @var ServiceRegistration[] */ private $templateProviders = []; /** @var ServiceRegistration[] */ private $notifierServices = []; /** @var ServiceRegistration<\OCP\Authentication\TwoFactorAuth\IProvider>[] */ private $twoFactorProviders = []; /** @var ServiceRegistration[] */ private $calendarProviders = []; /** @var ServiceRegistration[] */ private array $referenceProviders = []; /** @var ParameterRegistration[] */ private $sensitiveMethods = []; /** @var LoggerInterface */ private $logger; /** @var PreviewProviderRegistration[] */ private $previewProviders = []; public function __construct(LoggerInterface $logger) { $this->logger = $logger; } public function for(string $appId): IRegistrationContext { return new class($appId, $this) implements IRegistrationContext { /** @var string */ private $appId; /** @var RegistrationContext */ private $context; public function __construct(string $appId, RegistrationContext $context) { $this->appId = $appId; $this->context = $context; } public function registerCapability(string $capability): void { $this->context->registerCapability( $this->appId, $capability ); } public function registerCrashReporter(string $reporterClass): void { $this->context->registerCrashReporter( $this->appId, $reporterClass ); } public function registerDashboardWidget(string $widgetClass): void { $this->context->registerDashboardPanel( $this->appId, $widgetClass ); } public function registerService(string $name, callable $factory, bool $shared = true): void { $this->context->registerService( $this->appId, $name, $factory, $shared ); } public function registerServiceAlias(string $alias, string $target): void { $this->context->registerServiceAlias( $this->appId, $alias, $target ); } public function registerParameter(string $name, $value): void { $this->context->registerParameter( $this->appId, $name, $value ); } public function registerEventListener(string $event, string $listener, int $priority = 0): void { $this->context->registerEventListener( $this->appId, $event, $listener, $priority ); } public function registerMiddleware(string $class, bool $global = false): void { $this->context->registerMiddleware( $this->appId, $class, $global, ); } public function registerSearchProvider(string $class): void { $this->context->registerSearchProvider( $this->appId, $class ); } public function registerAlternativeLogin(string $class): void { $this->context->registerAlternativeLogin( $this->appId, $class ); } public function registerInitialStateProvider(string $class): void { $this->context->registerInitialState( $this->appId, $class ); } public function registerWellKnownHandler(string $class): void { $this->context->registerWellKnown( $this->appId, $class ); } public function registerTemplateProvider(string $providerClass): void { $this->context->registerTemplateProvider( $this->appId, $providerClass ); } public function registerNotifierService(string $notifierClass): void { $this->context->registerNotifierService( $this->appId, $notifierClass ); } public function registerTwoFactorProvider(string $twoFactorProviderClass): void { $this->context->registerTwoFactorProvider( $this->appId, $twoFactorProviderClass ); } public function registerPreviewProvider(string $previewProviderClass, string $mimeTypeRegex): void { $this->context->registerPreviewProvider( $this->appId, $previewProviderClass, $mimeTypeRegex ); } public function registerCalendarProvider(string $class): void { $this->context->registerCalendarProvider( $this->appId, $class ); } public function registerReferenceProvider(string $class): void { $this->context->registerReferenceProvider( $this->appId, $class ); } public function registerProfileLinkAction(string $actionClass): void { $this->context->registerProfileLinkAction( $this->appId, $actionClass ); } public function registerTalkBackend(string $backend): void { $this->context->registerTalkBackend( $this->appId, $backend ); } public function registerCalendarResourceBackend(string $class): void { $this->context->registerCalendarResourceBackend( $this->appId, $class ); } public function registerCalendarRoomBackend(string $class): void { $this->context->registerCalendarRoomBackend( $this->appId, $class ); } public function registerUserMigrator(string $migratorClass): void { $this->context->registerUserMigrator( $this->appId, $migratorClass ); } public function registerSensitiveMethods(string $class, array $methods): void { $this->context->registerSensitiveMethods( $this->appId, $class, $methods ); } }; } /** * @psalm-param class-string $capability */ public function registerCapability(string $appId, string $capability): void { $this->capabilities[] = new ServiceRegistration($appId, $capability); } /** * @psalm-param class-string $capability */ public function registerCrashReporter(string $appId, string $reporterClass): void { $this->crashReporters[] = new ServiceRegistration($appId, $reporterClass); } /** * @psalm-param class-string $capability */ public function registerDashboardPanel(string $appId, string $panelClass): void { $this->dashboardPanels[] = new ServiceRegistration($appId, $panelClass); } public function registerService(string $appId, string $name, callable $factory, bool $shared = true): void { $this->services[] = new ServiceFactoryRegistration($appId, $name, $factory, $shared); } public function registerServiceAlias(string $appId, string $alias, string $target): void { $this->aliases[] = new ServiceAliasRegistration($appId, $alias, $target); } public function registerParameter(string $appId, string $name, $value): void { $this->parameters[] = new ParameterRegistration($appId, $name, $value); } public function registerEventListener(string $appId, string $event, string $listener, int $priority = 0): void { $this->eventListeners[] = new EventListenerRegistration($appId, $event, $listener, $priority); } /** * @psalm-param class-string $class */ public function registerMiddleware(string $appId, string $class, bool $global): void { $this->middlewares[] = new MiddlewareRegistration($appId, $class, $global); } public function registerSearchProvider(string $appId, string $class) { $this->searchProviders[] = new ServiceRegistration($appId, $class); } public function registerAlternativeLogin(string $appId, string $class): void { $this->alternativeLogins[] = new ServiceRegistration($appId, $class); } public function registerInitialState(string $appId, string $class): void { $this->initialStates[] = new ServiceRegistration($appId, $class); } public function registerWellKnown(string $appId, string $class): void { $this->wellKnownHandlers[] = new ServiceRegistration($appId, $class); } public function registerTemplateProvider(string $appId, string $class): void { $this->templateProviders[] = new ServiceRegistration($appId, $class); } public function registerNotifierService(string $appId, string $class): void { $this->notifierServices[] = new ServiceRegistration($appId, $class); } public function registerTwoFactorProvider(string $appId, string $class): void { $this->twoFactorProviders[] = new ServiceRegistration($appId, $class); } public function registerPreviewProvider(string $appId, string $class, string $mimeTypeRegex): void { $this->previewProviders[] = new PreviewProviderRegistration($appId, $class, $mimeTypeRegex); } public function registerCalendarProvider(string $appId, string $class): void { $this->calendarProviders[] = new ServiceRegistration($appId, $class); } public function registerReferenceProvider(string $appId, string $class): void { $this->referenceProviders[] = new ServiceRegistration($appId, $class); } /** * @psalm-param class-string $actionClass */ public function registerProfileLinkAction(string $appId, string $actionClass): void { $this->profileLinkActions[] = new ServiceRegistration($appId, $actionClass); } /** * @psalm-param class-string $backend */ public function registerTalkBackend(string $appId, string $backend) { // Some safeguards for invalid registrations if ($appId !== 'spreed') { throw new RuntimeException("Only the Talk app is allowed to register a Talk backend"); } if ($this->talkBackendRegistration !== null) { throw new RuntimeException("There can only be one Talk backend"); } $this->talkBackendRegistration = new ServiceRegistration($appId, $backend); } public function registerCalendarResourceBackend(string $appId, string $class) { $this->calendarResourceBackendRegistrations[] = new ServiceRegistration( $appId, $class, ); } public function registerCalendarRoomBackend(string $appId, string $class) { $this->calendarRoomBackendRegistrations[] = new ServiceRegistration( $appId, $class, ); } /** * @psalm-param class-string $migratorClass */ public function registerUserMigrator(string $appId, string $migratorClass): void { $this->userMigrators[] = new ServiceRegistration($appId, $migratorClass); } public function registerSensitiveMethods(string $appId, string $class, array $methods): void { $methods = array_filter($methods, 'is_string'); $this->sensitiveMethods[] = new ParameterRegistration($appId, $class, $methods); } /** * @param App[] $apps */ public function delegateCapabilityRegistrations(array $apps): void { while (($registration = array_shift($this->capabilities)) !== null) { $appId = $registration->getAppId(); if (!isset($apps[$appId])) { // If we land here something really isn't right. But at least we caught the // notice that is otherwise emitted for the undefined index $this->logger->error("App $appId not loaded for the capability registration"); continue; } try { $apps[$appId] ->getContainer() ->registerCapability($registration->getService()); } catch (Throwable $e) { $this->logger->error("Error during capability registration of $appId: " . $e->getMessage(), [ 'exception' => $e, ]); } } } /** * @param App[] $apps */ public function delegateCrashReporterRegistrations(array $apps, Registry $registry): void { while (($registration = array_shift($this->crashReporters)) !== null) { try { $registry->registerLazy($registration->getService()); } catch (Throwable $e) { $appId = $registration->getAppId(); $this->logger->error("Error during crash reporter registration of $appId: " . $e->getMessage(), [ 'exception' => $e, ]); } } } /** * @param App[] $apps */ public function delegateDashboardPanelRegistrations(IManager $dashboardManager): void { while (($panel = array_shift($this->dashboardPanels)) !== null) { try { $dashboardManager->lazyRegisterWidget($panel->getService(), $panel->getAppId()); } catch (Throwable $e) { $appId = $panel->getAppId(); $this->logger->error("Error during dashboard registration of $appId: " . $e->getMessage(), [ 'exception' => $e, ]); } } } public function delegateEventListenerRegistrations(IEventDispatcher $eventDispatcher): void { while (($registration = array_shift($this->eventListeners)) !== null) { try { $eventDispatcher->addServiceListener( $registration->getEvent(), $registration->getService(), $registration->getPriority() ); } catch (Throwable $e) { $appId = $registration->getAppId(); $this->logger->error("Error during event listener registration of $appId: " . $e->getMessage(), [ 'exception' => $e, ]); } } } /** * @param App[] $apps */ public function delegateContainerRegistrations(array $apps): void { while (($registration = array_shift($this->services)) !== null) { $appId = $registration->getAppId(); if (!isset($apps[$appId])) { // If we land here something really isn't right. But at least we caught the // notice that is otherwise emitted for the undefined index $this->logger->error("App $appId not loaded for the container service registration"); continue; } try { /** * Register the service and convert the callable into a \Closure if necessary */ $apps[$appId] ->getContainer() ->registerService( $registration->getName(), Closure::fromCallable($registration->getFactory()), $registration->isShared() ); } catch (Throwable $e) { $this->logger->error("Error during service registration of $appId: " . $e->getMessage(), [ 'exception' => $e, ]); } } while (($registration = array_shift($this->aliases)) !== null) { $appId = $registration->getAppId(); if (!isset($apps[$appId])) { // If we land here something really isn't right. But at least we caught the // notice that is otherwise emitted for the undefined index $this->logger->error("App $appId not loaded for the container alias registration"); continue; } try { $apps[$appId] ->getContainer() ->registerAlias( $registration->getAlias(), $registration->getTarget() ); } catch (Throwable $e) { $this->logger->error("Error during service alias registration of $appId: " . $e->getMessage(), [ 'exception' => $e, ]); } } while (($registration = array_shift($this->parameters)) !== null) { $appId = $registration->getAppId(); if (!isset($apps[$appId])) { // If we land here something really isn't right. But at least we caught the // notice that is otherwise emitted for the undefined index $this->logger->error("App $appId not loaded for the container parameter registration"); continue; } try { $apps[$appId] ->getContainer() ->registerParameter( $registration->getName(), $registration->getValue() ); } catch (Throwable $e) { $this->logger->error("Error during service parameter registration of $appId: " . $e->getMessage(), [ 'exception' => $e, ]); } } } /** * @return MiddlewareRegistration[] */ public function getMiddlewareRegistrations(): array { return $this->middlewares; } /** * @return ServiceRegistration[] */ public function getSearchProviders(): array { return $this->searchProviders; } /** * @return ServiceRegistration[] */ public function getAlternativeLogins(): array { return $this->alternativeLogins; } /** * @return ServiceRegistration[] */ public function getInitialStates(): array { return $this->initialStates; } /** * @return ServiceRegistration[] */ public function getWellKnownHandlers(): array { return $this->wellKnownHandlers; } /** * @return ServiceRegistration[] */ public function getTemplateProviders(): array { return $this->templateProviders; } /** * @return ServiceRegistration[] */ public function getNotifierServices(): array { return $this->notifierServices; } /** * @return ServiceRegistration<\OCP\Authentication\TwoFactorAuth\IProvider>[] */ public function getTwoFactorProviders(): array { return $this->twoFactorProviders; } /** * @return PreviewProviderRegistration[] */ public function getPreviewProviders(): array { return $this->previewProviders; } /** * @return ServiceRegistration[] */ public function getCalendarProviders(): array { return $this->calendarProviders; } /** * @return ServiceRegistration[] */ public function getReferenceProviders(): array { return $this->referenceProviders; } /** * @return ServiceRegistration[] */ public function getProfileLinkActions(): array { return $this->profileLinkActions; } /** * @return ServiceRegistration|null * @psalm-return ServiceRegistration|null */ public function getTalkBackendRegistration(): ?ServiceRegistration { return $this->talkBackendRegistration; } /** * @return ServiceRegistration[] * @psalm-return ServiceRegistration[] */ public function getCalendarResourceBackendRegistrations(): array { return $this->calendarResourceBackendRegistrations; } /** * @return ServiceRegistration[] * @psalm-return ServiceRegistration[] */ public function getCalendarRoomBackendRegistrations(): array { return $this->calendarRoomBackendRegistrations; } /** * @return ServiceRegistration[] */ public function getUserMigrators(): array { return $this->userMigrators; } /** * @return ParameterRegistration[] */ public function getSensitiveMethods(): array { return $this->sensitiveMethods; } }