From 92f91c533c09031c796da95983d46c1162defc0f Mon Sep 17 00:00:00 2001 From: Florent Morselli Date: Tue, 5 Mar 2024 12:33:30 +0100 Subject: [PATCH] Refactoring ideas (#90) Refactoring ideas --- ecs.php | 4 +- phpstan-baseline.neon | 127 ++- src/Command/CreateServiceWorkerCommand.php | 6 +- src/DependencyInjection/Configuration.php | 828 ------------------ .../SpomkyLabsPwaExtension.php | 91 -- .../config/definition/asset_public_prefix.php | 16 + .../config/definition/image_processor.php | 17 + src/Resources/config/definition/manifest.php | 139 +++ .../config/definition/path_type_reference.php | 32 + .../config/definition/service_worker.php | 200 +++++ .../config/definition/utils/file_handlers.php | 38 + .../config/definition/utils/icons.php | 65 ++ .../definition/utils/launch_handler.php | 32 + .../definition/utils/protocol_handlers.php | 31 + .../definition/utils/related_applications.php | 39 + .../config/definition/utils/screenshots.php | 58 ++ .../config/definition/utils/shared_target.php | 56 ++ .../config/definition/utils/shortcuts.php | 40 + .../config/definition/utils/url_node.php | 42 + .../config/definition/utils/widgets.php | 78 ++ .../config/definition/web_client.php | 15 + src/SpomkyLabsPwaBundle.php | 74 +- src/Subscriber/PwaDevServerSubscriber.php | 10 +- .../WorkboxCompileEventListener.php | 8 +- 24 files changed, 1072 insertions(+), 974 deletions(-) delete mode 100644 src/DependencyInjection/Configuration.php delete mode 100644 src/DependencyInjection/SpomkyLabsPwaExtension.php create mode 100644 src/Resources/config/definition/asset_public_prefix.php create mode 100644 src/Resources/config/definition/image_processor.php create mode 100644 src/Resources/config/definition/manifest.php create mode 100644 src/Resources/config/definition/path_type_reference.php create mode 100644 src/Resources/config/definition/service_worker.php create mode 100644 src/Resources/config/definition/utils/file_handlers.php create mode 100644 src/Resources/config/definition/utils/icons.php create mode 100644 src/Resources/config/definition/utils/launch_handler.php create mode 100644 src/Resources/config/definition/utils/protocol_handlers.php create mode 100644 src/Resources/config/definition/utils/related_applications.php create mode 100644 src/Resources/config/definition/utils/screenshots.php create mode 100644 src/Resources/config/definition/utils/shared_target.php create mode 100644 src/Resources/config/definition/utils/shortcuts.php create mode 100644 src/Resources/config/definition/utils/url_node.php create mode 100644 src/Resources/config/definition/utils/widgets.php create mode 100644 src/Resources/config/definition/web_client.php diff --git a/ecs.php b/ecs.php index 917d102..62aa8e7 100644 --- a/ecs.php +++ b/ecs.php @@ -92,8 +92,8 @@ $config->skip([ PhpUnitTestClassRequiresCoversFixer::class, - MethodChainingIndentationFixer::class => [__DIR__ . '/src/DependencyInjection/Configuration.php'], - MethodChainingNewlineFixer::class => [__DIR__ . '/src/DependencyInjection/Configuration.php'], + MethodChainingIndentationFixer::class => [__DIR__ . '/src/Resources/config'], + MethodChainingNewlineFixer::class => [__DIR__ . '/src/Resources/config'], ]); $config->parallel(); diff --git a/phpstan-baseline.neon b/phpstan-baseline.neon index 183af00..f786269 100644 --- a/phpstan-baseline.neon +++ b/phpstan-baseline.neon @@ -100,41 +100,6 @@ parameters: count: 1 path: src/Command/CreateServiceWorkerCommand.php - - - message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" - count: 4 - path: src/DependencyInjection/Configuration.php - - - - message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:append\\(\\)\\.$#" - count: 2 - path: src/DependencyInjection/Configuration.php - - - - message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:beforeNormalization\\(\\)\\.$#" - count: 1 - path: src/DependencyInjection/Configuration.php - - - - message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:end\\(\\)\\.$#" - count: 1 - path: src/DependencyInjection/Configuration.php - - - - message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:scalarNode\\(\\)\\.$#" - count: 5 - path: src/DependencyInjection/Configuration.php - - - - message: "#^Cannot access offset 'public_prefix' on mixed\\.$#" - count: 1 - path: src/DependencyInjection/SpomkyLabsPwaExtension.php - - - - message: "#^Method SpomkyLabs\\\\PwaBundle\\\\DependencyInjection\\\\SpomkyLabsPwaExtension\\:\\:getConfiguration\\(\\) has parameter \\$config with no value type specified in iterable type array\\.$#" - count: 1 - path: src/DependencyInjection/SpomkyLabsPwaExtension.php - - message: "#^Class SpomkyLabs\\\\PwaBundle\\\\Dto\\\\File has an uninitialized property \\$accept\\. Give it default value or assign it in the constructor\\.$#" count: 1 @@ -475,7 +440,97 @@ parameters: count: 1 path: src/Normalizer/UrlNormalizer.php + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/asset_public_prefix.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/image_processor.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/manifest.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/path_type_reference.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/service_worker.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:end\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/file_handlers.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/icons.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:beforeNormalization\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/launch_handler.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:append\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/protocol_handlers.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:append\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/related_applications.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/screenshots.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:scalarNode\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/shared_target.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:scalarNode\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/shortcuts.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/url_node.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeParentInterface\\:\\:scalarNode\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/utils/widgets.php + + - + message: "#^Call to an undefined method Symfony\\\\Component\\\\Config\\\\Definition\\\\Builder\\\\NodeDefinition\\:\\:children\\(\\)\\.$#" + count: 1 + path: src/Resources/config/definition/web_client.php + - message: "#^Property SpomkyLabs\\\\PwaBundle\\\\Service\\\\ServiceWorkerCompiler\\:\\:\\$jsonOptions type has no value type specified in iterable type array\\.$#" count: 1 - path: src/Service/ServiceWorkerCompiler.php \ No newline at end of file + path: src/Service/ServiceWorkerCompiler.php + + - + message: "#^Cannot access offset 'public_prefix' on mixed\\.$#" + count: 1 + path: src/SpomkyLabsPwaBundle.php + + - + message: "#^Method SpomkyLabs\\\\PwaBundle\\\\SpomkyLabsPwaBundle\\:\\:loadExtension\\(\\) has parameter \\$config with no value type specified in iterable type array\\.$#" + count: 1 + path: src/SpomkyLabsPwaBundle.php \ No newline at end of file diff --git a/src/Command/CreateServiceWorkerCommand.php b/src/Command/CreateServiceWorkerCommand.php index a5ac1d9..c5f3483 100644 --- a/src/Command/CreateServiceWorkerCommand.php +++ b/src/Command/CreateServiceWorkerCommand.php @@ -5,6 +5,7 @@ namespace SpomkyLabs\PwaBundle\Command; use Symfony\Component\AssetMapper\AssetMapperInterface; +use Symfony\Component\Config\FileLocator; use Symfony\Component\Console\Attribute\AsCommand; use Symfony\Component\Console\Command\Command; use Symfony\Component\Console\Input\InputArgument; @@ -14,7 +15,6 @@ use Symfony\Component\Console\Style\SymfonyStyle; use Symfony\Component\DependencyInjection\Attribute\Autowire; use Symfony\Component\Filesystem\Filesystem; -use Symfony\Component\HttpKernel\Config\FileLocator; use Symfony\Component\Yaml\Yaml; use function count; @@ -24,7 +24,6 @@ final class CreateServiceWorkerCommand extends Command public function __construct( private readonly AssetMapperInterface $assetMapper, private readonly Filesystem $filesystem, - private readonly FileLocator $fileLocator, #[Autowire('%spomky_labs_pwa.asset_public_prefix%')] private readonly string $assetPublicPrefix, #[Autowire('%kernel.project_dir%')] @@ -57,7 +56,8 @@ protected function execute(InputInterface $input, OutputInterface $output): int return self::SUCCESS; } - $resourcePath = $this->fileLocator->locate('@SpomkyLabsPwaBundle/Resources/sw-skeleton.js', null, false); + $fileLocator = new FileLocator(__DIR__ . '/../Resources'); + $resourcePath = $fileLocator->locate('sw-skeleton.js', null, false); if (count($resourcePath) !== 1) { $io->error('Unable to find the Workbox resource.'); return Command::FAILURE; diff --git a/src/DependencyInjection/Configuration.php b/src/DependencyInjection/Configuration.php deleted file mode 100644 index a4a867f..0000000 --- a/src/DependencyInjection/Configuration.php +++ /dev/null @@ -1,828 +0,0 @@ -alias); - $rootNode = $treeBuilder->getRootNode(); - assert($rootNode instanceof ArrayNodeDefinition); - $rootNode->addDefaultsIfNotSet(); - - $this->setupServices($rootNode); - $this->setupManifest($rootNode); - $this->setupServiceWorker($rootNode); - - return $treeBuilder; - } - - private function setupServices(ArrayNodeDefinition $node): void - { - $node->children() - ->integerNode('path_type_reference') - ->defaultValue(UrlGeneratorInterface::ABSOLUTE_PATH) - ->info( - 'The path type reference to generate paths/URLs. See https://symfony.com/doc/current/routing.html#generating-urls-in-controllers for more information.' - ) - ->example( - [ - UrlGeneratorInterface::ABSOLUTE_PATH, - UrlGeneratorInterface::ABSOLUTE_URL, - UrlGeneratorInterface::NETWORK_PATH, - UrlGeneratorInterface::RELATIVE_PATH, - ] - ) - ->validate() - ->ifNotInArray( - [ - UrlGeneratorInterface::ABSOLUTE_PATH, - UrlGeneratorInterface::ABSOLUTE_URL, - UrlGeneratorInterface::NETWORK_PATH, - UrlGeneratorInterface::RELATIVE_PATH, - ] - ) - ->thenInvalid('Invalid path type reference "%s".') - ->end() - ->end() - ->scalarNode('image_processor') - ->defaultNull() - ->info('The image processor to use to generate the icons of different sizes.') - ->example(GDImageProcessor::class) - ->end() - ->scalarNode('asset_public_prefix') - ->cannotBeOverwritten() - ->defaultNull() - ->info('The public prefix of the assets. Shall be the same as the one used in the asset mapper.') - ->end() - ->scalarNode('web_client') - ->defaultNull() - ->info('The Panther Client for generating screenshots. If not set, the default client will be used.') - ->end() - ->end(); - } - - private function setupServiceWorker(ArrayNodeDefinition $node): void - { - $node->children() - ->arrayNode('serviceworker') - ->canBeEnabled() - ->beforeNormalization() - ->ifString() - ->then(static fn (string $v): array => [ - 'enabled' => true, - 'src' => $v, - ]) - ->end() - ->children() - ->scalarNode('src') - ->isRequired() - ->info('The path to the service worker source file. Can be served by Asset Mapper.') - ->example('script/sw.js') - ->end() - ->scalarNode('dest') - ->cannotBeEmpty() - ->defaultValue('/sw.js') - ->info('The public URL to the service worker.') - ->example('/sw.js') - ->end() - ->booleanNode('skip_waiting') - ->defaultFalse() - ->info('Whether to skip waiting for the service worker to be activated.') - ->end() - ->arrayNode('workbox') - ->info('The configuration of the workbox.') - ->canBeDisabled() - ->children() - ->booleanNode('use_cdn') - ->defaultFalse() - ->info('Whether to use the local workbox or the CDN.') - ->end() - ->scalarNode('version') - ->defaultValue('7.0.0') - ->info( - 'The version of workbox. When using local files, the version shall be "7.0.0."' - ) - ->end() - ->scalarNode('workbox_public_url') - ->defaultValue('/workbox') - ->info('The public path to the local workbox. Only used if use_cdn is false.') - ->end() - ->scalarNode('workbox_import_placeholder') - ->setDeprecated( - 'spomky-labs/phpwa', - '1.1.0', - 'The "%node%" option is deprecated and will be removed in 2.0.0. No replacement.' - ) - ->defaultValue('//WORKBOX_IMPORT_PLACEHOLDER') - ->info( - 'The placeholder for the workbox import. Will be replaced by the workbox import.' - ) - ->example('//WORKBOX_IMPORT_PLACEHOLDER') - ->end() - ->scalarNode('standard_rules_placeholder') - ->setDeprecated( - 'spomky-labs/phpwa', - '1.1.0', - 'The "%node%" option is deprecated and will be removed in 2.0.0. No replacement.' - ) - ->defaultValue('//STANDARD_RULES_PLACEHOLDER') - ->info( - 'The placeholder for the standard rules. Will be replaced by caching strategies.' - ) - ->example('//STANDARD_RULES_PLACEHOLDER') - ->end() - ->scalarNode('offline_fallback_placeholder') - ->setDeprecated( - 'spomky-labs/phpwa', - '1.1.0', - 'The "%node%" option is deprecated and will be removed in 2.0.0. No replacement.' - ) - ->defaultValue('//OFFLINE_FALLBACK_PLACEHOLDER') - ->info('The placeholder for the offline fallback. Will be replaced by the URL.') - ->example('//OFFLINE_FALLBACK_PLACEHOLDER') - ->end() - ->scalarNode('widgets_placeholder') - ->setDeprecated( - 'spomky-labs/phpwa', - '1.1.0', - 'The "%node%" option is deprecated and will be removed in 2.0.0. No replacement.' - ) - ->defaultValue('//WIDGETS_PLACEHOLDER') - ->info( - 'The placeholder for the widgets. Will be replaced by the widgets management events.' - ) - ->example('//WIDGETS_PLACEHOLDER') - ->end() - ->booleanNode('clear_cache') - ->defaultTrue() - ->info('Whether to clear the cache during the service worker activation.') - ->end() - ->scalarNode('image_cache_name') - ->defaultValue('images') - ->info('The name of the image cache.') - ->end() - ->scalarNode('font_cache_name') - ->defaultValue('fonts') - ->info('The name of the font cache.') - ->end() - ->scalarNode('page_cache_name') - ->defaultValue('pages') - ->info('The name of the page cache.') - ->end() - ->scalarNode('asset_cache_name') - ->defaultValue('assets') - ->info('The name of the asset cache.') - ->end() - ->append($this->getUrlNode('page_fallback', 'The URL of the offline page fallback.')) - ->append($this->getUrlNode('image_fallback', 'The URL of the offline image fallback.')) - ->append($this->getUrlNode('font_fallback', 'The URL of the offline font fallback.')) - ->scalarNode('image_regex') - ->defaultValue('/\.(ico|png|jpe?g|gif|svg|webp|bmp)$/') - ->info('The regex to match the images.') - ->example('/\.(ico|png|jpe?g|gif|svg|webp|bmp)$/') - ->end() - ->scalarNode('static_regex') - ->defaultValue('/\.(css|js|json|xml|txt|map|webmanifest)$/') - ->info('The regex to match the static files.') - ->example('/\.(css|js|json|xml|txt|woff2|ttf|eot|otf|map|webmanifest)$/') - ->end() - ->scalarNode('font_regex') - ->defaultValue('/\.(ttf|eot|otf|woff2)$/') - ->info('The regex to match the static files.') - ->example('/\.(ttf|eot|otf|woff2)$/') - ->end() - ->integerNode('max_image_cache_entries') - ->defaultValue(60) - ->info('The maximum number of entries in the image cache.') - ->example([50, 100, 200]) - ->end() - ->integerNode('max_image_age') - ->defaultValue(60 * 60 * 24 * 365) - ->info('The maximum number of seconds before the image cache is invalidated.') - ->example([60 * 60 * 24 * 365, 60 * 60 * 24 * 30, 60 * 60 * 24 * 7]) - ->end() - ->integerNode('max_font_cache_entries') - ->defaultValue(30) - ->info('The maximum number of entries in the font cache.') - ->example([30, 50, 100]) - ->end() - ->integerNode('max_font_age') - ->defaultValue(60 * 60 * 24 * 365) - ->info('The maximum number of seconds before the font cache is invalidated.') - ->example([60 * 60 * 24 * 365, 60 * 60 * 24 * 30, 60 * 60 * 24 * 7]) - ->end() - ->integerNode('network_timeout_seconds') - ->defaultValue(3) - ->info( - 'The network timeout in seconds before cache is called (for warm cache URLs only).' - ) - ->example([1, 2, 5]) - ->end() - ->arrayNode('warm_cache_urls') - ->treatNullLike([]) - ->treatFalseLike([]) - ->treatTrueLike([]) - ->info('The URLs to warm the cache. The URLs shall be served by the application.') - ->arrayPrototype() - ->beforeNormalization() - ->ifString() - ->then(static fn (string $v): array => [ - 'path' => $v, - ]) - ->end() - ->children() - ->scalarNode('path') - ->isRequired() - ->info('The URL of the shortcut.') - ->example('app_homepage') - ->end() - ->arrayNode('params') - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->prototype('variable')->end() - ->info('The parameters of the action.') - ->end() - ->end() - ->end() - ->end() - ->end() - ->end() - ->scalarNode('scope') - ->cannotBeEmpty() - ->defaultValue('/') - ->info('The scope of the service worker.') - ->example('/app/') - ->end() - ->booleanNode('use_cache') - ->defaultTrue() - ->info('Whether the service worker should use the cache.') - ->end() - ->end() - ->end() - ->end() - ->end(); - } - - private function setupShortcuts(): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('shortcuts'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - $node - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->info('The shortcuts of the application.') - ->arrayPrototype() - ->children() - ->scalarNode('name') - ->isRequired() - ->info('The name of the shortcut.') - ->example('Awesome shortcut') - ->end() - ->scalarNode('short_name') - ->info('The short name of the shortcut.') - ->example('shortcut') - ->end() - ->scalarNode('description') - ->info('The description of the shortcut.') - ->example('This is an awesome shortcut') - ->end() - ->append($this->getUrlNode('url', 'The URL of the shortcut.')) - ->append($this->getIconsNode('The icons of the shortcut.')) - ->end() - ->end() - ->end(); - - return $node; - } - - private function getFileHandlersNode(): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('file_handlers'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - - $node->info( - 'It specifies an array of objects representing the types of files an installed progressive web app (PWA) can handle.' - ) - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->arrayPrototype() - ->children() - ->append($this->getUrlNode('action', 'The action to take.', ['/handle-audio-file'])) - ->arrayNode('accept') - ->requiresAtLeastOneElement() - ->useAttributeAsKey('name') - ->arrayPrototype() - ->scalarPrototype()->end() - ->end() - ->info('The file types that the action will be applied to.') - ->example('image/*') - ->end() - ->end() - ->end() - ->end(); - - return $node; - } - - private function setupSharedTarget(): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('share_target'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - - $node - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->info('The share target of the application.') - ->children() - ->append( - $this->getUrlNode('action', 'The action of the share target.', ['/shared-content-receiver/']) - ) - ->scalarNode('method') - ->info('The method of the share target.') - ->example('GET') - ->end() - ->scalarNode('enctype') - ->info('The enctype of the share target. Ignored if method is GET.') - ->example('multipart/form-data') - ->end() - ->arrayNode('params') - ->isRequired() - ->info('The parameters of the share target.') - ->children() - ->scalarNode('title') - ->info('The title of the share target.') - ->example('name') - ->end() - ->scalarNode('text') - ->info('The text of the share target.') - ->example('description') - ->end() - ->scalarNode('url') - ->info('The URL of the share target.') - ->example('link') - ->end() - ->arrayNode('files') - ->info('The files of the share target.') - ->scalarPrototype()->end() - ->end() - ->end() - ->end() - ->end() - ->end(); - - return $node; - } - - private function getProtocolHandlersNode(): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('protocol_handlers'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - - $node->info('The protocol handlers of the application.') - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->arrayPrototype() - ->children() - ->scalarNode('protocol') - ->isRequired() - ->info('The protocol of the handler.') - ->example('web+jngl') - ->end() - ->append($this->getUrlNode('url', 'The URL of the handler.')) - ->end() - ->end() - ->end(); - - return $node; - } - - private function getLaunchHandlerNode(): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('launch_handler'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - - $node->info('The launch handler of the application.') - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->children() - ->arrayNode('client_mode') - ->info('The client mode of the application.') - ->example(['focus-existing', 'auto']) - ->scalarPrototype()->end() - ->beforeNormalization() - ->castToArray() - ->end() - ->end() - ->end() - ->end(); - - return $node; - } - - private function setupRelatedApplications(): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('related_applications'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - $node - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->info('The related applications of the application.') - ->arrayPrototype() - ->children() - ->scalarNode('platform') - ->isRequired() - ->info('The platform of the application.') - ->example('play') - ->end() - ->append( - $this->getUrlNode('url', 'The URL of the application.', [ - 'https://play.google.com/store/apps/details?id=com.example.app1', - ]) - ) - ->scalarNode('id') - ->info('The ID of the application.') - ->example('com.example.app1') - ->end() - ->end() - ->end() - ->end(); - - return $node; - } - - private function setupManifest(ArrayNodeDefinition $node): void - { - $node->children() - ->arrayNode('manifest') - ->canBeEnabled() - ->children() - ->scalarNode('public_url') - ->defaultValue('/site.webmanifest') - ->cannotBeEmpty() - ->info('The public URL of the manifest file.') - ->example('/site.manifest') - ->end() - ->scalarNode('background_color') - ->info( - 'The background color of the application. It should match the background-color CSS property in the sites stylesheet for a smooth transition between launching the web application and loading the site\'s content.' - ) - ->example('red') - ->end() - ->arrayNode('categories') - ->info('The categories of the application.') - ->example([['news', 'sports', 'lifestyle']]) - ->scalarPrototype()->end() - ->end() - ->scalarNode('description') - ->info('The description of the application.') - ->example('My awesome application') - ->end() - ->scalarNode('display') - ->info('The display mode of the application.') - ->example('standalone') - ->end() - ->arrayNode('display_override') - ->info( - 'A sequence of display modes that the browser will consider before using the display member.' - ) - ->example([['fullscreen', 'minimal-ui']]) - ->scalarPrototype()->end() - ->end() - ->scalarNode('id') - ->info('A string that represents the identity of the web application.') - ->example('?homescreen=1') - ->end() - ->scalarNode('orientation') - ->info('The orientation of the application.') - ->example('portrait-primary') - ->end() - ->scalarNode('dir') - ->info('The direction of the application.') - ->example('rtl') - ->end() - ->scalarNode('lang') - ->info('The language of the application.') - ->example('ar') - ->end() - ->scalarNode('name') - ->info('The name of the application.') - ->example('My awesome application') - ->end() - ->scalarNode('short_name') - ->info('The short name of the application.') - ->example('awesome_app') - ->end() - ->scalarNode('scope') - ->info('The scope of the application.') - ->example('/app/') - ->end() - ->scalarNode('start_url') - ->info('The start URL of the application.') - ->example('https://example.com') - ->end() - ->scalarNode('theme_color') - ->info('The theme color of the application.') - ->example('red') - ->end() - ->arrayNode('edge_side_panel') - ->info('Specifies whether or not your app supports the side panel view in Microsoft Edge.') - ->children() - ->integerNode('preferred_width') - ->info('Specifies the preferred width of the side panel view in Microsoft Edge.') - ->end() - ->end() - ->end() - ->scalarNode('iarc_rating_id') - ->info( - 'Specifies the International Age Rating Coalition (IARC) rating ID for the app. See https://www.globalratings.com/how-iarc-works.aspx for more information.' - ) - ->end() - ->arrayNode('scope_extensions') - ->info( - 'Specifies a list of origin patterns to associate with. This allows for your app to control multiple subdomains and top-level domains as a single entity.' - ) - ->arrayPrototype() - ->children() - ->scalarNode('origin') - ->isRequired() - ->info('Specifies the origin pattern to associate with.') - ->example('*.foo.com') - ->end() - ->end() - ->end() - ->end() - ->scalarNode('handle_links') - ->info('Specifies the default link handling for the web app.') - ->example(['auto', 'preferred', 'not-preferred']) - ->end() - ->append($this->getIconsNode('The icons of the application.')) - ->append($this->getScreenshotsNode('The screenshots of the application.')) - ->append($this->getFileHandlersNode()) - ->append($this->getLaunchHandlerNode()) - ->append($this->getProtocolHandlersNode()) - ->booleanNode('prefer_related_applications') - ->info('The prefer related native applications of the application.') - ->end() - ->append($this->setupRelatedApplications()) - ->append($this->setupShortcuts()) - ->append($this->setupSharedTarget()) - ->append($this->setupWidgets()) - ->end() - ->end() - ->end(); - } - - private function getIconsNode(string $info): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('icons'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - $node->info($info) - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->arrayPrototype() - ->beforeNormalization() - ->ifString() - ->then(static fn (string $v): array => [ - 'src' => $v, - ]) - ->end() - ->children() - ->scalarNode('src') - ->isRequired() - ->info('The path to the icon. Can be served by Asset Mapper.') - ->example('icon/logo.svg') - ->end() - ->arrayNode('sizes') - ->beforeNormalization() - ->ifTrue(static fn (mixed $v): bool => is_int($v)) - ->then(static fn (int $v): array => [$v]) - ->end() - ->beforeNormalization() - ->ifTrue(static fn (mixed $v): bool => is_string($v)) - ->then(static function (string $v): array { - if ($v === 'any') { - return [0]; - } - - return [(int) $v]; - }) - ->end() - ->info( - 'The sizes of the icon. 16 means 16x16, 32 means 32x32, etc. 0 means "any" (i.e. it is a vector image).' - ) - ->example([['16', '32']]) - ->integerPrototype()->end() - ->end() - ->scalarNode('type') - ->info('The icon mime type.') - ->example(['image/webp', 'image/png']) - ->end() - ->scalarNode('purpose') - ->info('The purpose of the icon.') - ->example(['any', 'maskable', 'monochrome']) - ->end() - ->end() - ->end() - ; - - return $node; - } - - private function getScreenshotsNode(string $info): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('screenshots'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - $node - ->info($info) - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->arrayPrototype() - ->beforeNormalization() - ->ifString() - ->then(static fn (string $v): array => [ - 'src' => $v, - ]) - ->end() - ->children() - ->scalarNode('src') - ->info('The path to the screenshot. Can be served by Asset Mapper.') - ->example('screenshot/lowres.webp') - ->end() - ->scalarNode('height') - ->defaultNull() - ->example('1080') - ->end() - ->scalarNode('width') - ->defaultNull() - ->example('1080') - ->end() - ->scalarNode('form_factor') - ->info('The form factor of the screenshot. Will guess the form factor if not set.') - ->example(['wide', 'narrow']) - ->end() - ->scalarNode('label') - ->info('The label of the screenshot.') - ->example('Homescreen of Awesome App') - ->end() - ->scalarNode('platform') - ->info('The platform of the screenshot.') - ->example( - ['android', 'windows', 'chromeos', 'ipados', 'ios', 'kaios', 'macos', 'windows', 'xbox'] - ) - ->end() - ->scalarNode('format') - ->info('The format of the screenshot. Will convert the file if set.') - ->example(['image/jpg', 'image/png', 'image/webp']) - ->end() - ->end() - ->end(); - - return $node; - } - - private function setupWidgets(): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder('widgets'); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - $node - ->info( - 'EXPERIMENTAL. Specifies PWA-driven widgets. See https://learn.microsoft.com/en-us/microsoft-edge/progressive-web-apps-chromium/how-to/widgets for more information' - ) - ->arrayPrototype() - ->children() - ->scalarNode('name') - ->isRequired() - ->info('The title of the widget, presented to users.') - ->end() - ->scalarNode('short_name') - ->info('An alternative short version of the name.') - ->end() - ->scalarNode('description') - ->isRequired() - ->info('The description of the widget.') - ->example('My awesome widget') - ->end() - ->append( - $this->getIconsNode( - 'An array of icons to be used for the widget. If missing, the icons manifest member is used instead. Icons larger than 1024x1024 are ignored.' - ) - ) - ->append( - $this->getScreenshotsNode('The screenshots of the widget')->requiresAtLeastOneElement() - ) - ->scalarNode('tag') - ->isRequired() - ->info('A string used to reference the widget in the PWA service worker.') - ->end() - ->scalarNode('template') - ->info( - 'The template to use to display the widget in the operating system widgets dashboard. Note: this property is currently only informational and not used. See ms_ac_template below.' - ) - ->end() - ->append( - $this->getUrlNode( - 'ms_ac_template', - 'The URL of the custom Adaptive Cards template to use to display the widget in the operating system widgets dashboard.' - ) - ) - ->append( - $this->getUrlNode( - 'data', - 'The URL where the data to fill the template with can be found. If present, this URL is required to return valid JSON.' - ) - ) - ->scalarNode('type') - ->info('The MIME type for the widget data.') - ->end() - ->booleanNode('auth') - ->info('A boolean indicating if the widget requires authentication.') - ->end() - ->integerNode('update') - ->info( - 'The frequency, in seconds, at which the widget will be updated. Code in your service worker must perform the updating; the widget is not updated automatically. See Access widget instances at runtime.' - ) - ->end() - ->booleanNode('multiple') - ->defaultTrue() - ->info( - 'A boolean indicating whether to allow multiple instances of the widget. Defaults to true.' - ) - ->end() - ->end() - ->end() - ->end(); - - return $node; - } - - /** - * @param array $examples - */ - private function getUrlNode(string $name, string $info, null|array $examples = null): ArrayNodeDefinition - { - $treeBuilder = new TreeBuilder($name); - $node = $treeBuilder->getRootNode(); - assert($node instanceof ArrayNodeDefinition); - $node - ->info($info) - ->beforeNormalization() - ->ifString() - ->then(static fn (string $v): array => [ - 'path' => $v, - ]) - ->end() - ->children() - ->scalarNode('path') - ->isRequired() - ->info('The URL or route name.') - ->example($examples ?? ['https://example.com', 'app_action_route', '/do/action']) - ->end() - ->arrayNode('params') - ->treatFalseLike([]) - ->treatTrueLike([]) - ->treatNullLike([]) - ->prototype('variable')->end() - ->info('The parameters of the action. Only used if the action is a route to a controller.') - ->end() - ->end() - ->end(); - - return $node; - } -} diff --git a/src/DependencyInjection/SpomkyLabsPwaExtension.php b/src/DependencyInjection/SpomkyLabsPwaExtension.php deleted file mode 100644 index 6bddbec..0000000 --- a/src/DependencyInjection/SpomkyLabsPwaExtension.php +++ /dev/null @@ -1,91 +0,0 @@ -processConfiguration($this->getConfiguration($configs, $container), $configs); - $loader = new PhpFileLoader($container, new FileLocator(__DIR__ . '/../Resources/config')); - $loader->load('services.php'); - - if ($config['image_processor'] !== null) { - $container->setAlias(ImageProcessor::class, $config['image_processor']); - } - if ($config['web_client'] !== null) { - $container->setAlias('pwa.web_client', $config['web_client']); - } - $container->setParameter( - 'spomky_labs_pwa.asset_public_prefix', - '/' . trim((string) $config['asset_public_prefix'], '/') - ); - $container->setParameter('spomky_labs_pwa.routes.reference_type', $config['path_type_reference']); - $serviceWorkerConfig = $config['serviceworker']; - $manifestConfig = $config['manifest']; - if ($serviceWorkerConfig['enabled'] === true && $manifestConfig['enabled'] === true) { - $manifestConfig['serviceworker'] = $serviceWorkerConfig; - } - - /*** Manifest ***/ - $container->setParameter('spomky_labs_pwa.manifest.enabled', $config['manifest']['enabled']); - $container->setParameter('spomky_labs_pwa.manifest.public_url', $config['manifest']['public_url'] ?? null); - $container->setParameter('spomky_labs_pwa.manifest.config', $manifestConfig); - - /*** Service Worker ***/ - $container->setParameter('spomky_labs_pwa.sw.enabled', $config['serviceworker']['enabled']); - $container->setParameter('spomky_labs_pwa.sw.public_url', $config['serviceworker']['dest'] ?? null); - $container->setParameter('spomky_labs_pwa.sw.config', $serviceWorkerConfig); - - if (! in_array($container->getParameter('kernel.environment'), ['dev', 'test'], true)) { - $container->removeDefinition(PwaDevServerSubscriber::class); - } - } - - public function getConfiguration(array $config, ContainerBuilder $container): ConfigurationInterface - { - return new Configuration(self::ALIAS); - } - - public function prepend(ContainerBuilder $container): void - { - $bundles = $container->getParameter('kernel.bundles'); - if (isset($bundles['FrameworkBundle'])) { - foreach ($container->getExtensions() as $name => $extension) { - if ($name !== 'framework') { - continue; - } - $config = $container->getExtensionConfig($name); - foreach ($config as $c) { - if (! isset($c['asset_mapper']['public_prefix'])) { - continue; - } - $container->prependExtensionConfig('pwa', [ - 'asset_public_prefix' => $c['asset_mapper']['public_prefix'], - ]); - } - } - } - } -} diff --git a/src/Resources/config/definition/asset_public_prefix.php b/src/Resources/config/definition/asset_public_prefix.php new file mode 100644 index 0000000..1ab6926 --- /dev/null +++ b/src/Resources/config/definition/asset_public_prefix.php @@ -0,0 +1,16 @@ +rootNode() + ->children() + ->scalarNode('asset_public_prefix') + ->cannotBeOverwritten() + ->defaultNull() + ->info('The public prefix of the assets. Shall be the same as the one used in the asset mapper.') + ->end() + ->end(); +}; diff --git a/src/Resources/config/definition/image_processor.php b/src/Resources/config/definition/image_processor.php new file mode 100644 index 0000000..8d1d442 --- /dev/null +++ b/src/Resources/config/definition/image_processor.php @@ -0,0 +1,17 @@ +rootNode() + ->children() + ->scalarNode('image_processor') + ->defaultNull() + ->info('The image processor to use to generate the icons of different sizes.') + ->example(GDImageProcessor::class) + ->end() + ->end(); +}; diff --git a/src/Resources/config/definition/manifest.php b/src/Resources/config/definition/manifest.php new file mode 100644 index 0000000..c86c111 --- /dev/null +++ b/src/Resources/config/definition/manifest.php @@ -0,0 +1,139 @@ +rootNode() + ->children() + ->arrayNode('manifest') + ->canBeEnabled() + ->children() + ->scalarNode('public_url') + ->defaultValue('/site.webmanifest') + ->cannotBeEmpty() + ->info('The public URL of the manifest file.') + ->example('/site.manifest') + ->end() + ->scalarNode('background_color') + ->info( + 'The background color of the application. It should match the background-color CSS property in the sites stylesheet for a smooth transition between launching the web application and loading the site\'s content.' + ) + ->example('red') + ->end() + ->arrayNode('categories') + ->info('The categories of the application.') + ->example([['news', 'sports', 'lifestyle']]) + ->scalarPrototype() + ->end() + ->end() + ->scalarNode('description') + ->info('The description of the application.') + ->example('My awesome application') + ->end() + ->scalarNode('display') + ->info('The display mode of the application.') + ->example('standalone') + ->end() + ->arrayNode('display_override') + ->info( + 'A sequence of display modes that the browser will consider before using the display member.' + ) + ->example([['fullscreen', 'minimal-ui']]) + ->scalarPrototype()->end() + ->end() + ->scalarNode('id') + ->info('A string that represents the identity of the web application.') + ->example('?homescreen=1') + ->end() + ->scalarNode('orientation') + ->info('The orientation of the application.') + ->example('portrait-primary') + ->end() + ->scalarNode('dir') + ->info('The direction of the application.') + ->example('rtl') + ->end() + ->scalarNode('lang') + ->info('The language of the application.') + ->example('ar') + ->end() + ->scalarNode('name') + ->info('The name of the application.') + ->example('My awesome application') + ->end() + ->scalarNode('short_name') + ->info('The short name of the application.') + ->example('awesome_app') + ->end() + ->scalarNode('scope') + ->info('The scope of the application.') + ->example('/app/') + ->end() + ->scalarNode('start_url') + ->info('The start URL of the application.') + ->example('https://example.com') + ->end() + ->scalarNode('theme_color') + ->info('The theme color of the application.') + ->example('red') + ->end() + ->arrayNode('edge_side_panel') + ->info('Specifies whether or not your app supports the side panel view in Microsoft Edge.') + ->children() + ->integerNode('preferred_width') + ->info('Specifies the preferred width of the side panel view in Microsoft Edge.') + ->end() + ->end() + ->end() + ->scalarNode('iarc_rating_id') + ->info( + 'Specifies the International Age Rating Coalition (IARC) rating ID for the app. See https://www.globalratings.com/how-iarc-works.aspx for more information.' + ) + ->end() + ->arrayNode('scope_extensions') + ->info( + 'Specifies a list of origin patterns to associate with. This allows for your app to control multiple subdomains and top-level domains as a single entity.' + ) + ->arrayPrototype() + ->children() + ->scalarNode('origin') + ->isRequired() + ->info('Specifies the origin pattern to associate with.') + ->example('*.foo.com') + ->end() + ->end() + ->end() + ->end() + ->scalarNode('handle_links') + ->info('Specifies the default link handling for the web app.') + ->example(['auto', 'preferred', 'not-preferred']) + ->end() + ->append(getIconsNode('The icons of the application.')) + ->append(getScreenshotsNode('The screenshots of the application.')) + ->append(getFileHandlersNode()) + ->append(getLaunchHandlerNode()) + ->append(getProtocolHandlersNode()) + ->booleanNode('prefer_related_applications') + ->info('The prefer related native applications of the application.') + ->end() + ->append(setupRelatedApplications()) + ->append(setupShortcuts()) + ->append(setupSharedTarget()) + ->append(setupWidgets()) + ->end() + ->end() + ->end(); +}; diff --git a/src/Resources/config/definition/path_type_reference.php b/src/Resources/config/definition/path_type_reference.php new file mode 100644 index 0000000..317d7f9 --- /dev/null +++ b/src/Resources/config/definition/path_type_reference.php @@ -0,0 +1,32 @@ +rootNode() + ->children() + ->integerNode('path_type_reference') + ->defaultValue(UrlGeneratorInterface::ABSOLUTE_PATH) + ->info( + 'The path type reference to generate paths/URLs. See https://symfony.com/doc/current/routing.html#generating-urls-in-controllers for more information.' + ) + ->example([ + UrlGeneratorInterface::ABSOLUTE_PATH, + UrlGeneratorInterface::ABSOLUTE_URL, + UrlGeneratorInterface::NETWORK_PATH, + UrlGeneratorInterface::RELATIVE_PATH, + ]) + ->validate() + ->ifNotInArray([ + UrlGeneratorInterface::ABSOLUTE_PATH, + UrlGeneratorInterface::ABSOLUTE_URL, + UrlGeneratorInterface::NETWORK_PATH, + UrlGeneratorInterface::RELATIVE_PATH, + ]) + ->thenInvalid('Invalid path type reference "%s".') + ->end() + ->end(); +}; diff --git a/src/Resources/config/definition/service_worker.php b/src/Resources/config/definition/service_worker.php new file mode 100644 index 0000000..c16e94a --- /dev/null +++ b/src/Resources/config/definition/service_worker.php @@ -0,0 +1,200 @@ +rootNode() + ->children() + ->arrayNode('serviceworker') + ->canBeEnabled() + ->beforeNormalization() + ->ifString() + ->then(static fn (string $v): array => [ + 'enabled' => true, + 'src' => $v, + ]) + ->end() + ->children() + ->scalarNode('src') + ->isRequired() + ->info('The path to the service worker source file. Can be served by Asset Mapper.') + ->example('script/sw.js') + ->end() + ->scalarNode('dest') + ->cannotBeEmpty() + ->defaultValue('/sw.js') + ->info('The public URL to the service worker.') + ->example('/sw.js') + ->end() + ->booleanNode('skip_waiting') + ->defaultFalse() + ->info('Whether to skip waiting for the service worker to be activated.') + ->end() + ->arrayNode('workbox') + ->info('The configuration of the workbox.') + ->canBeDisabled() + ->children() + ->booleanNode('use_cdn') + ->defaultFalse() + ->info('Whether to use the local workbox or the CDN.') + ->end() + ->scalarNode('version') + ->defaultValue('7.0.0') + ->info('The version of workbox. When using local files, the version shall be "7.0.0."') + ->end() + ->scalarNode('workbox_public_url') + ->defaultValue('/workbox') + ->info('The public path to the local workbox. Only used if use_cdn is false.') + ->end() + ->scalarNode('workbox_import_placeholder') + ->setDeprecated( + 'spomky-labs/phpwa', + '1.1.0', + 'The "%node%" option is deprecated and will be removed in 2.0.0. No replacement.' + ) + ->defaultValue('//WORKBOX_IMPORT_PLACEHOLDER') + ->info('The placeholder for the workbox import. Will be replaced by the workbox import.') + ->example('//WORKBOX_IMPORT_PLACEHOLDER') + ->end() + ->scalarNode('standard_rules_placeholder') + ->setDeprecated( + 'spomky-labs/phpwa', + '1.1.0', + 'The "%node%" option is deprecated and will be removed in 2.0.0. No replacement.' + ) + ->defaultValue('//STANDARD_RULES_PLACEHOLDER') + ->info('The placeholder for the standard rules. Will be replaced by caching strategies.') + ->example('//STANDARD_RULES_PLACEHOLDER') + ->end() + ->scalarNode('offline_fallback_placeholder') + ->setDeprecated( + 'spomky-labs/phpwa', + '1.1.0', + 'The "%node%" option is deprecated and will be removed in 2.0.0. No replacement.' + ) + ->defaultValue('//OFFLINE_FALLBACK_PLACEHOLDER') + ->info('The placeholder for the offline fallback. Will be replaced by the URL.') + ->example('//OFFLINE_FALLBACK_PLACEHOLDER') + ->end() + ->scalarNode('widgets_placeholder') + ->setDeprecated( + 'spomky-labs/phpwa', + '1.1.0', + 'The "%node%" option is deprecated and will be removed in 2.0.0. No replacement.' + ) + ->defaultValue('//WIDGETS_PLACEHOLDER') + ->info( + 'The placeholder for the widgets. Will be replaced by the widgets management events.' + ) + ->example('//WIDGETS_PLACEHOLDER') + ->end() + ->booleanNode('clear_cache') + ->defaultTrue() + ->info('Whether to clear the cache during the service worker activation.') + ->end() + ->scalarNode('image_cache_name') + ->defaultValue('images') + ->info('The name of the image cache.') + ->end() + ->scalarNode('font_cache_name') + ->defaultValue('fonts') + ->info('The name of the font cache.') + ->end() + ->scalarNode('page_cache_name') + ->defaultValue('pages') + ->info('The name of the page cache.') + ->end() + ->scalarNode('asset_cache_name') + ->defaultValue('assets') + ->info('The name of the asset cache.') + ->end() + ->append(getUrlNode('page_fallback', 'The URL of the offline page fallback.')) + ->append(getUrlNode('image_fallback', 'The URL of the offline image fallback.')) + ->append(getUrlNode('font_fallback', 'The URL of the offline font fallback.')) + ->scalarNode('image_regex') + ->defaultValue('/\.(ico|png|jpe?g|gif|svg|webp|bmp)$/') + ->info('The regex to match the images.') + ->example('/\.(ico|png|jpe?g|gif|svg|webp|bmp)$/') + ->end() + ->scalarNode('static_regex') + ->defaultValue('/\.(css|js|json|xml|txt|map|webmanifest)$/') + ->info('The regex to match the static files.') + ->example('/\.(css|js|json|xml|txt|woff2|ttf|eot|otf|map|webmanifest)$/') + ->end() + ->scalarNode('font_regex') + ->defaultValue('/\.(ttf|eot|otf|woff2)$/') + ->info('The regex to match the static files.') + ->example('/\.(ttf|eot|otf|woff2)$/') + ->end() + ->integerNode('max_image_cache_entries') + ->defaultValue(60) + ->info('The maximum number of entries in the image cache.') + ->example([50, 100, 200]) + ->end() + ->integerNode('max_image_age') + ->defaultValue(60 * 60 * 24 * 365) + ->info('The maximum number of seconds before the image cache is invalidated.') + ->example([60 * 60 * 24 * 365, 60 * 60 * 24 * 30, 60 * 60 * 24 * 7]) + ->end() + ->integerNode('max_font_cache_entries') + ->defaultValue(30) + ->info('The maximum number of entries in the font cache.') + ->example([30, 50, 100]) + ->end() + ->integerNode('max_font_age') + ->defaultValue(60 * 60 * 24 * 365) + ->info('The maximum number of seconds before the font cache is invalidated.') + ->example([60 * 60 * 24 * 365, 60 * 60 * 24 * 30, 60 * 60 * 24 * 7]) + ->end() + ->integerNode('network_timeout_seconds') + ->defaultValue(3) + ->info('The network timeout in seconds before cache is called (for warm cache URLs only).') + ->example([1, 2, 5]) + ->end() + ->arrayNode('warm_cache_urls') + ->treatNullLike([]) + ->treatFalseLike([]) + ->treatTrueLike([]) + ->info('The URLs to warm the cache. The URLs shall be served by the application.') + ->arrayPrototype() + ->beforeNormalization() + ->ifString() + ->then(static fn (string $v): array => [ + 'path' => $v, + ]) + ->end() + ->children() + ->scalarNode('path') + ->isRequired() + ->info('The URL of the shortcut.') + ->example('app_homepage') + ->end() + ->arrayNode('params') + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->prototype('variable')->end() + ->info('The parameters of the action.') + ->end() + ->end() + ->end() + ->end() + ->end() + ->end() + ->scalarNode('scope') + ->cannotBeEmpty() + ->defaultValue('/') + ->info('The scope of the service worker.') + ->example('/app/') + ->end() + ->booleanNode('use_cache') + ->defaultTrue() + ->info('Whether the service worker should use the cache.') + ->end() + ->end() + ->end() +->end() + ->end(); +}; diff --git a/src/Resources/config/definition/utils/file_handlers.php b/src/Resources/config/definition/utils/file_handlers.php new file mode 100644 index 0000000..5d0c967 --- /dev/null +++ b/src/Resources/config/definition/utils/file_handlers.php @@ -0,0 +1,38 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + + $node->info( + 'It specifies an array of objects representing the types of files an installed progressive web app (PWA) can handle.' + ) + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->arrayPrototype() + ->children() + ->append(getUrlNode('action', 'The action to take.', ['/handle-audio-file'])) + ->arrayNode('accept') + ->requiresAtLeastOneElement() + ->useAttributeAsKey('name') + ->arrayPrototype() + ->scalarPrototype() + ->end() + ->end() + ->info('The file types that the action will be applied to.') + ->example('image/*') + ->end() + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/utils/icons.php b/src/Resources/config/definition/utils/icons.php new file mode 100644 index 0000000..006ff5a --- /dev/null +++ b/src/Resources/config/definition/utils/icons.php @@ -0,0 +1,65 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + $node->info($info) + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->arrayPrototype() + ->beforeNormalization() + ->ifString() + ->then(static fn (string $v): array => [ + 'src' => $v, + ]) + ->end() + ->children() + ->scalarNode('src') + ->isRequired() + ->info('The path to the icon. Can be served by Asset Mapper.') + ->example('icon/logo.svg') + ->end() + ->arrayNode('sizes') + ->beforeNormalization() + ->ifTrue(static fn (mixed $v): bool => is_int($v)) + ->then(static fn (int $v): array => [$v]) + ->end() + ->beforeNormalization() + ->ifTrue(static fn (mixed $v): bool => is_string($v)) + ->then(static function (string $v): array { + if ($v === 'any') { + return [0]; + } + + return [(int) $v]; + }) + ->end() + ->info( + 'The sizes of the icon. 16 means 16x16, 32 means 32x32, etc. 0 means "any" (i.e. it is a vector image).' + ) + ->example([['16', '32']]) + ->integerPrototype() + ->end() + ->end() + ->scalarNode('type') + ->info('The icon mime type.') + ->example(['image/webp', 'image/png']) + ->end() + ->scalarNode('purpose') + ->info('The purpose of the icon.') + ->example(['any', 'maskable', 'monochrome']) + ->end() + ->end() + ->end() + ; + + return $node; +} diff --git a/src/Resources/config/definition/utils/launch_handler.php b/src/Resources/config/definition/utils/launch_handler.php new file mode 100644 index 0000000..9cb524a --- /dev/null +++ b/src/Resources/config/definition/utils/launch_handler.php @@ -0,0 +1,32 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + + $node->info('The launch handler of the application.') + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->children() + ->arrayNode('client_mode') + ->info('The client mode of the application.') + ->example(['focus-existing', 'auto']) + ->scalarPrototype() + ->end() + ->beforeNormalization() + ->castToArray() + ->end() + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/utils/protocol_handlers.php b/src/Resources/config/definition/utils/protocol_handlers.php new file mode 100644 index 0000000..a035999 --- /dev/null +++ b/src/Resources/config/definition/utils/protocol_handlers.php @@ -0,0 +1,31 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + + $node->info('The protocol handlers of the application.') + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->arrayPrototype() + ->children() + ->scalarNode('protocol') + ->isRequired() + ->info('The protocol of the handler.') + ->example('web+jngl') + ->end() + ->append(getUrlNode('url', 'The URL of the handler.')) + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/utils/related_applications.php b/src/Resources/config/definition/utils/related_applications.php new file mode 100644 index 0000000..e920436 --- /dev/null +++ b/src/Resources/config/definition/utils/related_applications.php @@ -0,0 +1,39 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + $node + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->info('The related applications of the application.') + ->arrayPrototype() + ->children() + ->scalarNode('platform') + ->isRequired() + ->info('The platform of the application.') + ->example('play') + ->end() + ->append( + getUrlNode('url', 'The URL of the application.', [ + 'https://play.google.com/store/apps/details?id=com.example.app1', + ]) + ) + ->scalarNode('id') + ->info('The ID of the application.') + ->example('com.example.app1') + ->end() + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/utils/screenshots.php b/src/Resources/config/definition/utils/screenshots.php new file mode 100644 index 0000000..7a27ad6 --- /dev/null +++ b/src/Resources/config/definition/utils/screenshots.php @@ -0,0 +1,58 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + $node + ->info($info) + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->arrayPrototype() + ->beforeNormalization() + ->ifString() + ->then(static fn (string $v): array => [ + 'src' => $v, + ]) + ->end() + ->children() + ->scalarNode('src') + ->info('The path to the screenshot. Can be served by Asset Mapper.') + ->example('screenshot/lowres.webp') + ->end() + ->scalarNode('height') + ->defaultNull() + ->example('1080') + ->end() + ->scalarNode('width') + ->defaultNull() + ->example('1080') + ->end() + ->scalarNode('form_factor') + ->info('The form factor of the screenshot. Will guess the form factor if not set.') + ->example(['wide', 'narrow']) + ->end() + ->scalarNode('label') + ->info('The label of the screenshot.') + ->example('Homescreen of Awesome App') + ->end() + ->scalarNode('platform') + ->info('The platform of the screenshot.') + ->example(['android', 'windows', 'chromeos', 'ipados', 'ios', 'kaios', 'macos', 'windows', 'xbox']) + ->end() + ->scalarNode('format') + ->info('The format of the screenshot. Will convert the file if set.') + ->example(['image/jpg', 'image/png', 'image/webp']) + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/utils/shared_target.php b/src/Resources/config/definition/utils/shared_target.php new file mode 100644 index 0000000..fda0a03 --- /dev/null +++ b/src/Resources/config/definition/utils/shared_target.php @@ -0,0 +1,56 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + + $node + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->info('The share target of the application.') + ->children() + ->append(getUrlNode('action', 'The action of the share target.', ['/shared-content-receiver/'])) + ->scalarNode('method') + ->info('The method of the share target.') + ->example('GET') + ->end() + ->scalarNode('enctype') + ->info('The enctype of the share target. Ignored if method is GET.') + ->example('multipart/form-data') + ->end() + ->arrayNode('params') + ->isRequired() + ->info('The parameters of the share target.') + ->children() + ->scalarNode('title') + ->info('The title of the share target.') + ->example('name') + ->end() + ->scalarNode('text') + ->info('The text of the share target.') + ->example('description') + ->end() + ->scalarNode('url') + ->info('The URL of the share target.') + ->example('link') + ->end() + ->arrayNode('files') + ->info('The files of the share target.') + ->scalarPrototype() + ->end() + ->end() + ->end() + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/utils/shortcuts.php b/src/Resources/config/definition/utils/shortcuts.php new file mode 100644 index 0000000..96b2dc7 --- /dev/null +++ b/src/Resources/config/definition/utils/shortcuts.php @@ -0,0 +1,40 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + $node + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->info('The shortcuts of the application.') + ->arrayPrototype() + ->children() + ->scalarNode('name') + ->isRequired() + ->info('The name of the shortcut.') + ->example('Awesome shortcut') + ->end() + ->scalarNode('short_name') + ->info('The short name of the shortcut.') + ->example('shortcut') + ->end() + ->scalarNode('description') + ->info('The description of the shortcut.') + ->example('This is an awesome shortcut') + ->end() + ->append(getUrlNode('url', 'The URL of the shortcut.')) + ->append(getIconsNode('The icons of the shortcut.')) + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/utils/url_node.php b/src/Resources/config/definition/utils/url_node.php new file mode 100644 index 0000000..0ba2464 --- /dev/null +++ b/src/Resources/config/definition/utils/url_node.php @@ -0,0 +1,42 @@ + $examples + */ +function getUrlNode(string $name, string $info, null|array $examples = null): ArrayNodeDefinition +{ + $treeBuilder = new TreeBuilder($name); + $node = $treeBuilder->getRootNode(); + assert($node instanceof ArrayNodeDefinition); + $node + ->info($info) + ->beforeNormalization() + ->ifString() + ->then(static fn (string $v): array => [ + 'path' => $v, + ]) + ->end() + ->children() + ->scalarNode('path') + ->isRequired() + ->info('The URL or route name.') + ->example($examples ?? ['https://example.com', 'app_action_route', '/do/action']) + ->end() + ->arrayNode('params') + ->treatFalseLike([]) + ->treatTrueLike([]) + ->treatNullLike([]) + ->prototype('variable') + ->end() + ->info('The parameters of the action. Only used if the action is a route to a controller.') + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/utils/widgets.php b/src/Resources/config/definition/utils/widgets.php new file mode 100644 index 0000000..641c3d1 --- /dev/null +++ b/src/Resources/config/definition/utils/widgets.php @@ -0,0 +1,78 @@ +getRootNode(); + assert($node instanceof ArrayNodeDefinition); + $node + ->info( + 'EXPERIMENTAL. Specifies PWA-driven widgets. See https://learn.microsoft.com/en-us/microsoft-edge/progressive-web-apps-chromium/how-to/widgets for more information' + ) + ->arrayPrototype() + ->children() + ->scalarNode('name') + ->isRequired() + ->info('The title of the widget, presented to users.') + ->end() + ->scalarNode('short_name') + ->info('An alternative short version of the name.') + ->end() + ->scalarNode('description') + ->isRequired() + ->info('The description of the widget.') + ->example('My awesome widget') + ->end() + ->append( + getIconsNode( + 'An array of icons to be used for the widget. If missing, the icons manifest member is used instead. Icons larger than 1024x1024 are ignored.' + ) + ) + ->append(getScreenshotsNode('The screenshots of the widget') ->requiresAtLeastOneElement()) + ->scalarNode('tag') + ->isRequired() + ->info('A string used to reference the widget in the PWA service worker.') + ->end() + ->scalarNode('template') + ->info( + 'The template to use to display the widget in the operating system widgets dashboard. Note: this property is currently only informational and not used. See ms_ac_template below.' + ) + ->end() + ->append( + getUrlNode( + 'ms_ac_template', + 'The URL of the custom Adaptive Cards template to use to display the widget in the operating system widgets dashboard.' + ) + ) + ->append( + getUrlNode( + 'data', + 'The URL where the data to fill the template with can be found. If present, this URL is required to return valid JSON.' + ) + ) + ->scalarNode('type') + ->info('The MIME type for the widget data.') + ->end() + ->booleanNode('auth') + ->info('A boolean indicating if the widget requires authentication.') + ->end() + ->integerNode('update') + ->info( + 'The frequency, in seconds, at which the widget will be updated. Code in your service worker must perform the updating; the widget is not updated automatically. See Access widget instances at runtime.' + ) + ->end() + ->booleanNode('multiple') + ->defaultTrue() + ->info('A boolean indicating whether to allow multiple instances of the widget. Defaults to true.') + ->end() + ->end() + ->end() + ->end(); + + return $node; +} diff --git a/src/Resources/config/definition/web_client.php b/src/Resources/config/definition/web_client.php new file mode 100644 index 0000000..fae787b --- /dev/null +++ b/src/Resources/config/definition/web_client.php @@ -0,0 +1,15 @@ +rootNode() + ->children() + ->scalarNode('web_client') + ->defaultNull() + ->info('The Panther Client for generating screenshots. If not set, the default client will be used.') + ->end() + ->end(); +}; diff --git a/src/SpomkyLabsPwaBundle.php b/src/SpomkyLabsPwaBundle.php index f1e545b..55cbf6d 100644 --- a/src/SpomkyLabsPwaBundle.php +++ b/src/SpomkyLabsPwaBundle.php @@ -4,13 +4,77 @@ namespace SpomkyLabs\PwaBundle; -use SpomkyLabs\PwaBundle\DependencyInjection\SpomkyLabsPwaExtension; -use Symfony\Component\HttpKernel\Bundle\Bundle; +use SpomkyLabs\PwaBundle\ImageProcessor\ImageProcessor; +use SpomkyLabs\PwaBundle\Subscriber\PwaDevServerSubscriber; +use Symfony\Component\Config\Definition\Configurator\DefinitionConfigurator; +use Symfony\Component\DependencyInjection\ContainerBuilder; +use Symfony\Component\DependencyInjection\Loader\Configurator\ContainerConfigurator; +use Symfony\Component\HttpKernel\Bundle\AbstractBundle; +use function in_array; -final class SpomkyLabsPwaBundle extends Bundle +final class SpomkyLabsPwaBundle extends AbstractBundle { - public function getContainerExtension(): SpomkyLabsPwaExtension + protected string $extensionAlias = 'pwa'; + + public function configure(DefinitionConfigurator $definition): void + { + $definition->import('Resources/config/definition/*.php'); + } + + public function loadExtension(array $config, ContainerConfigurator $container, ContainerBuilder $builder): void + { + $container->import('Resources/config/services.php'); + + if ($config['image_processor'] !== null) { + $builder->setAlias(ImageProcessor::class, $config['image_processor']); + } + if ($config['web_client'] !== null) { + $builder->setAlias('pwa.web_client', $config['web_client']); + } + $builder->setParameter('spomky_labs_pwa.routes.reference_type', $config['path_type_reference']); + $serviceWorkerConfig = $config['serviceworker']; + $manifestConfig = $config['manifest']; + if ($serviceWorkerConfig['enabled'] === true && $manifestConfig['enabled'] === true) { + $manifestConfig['serviceworker'] = $serviceWorkerConfig; + } + $builder->setParameter( + 'spomky_labs_pwa.asset_public_prefix', + '/' . trim((string) $config['asset_public_prefix'], '/') + ); + + /*** Manifest ***/ + $builder->setParameter('spomky_labs_pwa.manifest.enabled', $config['manifest']['enabled']); + $builder->setParameter('spomky_labs_pwa.manifest.public_url', $config['manifest']['public_url'] ?? null); + $builder->setParameter('spomky_labs_pwa.manifest.config', $manifestConfig); + + /*** Service Worker ***/ + $builder->setParameter('spomky_labs_pwa.sw.enabled', $config['serviceworker']['enabled']); + $builder->setParameter('spomky_labs_pwa.sw.public_url', $config['serviceworker']['dest'] ?? null); + $builder->setParameter('spomky_labs_pwa.sw.config', $serviceWorkerConfig); + + if (! in_array($builder->getParameter('kernel.environment'), ['dev', 'test'], true)) { + $builder->removeDefinition(PwaDevServerSubscriber::class); + } + } + + public function prependExtension(ContainerConfigurator $container, ContainerBuilder $builder): void { - return new SpomkyLabsPwaExtension(); + $bundles = $builder->getParameter('kernel.bundles'); + if (isset($bundles['FrameworkBundle'])) { + foreach ($builder->getExtensions() as $name => $extension) { + if ($name !== 'framework') { + continue; + } + $config = $builder->getExtensionConfig($name); + foreach ($config as $c) { + if (! isset($c['asset_mapper']['public_prefix'])) { + continue; + } + $builder->prependExtensionConfig('pwa', [ + 'asset_public_prefix' => $c['asset_mapper']['public_prefix'], + ]); + } + } + } } } diff --git a/src/Subscriber/PwaDevServerSubscriber.php b/src/Subscriber/PwaDevServerSubscriber.php index e917913..814e55b 100644 --- a/src/Subscriber/PwaDevServerSubscriber.php +++ b/src/Subscriber/PwaDevServerSubscriber.php @@ -7,10 +7,10 @@ use SpomkyLabs\PwaBundle\Dto\Manifest; use SpomkyLabs\PwaBundle\Dto\ServiceWorker; use SpomkyLabs\PwaBundle\Service\ServiceWorkerCompiler; +use Symfony\Component\Config\FileLocator; use Symfony\Component\DependencyInjection\Attribute\Autowire; use Symfony\Component\EventDispatcher\EventSubscriberInterface; use Symfony\Component\HttpFoundation\Response; -use Symfony\Component\HttpKernel\Config\FileLocator; use Symfony\Component\HttpKernel\Event\RequestEvent; use Symfony\Component\HttpKernel\Event\ResponseEvent; use Symfony\Component\HttpKernel\KernelEvents; @@ -39,7 +39,6 @@ private null|string $workboxVersion; public function __construct( - private FileLocator $fileLocator, private ServiceWorkerCompiler $serviceWorkerBuilder, private SerializerInterface $serializer, private Manifest $manifest, @@ -87,6 +86,8 @@ public function onKernelRequest(RequestEvent $event): void ): $this->serveWorkboxFile($event, $pathInfo); break; + default: + // Do nothing } } @@ -152,8 +153,9 @@ private function serveWorkboxFile(RequestEvent $event, string $pathInfo): void return; } $asset = mb_substr($pathInfo, mb_strlen((string) $this->workboxPublicUrl)); - $resource = sprintf('@SpomkyLabsPwaBundle/Resources/workbox-v%s%s', $this->workboxVersion, $asset); - $resourcePath = $this->fileLocator->locate($resource, null, false); + $fileLocator = new FileLocator(__DIR__ . '/../Resources'); + $resource = sprintf('workbox-v%s%s', $this->workboxVersion, $asset); + $resourcePath = $fileLocator->locate($resource, null, false); if (is_array($resourcePath)) { if (count($resourcePath) === 1) { $resourcePath = $resourcePath[0]; diff --git a/src/Subscriber/WorkboxCompileEventListener.php b/src/Subscriber/WorkboxCompileEventListener.php index af9fe8e..2c56d76 100644 --- a/src/Subscriber/WorkboxCompileEventListener.php +++ b/src/Subscriber/WorkboxCompileEventListener.php @@ -7,9 +7,9 @@ use SpomkyLabs\PwaBundle\Dto\Manifest; use Symfony\Component\AssetMapper\Event\PreAssetsCompileEvent; use Symfony\Component\AssetMapper\Path\PublicAssetsFilesystemInterface; +use Symfony\Component\Config\FileLocator; use Symfony\Component\DependencyInjection\Attribute\Autowire; use Symfony\Component\EventDispatcher\Attribute\AsEventListener; -use Symfony\Component\HttpKernel\Config\FileLocator; use function assert; use function in_array; use function is_array; @@ -24,7 +24,6 @@ public function __construct( #[Autowire('@asset_mapper.local_public_assets_filesystem')] private PublicAssetsFilesystemInterface $assetsFilesystem, private Manifest $manifest, - private FileLocator $fileLocator, ) { } @@ -40,9 +39,8 @@ public function __invoke(PreAssetsCompileEvent $event): void $workboxVersion = $serviceWorker->workbox->version; $workboxPublicUrl = '/' . trim($serviceWorker->workbox->workboxPublicUrl, '/'); - $resourcePath = $this->fileLocator->locate( - sprintf('@SpomkyLabsPwaBundle/Resources/workbox-v%s', $workboxVersion) - ); + $fileLocator = new FileLocator(__DIR__ . '/../Resources'); + $resourcePath = $fileLocator->locate(sprintf('workbox-v%s', $workboxVersion)); if (! is_string($resourcePath)) { return; }