From 0ce43186c80ec48ccc68c3ca0e280bb7895bd2fd Mon Sep 17 00:00:00 2001 From: Rob Frawley 2nd Date: Sat, 3 Sep 2016 19:15:49 -0400 Subject: [PATCH] squash 13 commits squash 12 commits squash 11 commits squash 10 commits license and readme updates readme bugfix readme bugfix readme bugfix readme bugfix bugfix bugfix bugfix removal of editor config file updated to RST docs and README additional docs fixes and updates updated readme for updated RST links updates to RST docs --- Binary/BinaryInterface.php | 9 + Binary/FileBinaryInterface.php | 9 + Binary/Loader/AbstractDoctrineLoader.php | 9 + Binary/Loader/FileSystemLoader.php | 9 + Binary/Loader/FlysystemLoader.php | 9 + Binary/Loader/GridFSLoader.php | 9 + Binary/Loader/LoaderInterface.php | 9 + Binary/Loader/StreamLoader.php | 9 + Binary/MimeTypeGuesserInterface.php | 9 + Binary/SimpleMimeTypeGuesser.php | 9 + CHANGELOG.md | 7 +- Command/RemoveCacheCommand.php | 9 + Command/ResolveCacheCommand.php | 9 + Controller/ImagineController.php | 9 + .../Compiler/FiltersCompilerPass.php | 9 + .../Compiler/LoadersCompilerPass.php | 9 + .../Compiler/PostProcessorsCompilerPass.php | 9 + .../Compiler/ResolversCompilerPass.php | 9 + DependencyInjection/Configuration.php | 9 + .../Loader/FileSystemLoaderFactory.php | 9 + .../Factory/Loader/FlysystemLoaderFactory.php | 9 + .../Factory/Loader/LoaderFactoryInterface.php | 9 + .../Factory/Loader/StreamLoaderFactory.php | 9 + .../Factory/Resolver/AwsS3ResolverFactory.php | 9 + .../Resolver/FlysystemResolverFactory.php | 9 + .../Resolver/ResolverFactoryInterface.php | 9 + .../Resolver/WebPathResolverFactory.php | 9 + DependencyInjection/LiipImagineExtension.php | 9 + Events/CacheResolveEvent.php | 9 + .../Binary/Loader/NotLoadableException.php | 9 + Exception/ExceptionInterface.php | 9 + .../Cache/Resolver/NotResolvableException.php | 9 + .../Cache/Resolver/NotStorableException.php | 9 + .../Filter/NonExistingFilterException.php | 9 + Exception/InvalidArgumentException.php | 9 + Form/Type/ImageType.php | 9 + Imagine/Cache/CacheManager.php | 9 + Imagine/Cache/CacheManagerAwareInterface.php | 9 + Imagine/Cache/CacheManagerAwareTrait.php | 9 + .../Resolver/AbstractFilesystemResolver.php | 9 + Imagine/Cache/Resolver/AmazonS3Resolver.php | 9 + Imagine/Cache/Resolver/AwsS3Resolver.php | 9 + Imagine/Cache/Resolver/CacheResolver.php | 9 + Imagine/Cache/Resolver/FlysystemResolver.php | 9 + .../Cache/Resolver/NoCacheWebPathResolver.php | 9 + Imagine/Cache/Resolver/ProxyResolver.php | 9 + Imagine/Cache/Resolver/ResolverInterface.php | 9 + Imagine/Cache/Resolver/WebPathResolver.php | 9 + Imagine/Cache/Signer.php | 9 + Imagine/Cache/SignerInterface.php | 9 + Imagine/Data/DataManager.php | 9 + .../Data/Transformer/TransformerInterface.php | 9 + Imagine/Filter/FilterConfiguration.php | 9 + Imagine/Filter/FilterManager.php | 9 + .../Filter/Loader/AutoRotateFilterLoader.php | 9 + .../Filter/Loader/BackgroundFilterLoader.php | 9 + Imagine/Filter/Loader/CropFilterLoader.php | 9 + .../Filter/Loader/DownscaleFilterLoader.php | 9 + .../Filter/Loader/GrayscaleFilterLoader.php | 9 + .../Filter/Loader/InterlaceFilterLoader.php | 9 + Imagine/Filter/Loader/LoaderInterface.php | 9 + Imagine/Filter/Loader/PasteFilterLoader.php | 9 + .../Loader/RelativeResizeFilterLoader.php | 9 + Imagine/Filter/Loader/ResizeFilterLoader.php | 9 + Imagine/Filter/Loader/RotateFilterLoader.php | 9 + Imagine/Filter/Loader/ScaleFilterLoader.php | 9 + Imagine/Filter/Loader/StripFilterLoader.php | 9 + .../Filter/Loader/ThumbnailFilterLoader.php | 9 + Imagine/Filter/Loader/UpscaleFilterLoader.php | 9 + .../Filter/Loader/WatermarkFilterLoader.php | 9 + .../ConfigurablePostProcessorInterface.php | 9 + .../PostProcessor/JpegOptimPostProcessor.php | 9 + .../PostProcessor/MozJpegPostProcessor.php | 9 + .../PostProcessor/OptiPngPostProcessor.php | 9 + .../PostProcessor/PngquantPostProcessor.php | 9 + .../PostProcessor/PostProcessorInterface.php | 9 + Imagine/Filter/RelativeResize.php | 9 + ImagineEvents.php | 9 + LICENSE.md | 20 + LiipImagineBundle.php | 9 + Model/Binary.php | 9 + Model/FileBinary.php | 9 + README.md | 428 ++++++++++++----- Resources/doc/basic-usage.rst | 213 +++++++-- Resources/doc/cache-manager.rst | 7 +- Resources/doc/cache-resolver/amazons3.rst | 109 +++-- Resources/doc/cache-resolver/aws_s3.rst | 229 ++++++--- Resources/doc/cache-resolver/cache.rst | 98 +++- Resources/doc/cache-resolver/flysystem.rst | 76 ++- Resources/doc/cache-resolver/proxy.rst | 68 ++- Resources/doc/cache-resolver/web_path.rst | 44 +- Resources/doc/cache-resolvers.rst | 186 ++++++-- Resources/doc/commands.rst | 10 +- Resources/doc/configuration.rst | 2 + Resources/doc/data-loader/filesystem.rst | 27 +- Resources/doc/data-loader/flysystem.rst | 41 +- Resources/doc/data-loader/gridfs.rst | 62 ++- Resources/doc/data-loader/stream.rst | 27 +- Resources/doc/data-loaders.rst | 132 ++++-- Resources/doc/filters.rst | 438 +++++++----------- Resources/doc/index.rst | 34 +- Resources/doc/installation.rst | 41 +- Resources/doc/introduction.rst | 11 +- Resources/doc/post-processors.rst | 434 +++++++++++++++++ Resources/meta/LICENSE | 19 - Templating/Helper/ImagineHelper.php | 9 + Templating/ImagineExtension.php | 9 + Tests/AbstractTest.php | 9 + .../Loader/AbstractDoctrineLoaderTest.php | 9 + Tests/Binary/Loader/FileSystemLoaderTest.php | 9 + Tests/Binary/Loader/FlysystemLoaderTest.php | 9 + Tests/Binary/Loader/GridFSLoaderTest.php | 9 + Tests/Binary/Loader/StreamLoaderTest.php | 9 + Tests/Binary/SimpleMimeTypeGuesserTest.php | 9 + Tests/Controller/ImagineControllerTest.php | 9 + .../Compiler/FiltersCompilerPassTest.php | 9 + .../Compiler/LoadersCompilerPassTest.php | 9 + .../Compiler/ResolversCompilerPassTest.php | 9 + .../DependencyInjection/ConfigurationTest.php | 9 + .../Loader/FileSystemLoaderFactoryTest.php | 9 + .../Loader/FlysystemLoaderFactoryTest.php | 9 + .../Loader/StreamLoaderFactoryTest.php | 9 + .../Resolver/AwsS3ResolverFactoryTest.php | 9 + .../Resolver/FlysystemResolverFactoryTest.php | 9 + .../Resolver/WebPathResolverFactoryTest.php | 9 + .../LiipImagineExtensionTest.php | 9 + Tests/Events/CacheResolveEventTest.php | 9 + .../Resolver/NotResolvableExceptionTest.php | 9 + .../Resolver/NotStorableExceptionTest.php | 9 + Tests/Fixtures/CacheManagerAwareResolver.php | 9 + Tests/Fixtures/Model.php | 9 + Tests/Form/Type/ImageTypeTest.php | 9 + .../Binary/SimpleMimeTypeGuesserTest.php | 9 + Tests/Functional/Command/RemoveCacheTest.php | 9 + Tests/Functional/Command/ResolveCacheTest.php | 9 + .../Controller/ImagineControllerTest.php | 9 + Tests/Functional/ExtensionGuesserTest.php | 9 + .../Resolver/NoCacheWebPathResolverTest.php | 9 + .../Imagine/Data/DataManagerTest.php | 9 + .../Imagine/Filter/FilterManagerTest.php | 9 + .../Loader/DownscaleFilterLoaderTest.php | 9 + .../Loader/GrayscaleFilterLoaderTest.php | 9 + .../Loader/InterlaceFilterLoaderTest.php | 9 + .../Filter/Loader/RotateFilterLoaderTest.php | 9 + Tests/Functional/MimeTypeGuesserTest.php | 9 + Tests/Functional/SignerTest.php | 9 + Tests/Functional/WebTestCase.php | 9 + Tests/Functional/app/AppKernel.php | 9 + Tests/Imagine/Cache/CacheManagerTest.php | 9 + .../Cache/Resolver/AmazonS3ResolverTest.php | 9 + .../Cache/Resolver/AwsS3ResolverTest.php | 9 + .../Cache/Resolver/CacheResolverTest.php | 9 + .../Cache/Resolver/FlysystemResolverTest.php | 9 + .../Resolver/NoCacheWebPathResolverTest.php | 9 + .../Cache/Resolver/ProxyResolverTest.php | 9 + .../Cache/Resolver/WebPathResolverTest.php | 9 + Tests/Imagine/Cache/SignerTest.php | 9 + Tests/Imagine/Data/DataManagerTest.php | 9 + .../Filter/FilterConfigurationTest.php | 9 + Tests/Imagine/Filter/FilterManagerTest.php | 9 + .../Loader/AutoRotateFilterLoaderTest.php | 9 + .../Loader/DownscaleFilterLoaderTest.php | 9 + ...ntCastByRoundDownscaleFilterLoaderTest.php | 9 + ...oIntCastByRoundUpscaleFilterLoaderTest.php | 9 + .../Loader/GrayscaleFilterLoaderTest.php | 9 + .../Loader/InterlaceFilterLoaderTest.php | 9 + .../Filter/Loader/RotateFilterLoaderTest.php | 9 + Tests/LiipImagineBundleTest.php | 9 + Tests/Model/BinaryTest.php | 9 + Tests/Templating/Helper/ImagineHelperTest.php | 9 + Tests/Templating/ImagineExtensionTest.php | 9 + Tests/bootstrap.php | 9 + 172 files changed, 3356 insertions(+), 730 deletions(-) create mode 100644 LICENSE.md create mode 100644 Resources/doc/post-processors.rst delete mode 100644 Resources/meta/LICENSE diff --git a/Binary/BinaryInterface.php b/Binary/BinaryInterface.php index d6f2c5a69..14a72ab00 100644 --- a/Binary/BinaryInterface.php +++ b/Binary/BinaryInterface.php @@ -1,5 +1,14 @@ , GitHub Contributors <[github.com/liip/LiipImagineBundle/graphs/contributors](https://github.com/liip/LiipImagineBundle/graphs/contributors)>** + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of +the Software, and to permit persons to whom the Software is furnished to do so, +subject to the following conditions: + +*The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software.* + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS +FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR +COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER +IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/LiipImagineBundle.php b/LiipImagineBundle.php index a0dbd2a6c..79b7fd38f 100644 --- a/LiipImagineBundle.php +++ b/LiipImagineBundle.php @@ -1,5 +1,14 @@ +``` -This bundle is a fork of AvalancheImagineBundle which provides easy image -manipulation support for Symfony2. The goal of the fork is to make the -code more extensible and as a result applicable for more use cases. -For more details on the reason for the fork see: -https://github.com/avalanche123/AvalancheImagineBundle/pull/25 +### Attribution -For example with this bundle the following is possible: +- Thanks to the many [contributors](https://github.com/liip/LiipImagineBundle/graphs/contributors) + who have dedicated their time and code to this project. + +- The standalone PHP + [Imagine Library](https://github.com/avalanche123/Imagine) + is used by this bundle for image transformations. + +- This package was forked from + [AvalancheImagineBundle](https://github.com/avalanche123/AvalancheImagineBundle) + with the goal of making the code more extensible. Reference + [AvalancheImagineBundle#25](https://github.com/avalanche123/AvalancheImagineBundle/pull/25) + for additional information on the reasoning for this fork. -``` jinja - -``` -This will perform the transformation called `thumbnail`, which you can define -to do a number of different things, such as resizing, cropping, drawing, -masking, etc. +## Setup -This bundle integrates the standalone PHP "[Imagine library](https://github.com/avalanche123/Imagine)". -[![Build Status](https://secure.travis-ci.org/liip/LiipImagineBundle.png)](http://travis-ci.org/liip/LiipImagineBundle) -[![Total Downloads](https://poser.pugx.org/liip/imagine-bundle/downloads.png)](https://packagist.org/packages/liip/imagine-bundle) -[![Latest Stable Version](https://poser.pugx.org/liip/imagine-bundle/v/stable.png)](https://packagist.org/packages/liip/imagine-bundle) +### Installation +Using this package is similar to all Symfony bundles. The following steps must +be performed -## Installation +1. [Download the Bundle](http://symfony.com/doc/master/bundles/LiipImagineBundle/installation.html#step-1-download-the-bundle) +2. [Enable the Bundle](http://symfony.com/doc/master/bundles/LiipImagineBundle/installation.html#step-2-enable-the-bundle) +3. [Register the Routes](http://symfony.com/doc/master/bundles/LiipImagineBundle/installation.html#step-3-register-the-bundle-s-routes) + +Detailed setup instructions can be found in the +[installation](http://symfony.com/doc/master/bundles/LiipImagineBundle/installation.html) +chapter of the documentation. -In case you are not sure how to install this bundle, see the [installation instructions](http://symfony.com/doc/master/bundles/LiipImagineBundle/installation.html). ### Configuration -After installing the bundle, make sure you add this route to your routing: +Detailed information on all available configuration options can be found in the +[configuration](http://symfony.com/doc/master/bundles/LiipImagineBundle/configuration.html) +chapter of the documentation. + + +## Usage Primer + +Generally, this bundle works by applying *filter sets* to images from inside +a template. Your *filter sets* are defined within the application's configuration +file (often `app/config/config.yml`) and are comprised of a collection of +*filters*, *post-processors*, and other optional parameters. + +We'll learn more about *post-processors* and other available parameters later, +but for now lets focus on how to define a simple *filter set* comprised of a +few *filters*. + + +### Create Thumbnails + +Before we get started, there is a small amount of configuration needed to ensure +our [data loaders](http://symfony.com/doc/master/bundles/LiipImagineBundle/data-loaders.html) +and [cache resolvers](http://symfony.com/doc/master/bundles/LiipImagineBundle/cache-resolvers.html) +operate correctly. Use the following boilerplate in your configuration file. + +```yml +# app/config/config.yml + +liip_imagine : + + # configure resolvers + resolvers : + + # setup the default resolver + default : + + # use the default web path + web_path : ~ -``` yaml -# app/config/routing.yml + # your filter sets are defined here + filter_sets : -_liip_imagine: - resource: "@LiipImagineBundle/Resources/config/routing.xml" + # use the default cache configuration + cache : ~ ``` -For a complete configuration drill-down see [the respective chapter in the documentation](http://symfony.com/doc/master/bundles/LiipImagineBundle/configuration.html). +With the basic configuration in place, we'll start with an example that fulfills a +common use-case: creating thumbnails. Lets assume we want the resulting thumbnails +to have the following transformations applied to them: -## Basic Usage +- Scale and crop the image to 120x90px. +- Add a 2px black border to the scaled image. +- Adjust the image quality to 75. -This bundle works by configuring a set of filters and then applying those -filters to images inside a template So, start by creating some sort of filter -that you need to apply somewhere in your application. For example, suppose -you want to thumbnail an image to a size of 120x90 pixels: +Adding onto our boilerplate from above, we need to define a *filter set* (which we'll +name `my_thumb`) with two *filters* configured: the `thumbnail` and `background` +*filters*. -``` yaml +```yml # app/config/config.yml -liip_imagine: - resolvers: - default: - web_path: ~ +liip_imagine : + resolvers : + default : + web_path : ~ + + filter_sets : + cache : ~ + + # the name of the "filter set" + my_thumb : + + # adjust the image quality to 75% + quality : 75 + + # list of transformations to apply (the "filters") + filters : - filter_sets: - cache: ~ - my_thumb: - quality: 75 - filters: - thumbnail: { size: [120, 90], mode: outbound } + # create a thumbnail: set size to 120x90 and use the "outbound" mode + # to crop the image when the size ratio of the input differs + thumbnail : { size : [120, 90], mode : outbound } + + # create a 2px black border: center the thumbnail on a black background + # 4px larger to create a 2px border around the final image + background : { size : [124, 94], position : center, color : '#000000' } ``` -You've now defined a filter set called `my_thumb` that performs a thumbnail transformation. -We'll learn more about available transformations later, but for now, this -new filter can be used immediately in a template: +You've now created a *filter set* called `my_thumb` that performs a thumbnail +transformation. The `thumbnail` filter sizes the image to the desired width +and height (in this example, 120x90px), and its `mode: outbound` option causes +the resulting image to be cropped if the input ratio differs. The `background` +filter results in a 2px black border by creating a black canvas 124x94px in size, +and positioning the thumbnail at its center. + +**Note:** +*A *filter set* can have any number of *filters* defined for it. Simple +transformations may only require a single *filter* while complex +transformations can have an unlimited number of *filters* defined for them.* + +There are a number of additional [filters](http://symfony.com/doc/master/bundles/LiipImagineBundle/filters.html), +but for now you can use your newly defined ``my_thumb`` *filter set* immediately +within a template. -``` jinja +*For Twig-based template, use:* + +```twig ``` -Or if you're using PHP templates: +*Or, for PHP-based template, use:* -``` php +```php ``` -Behind the scenes, the bundles applies the filter(s) to the image on the first -request and then caches the image to a similar path. On the next request, -the cached image would be served directly from the file system. +Behind the scenes, the bundle applies the filter(s) to the image on-the-fly +when the first page request is served. The transformed image is then cached +for subsequent requests. The final cached image path would be similar to +`/media/cache/my_thumb/relative/path/to/image.jpg`. -In this example, the final rendered path would be something like -`/media/cache/my_thumb/relative/path/to/image.jpg`. This is where Imagine -would save the filtered image file. +**Note:** +*Using the ``dev`` environment you might find that images are not properly +rendered via the template helper. This is often caused by having +`intercept_redirect` enabled in your application configuration. To ensure +images are rendered, it is strongly suggested to disable this option: -You can also pass some options at runtime: +```yml +# app/config/config_dev.yml -``` jinja +web_profiler : + intercept_redirects : false +``` + + +### Runtime Options + +Sometime, you may have a filter defined that fulfills 99% of your usage +scenarios. Instead of defining a new filter for the erroneous 1% of cases, +you may instead choose to alter the behavior of a filter at runtime by +passing the template helper an options array. + +*For Twig-based template, use:* + +```twig {% set runtimeConfig = {"thumbnail": {"size": [50, 50] }} %} + ``` -Or if you're using PHP templates: +*Or, for PHP-based template, use:* ```php filter('/relative/path/to/image.jpg', 'my_thumb', $runtimeConfig) ?>" /> ``` -Also you can resolve image url from console: -```jinja -app/console liip:imagine:cache:resolve relative/path/to/image.jpg relative/path/to/image2.jpg --filters=my_thumb --filters=thumbnail_default + + +### Path Resolution + +Sometime you need to resolve the image path returned by this bundle for a +filtered image. This can easily be achieved using Symfony's console binary +or pragmatically from within a controller or other piece of code. + + +#### Resolve with the Console + +You can resolve an image URL using the console command +`liip:imagine:cache:resolve`. The only required argument is one or more +relative image paths (which must be separated by a space). + +```bash +$ php app/console liip:imagine:cache:resolve relative/path/to/image1.jpg relative/path/to/image2.jpg +``` + +Additionally, you can use the ``--filters`` option to specify which filter +you want to resolve for (if the ``--filters`` option is omitted, all +available filters will be resolved). + +```bash +$ php app/console liip:imagine:cache:resolve relative/path/to/image1.jpg --filters=my_thumb ``` -Where only paths required parameter. They are separated by space. If you omit filters option will be applied all available filters. -If you need to access filtered image URL in your controller: -``` php -$this->get('liip_imagine.cache.manager')->getBrowserPath('/relative/path/to/image.jpg', 'my_thumb'), +#### Resolve Pragmatically + +You can resolve the image URL in your code using the `getBrowserPath` +method of the `liip_imagine.cache.manager` service. Assuming you already +have the service assigned to a variable called `$imagineCacheManager`, +you would run: + +```php +$imagineCacheManager->getBrowserPath('/relative/path/to/image.jpg', 'my_thumb'); ``` -In this case, the final rendered path would contain some random data in the path -`/media/cache/my_thumb/S8rrlhhQ/relative/path/to/image.jpg`. This is where Imagine -would save the filtered image file. +Often, you need to perform this operation in a controller. Assuming your +controller inherits from the base Symfony controller, you can take advantage +of the inherited ``get`` method to request the ``liip_imagine.cache.manager`` +service, from which you can call ``getBrowserPath`` on a relative image +path to get its resolved location. -Note: Using the ``dev`` environment you might find that the images are not properly rendered when -using the template helper. This is likely caused by having ``intercept_redirect`` enabled in your -application configuration. To ensure that the images are rendered disable this option: +```php +/** @var CacheManager */ +$imagineCacheManager = $this->get('liip_imagine.cache.manager'); -``` jinja -web_profiler: - intercept_redirects: false +/** @var string */ +$resolvedPath = $this->getBrowserPath('/relative/path/to/image.jpg', 'my_thumb'); ``` + ## Filters -The LiipImagineBundle provides a set of built-in filters. -You may easily roll your own filter, see [the filters chapter in the documentation](http://symfony.com/doc/master/bundles/LiipImagineBundle/filters.html). +This bundle provides a set of built-in filters andy ou may easily +define your own filters as well. Reference the +[filters chapter](http://symfony.com/doc/master/bundles/LiipImagineBundle/filters.html) +from our documentation. -## Using the controller as a service -If you need to use the filters in a controller, you can just load `ImagineController.php` controller as a service and handle the response: +## Use as a Service + +If you need to use your defined "filter sets" from within your controller, you +can fetch this bundle's controller from the service container and handle +the response yourself. + +```php +container - ->get('liip_imagine.controller') + /** @var ImagineController */ + $imagine = $this + ->container + ->get('liip_imagine.controller'); + + /** @var RedirectResponse */ + $imagemanagerResponse = $imagine ->filterAction( $this->request, // http request 'uploads/foo.jpg', // original image you want to apply a filter to 'my_thumb' // filter defined in config.yml ); - // string to put directly in the "src" of the tag - $cacheManager = $this->container->get('liip_imagine.cache.manager'); - $srcPath = $cacheManager->getBrowserPath('uploads/foo.jpg', 'my_thumb'); + /** @var CacheManager */ + $cacheManager = $this + ->container + ->get('liip_imagine.cache.manager'); + + /** @var string */ + $sourcePath = $cacheManager + ->getBrowserPath( + 'uploads/foo.jpg', + 'my_thumb' + ); // .. } } + +?> ``` -In case you need to add more logic the recommended solution is to either extend `ImagineController.php` controller or take the code from that controller as a basis for your own controller. +If you need to add more logic, the recommended solution is to either +extend `ImagineController.php` or use it as the basis for your own +implementation. + +To use the controller in another service, you have to simulate a new request. + +```php +container + ->get('liip_imagine.controller'); -If you want to use the service in another service, you have to simulate a new request: +/** @var Response */ +$response = $imagine + ->filterAction( + new Symfony\Component\HttpFoundation\Request(), + 'uploads/foo.jpg', + 'my_thumb' + ); -``` php -$imagemanagerResponse = $this->container - ->get('liip_imagine.controller') - ->filterAction(new Symfony\Component\HttpFoundation\Request(), 'uploads/foo.jpg', 'my_thumb'); +?> ``` -## Outside the web root -When your setup requires your source images to live outside the web root, or if that's just the way you roll, -you have to set your loader's parameter `data_root` in the `config.yml` with the absolute path where your source images are -located: +## Images Outside Data Root + +When your setup requires your source images reside outside the web root, +you have to set your loader's `data_root` parameter in your configuration +(often `app/config/config.yml`) with the absolute path to your source image +location. -``` yaml +```yml liip_imagine: loaders: default: @@ -188,10 +382,27 @@ liip_imagine: data_root: /path/to/source/images/dir ``` -Afterwards, you need to grant read access on Apache to access the images source directory. For achieving it you have -to add the following directive to your project's vhost file: +This location must be readable by your web server. On a system that supports +`setfacl` (such as Linux/BSD), use -``` xml +```sh +HTTPDUSER=`ps axo user,comm | grep -E '[a]pache|[h]ttpd|[_]www|[w]ww-data|[n]ginx' | grep -v root | head -1 | cut -d\ -f1` + +sudo setfacl -R -m u:"$HTTPDUSER":rwX -m u:`whoami`:rwX /path/to/source/images/dir + +sudo setfacl -dR -m u:"$HTTPDUSER":rwX -m u:`whoami`:rwX /path/to/source/images/dir +``` + +See the [Symfony Permissions documentation](http://symfony.com/doc/current/setup/file_permissions.html) +for commands compatible with macOS and other environments. + + +### Using Apache + +You need to grant read access for Apache by adding the following to your +Apache VHost configuration + +```xml @@ -203,10 +414,12 @@ to add the following directive to your project's vhost file: ``` -Another way would be placing the directive in a separate file living inside your project. For instance, -you can create a file `app/config/apache/photos.xml` and add to the project's vhost the following directive: +Alternatively, you can place the directive in a separate file within your +project, and include it within your Apache VHost configuration. For example, +you can create the file `app/config/apache/photos.xml` and add the following +to your VHost file -``` xml +```xml @@ -214,12 +427,15 @@ you can create a file `app/config/apache/photos.xml` and add to the project's vh ``` -This way you keep the file along with your code and you are able to change your files directory access easily or create -different environment-dependant configuration files. +This method keeps the file with the rest of your code, allowing you to change +it easily or create different environment-dependent configuration files. + +Once you have configured Apache properly, the relative path to an image with +the following absolute path `/path/to/source/images/dir/logo.png` must be +`/FavouriteAlias/logo.png`. -Either way, once you have granted access on Apache to read the `data_root` files, the relative path of an image with this -absolute path `/path/to/source/images/dir/logo.png` must be `/FavouriteAlias/logo.png` to be readable. ## Documentation -For more detailed information about the features of this bundle, please refer to [the documentation](http://symfony.com/doc/master/bundles/LiipImagineBundle/index.html). +For more detailed information about the features of this bundle, refer to +the [documentation](http://symfony.com/doc/master/bundles/LiipImagineBundle). diff --git a/Resources/doc/basic-usage.rst b/Resources/doc/basic-usage.rst index 901fa46e4..732577551 100644 --- a/Resources/doc/basic-usage.rst +++ b/Resources/doc/basic-usage.rst @@ -1,55 +1,148 @@ + Basic Usage =========== -This bundle works by configuring a set of filters and then applying those -filters to images inside a template. So, start by creating some sort of filter -that you need to apply somewhere in your application. For example, suppose -you want to thumbnail an image to a size of 120x90 pixels: +Generally, this bundle works by applying filter sets to images from inside +a template. Your filter sets are defined within the application's configuration +file (often ``app/config/config.yml``) and are comprised of a collection of +filters, post-processors, and other optional parameters. + +We'll learn more about post-processors and other available parameters later, +but for now lets focus on how to define a simple filter set comprised of a +few filters. + + +Create Thumbnails +----------------- + +.. _usage-create-thumbnails: + +Before we get started, there is a small amount of configuration needed to ensure +our :doc:`data loaders ` and :doc:`cache-resolvers ` +operate correctly. Use the following configuration boilerplate. + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine : + + # configure resolvers + resolvers : + + # setup the default resolver + default : + + # use the default web path + web_path : ~ + + # your filter sets are defined here + filter_sets : + + # use the default cache configuration + cache : ~ + +With the basic configuration in place, we'll start with an example that fulfills a +common use-case: creating thumbnails. Lets assume we want the resulting thumbnails +to have the following transformations applied to them: + +* Scale and crop the image to 120x90px. +* Add a 2px black border to the scaled image. +* Adjust the image quality to 75. + +Adding onto our boilerplate from above, we need to define a filter set (which we'll +name ``my_thumb``) with two filters configured: the ``thumbnail`` and ``background`` +filters. .. code-block:: yaml # app/config/config.yml - liip_imagine: - resolvers: - default: - web_path: ~ - - filter_sets: - cache: ~ - my_thumb: - quality: 75 - filters: - thumbnail: { size: [120, 90], mode: outbound } - -You've now defined a filter set called ``my_thumb`` that performs a thumbnail -transformation. We'll learn more about available transformations later, but -for now, this new filter can be used immediately in a template: + + liip_imagine : + resolvers : + default : + web_path : ~ + + filter_sets : + cache : ~ + + # the name of the "filter set" + my_thumb : + + # adjust the image quality to 75% + quality : 75 + + # list of transformations to apply (the "filters") + filters : + + # create a thumbnail: set size to 120x90 and use the "outbound" mode + # to crop the image when the size ratio of the input differs + thumbnail : { size : [120, 90], mode : outbound } + + # create a 2px black border: center the thumbnail on a black background + # 4px larger to create a 2px border around the final image + background : { size : [124, 94], position : center, color : '#000000' } + +You've now created a filter set called ``my_thumb`` that performs a thumbnail +transformation. The ``thumbnail`` filter sizes the image to the desired width +and height (120x90px), and its ``mode: outbound`` option causes +the resulting image to be cropped if the input ratio differs. The ``background`` +filter results in a 2px black border by creating a black canvas 124x94px in size, +and positioning the thumbnail at its center. + +.. note:: + + A filter set can have any number of filters defined for it. Simple + transformations may only require a single filter, while more complex + transformations can have any number of filters defined for them. + +There are a number of additional :doc:`filters `, but for now you can use +your newly defined ``my_thumb`` filter set immediately within a template. .. configuration-block:: - .. code-block:: html+jinja + .. code-block:: html+twig - + .. code-block:: html+php -Behind the scenes, the bundles applies the filter(s) to the image on the -first request and then caches the image to a similar path. On the next request, -the cached image would be served directly from the file system. +Behind the scenes, the bundle applies the filter(s) to the image on-the-fly +when the first page request is served. The transformed image is then cached +for subsequent requests. The final cached image path would be similar to +``/media/cache/my_thumb/relative/path/to/image.jpg``. + +.. note:: + + Using the ``dev`` environment you might find that images are not properly + rendered via the template helper. This is often caused by having + ``intercept_redirect`` enabled in your application configuration. To ensure + images are rendered, it is strongly suggested to disable this option: + + .. code-block:: yaml + + # app/config/config_dev.yml + + web_profiler : + intercept_redirects : false -In this example, the final rendered path would be something like -``/media/cache/my_thumb/relative/path/to/image.jpg``. This is where Imagine -would save the filtered image file. -You can also pass some options at runtime: +Runtime Options +--------------- + +Sometime, you may have a filter defined that fulfills 99% of your usage +scenarios. Instead of defining a new filter for the erroneous 1% of cases, +you may instead choose to alter the behavior of a filter at runtime by +passing the template helper an options array. .. configuration-block:: - .. code-block:: html+jinja + .. code-block:: html+twig {% set runtimeConfig = {"thumbnail": {"size": [50, 50] }} %} + .. code-block:: html+php @@ -62,35 +155,59 @@ You can also pass some options at runtime: ); ?> - + + -Also you can resolve image url from console: +Path Resolution +--------------- + +Sometime you need to resolve the image path returned by this bundle for a +filtered image. This can easily be achieved using Symfony's console binary +or pragmatically from within a controller or other piece of code. + + +Resolve with the Console +~~~~~~~~~~~~~~~~~~~~~~~~ + +You can resolve an image URL using the console command +``liip:imagine:cache:resolve``. The only required argument is one or more +relative image paths (which must be separated by a space). .. code-block:: bash - $ php app/console liip:imagine:cache:resolve relative/path/to/image.jpg relative/path/to/image2.jpg --filters=my_thumb --filters=thumbnail_default + $ php app/console liip:imagine:cache:resolve relative/path/to/image1.jpg relative/path/to/image2.jpg -Where only paths required parameter. They are separated by space. If you -omit filters option will be applied all available filters. +Additionally, you can use the ``--filters`` option to specify which filter +you want to resolve for (if the ``--filters`` option is omitted, all +available filters will be resolved). -If you need to access filtered image URL in your controller: +.. code-block:: bash -.. code-block:: php + $ php app/console liip:imagine:cache:resolve relative/path/to/image1.jpg --filters=my_thumb - $this->get('liip_imagine.cache.manager')->getBrowserPath('/relative/path/to/image.jpg', 'my_thumb') -In this case, the final rendered path would contain some random data in the -path ``/media/cache/my_thumb/S8rrlhhQ/relative/path/to/image.jpg``. This is where -Imagine would save the filtered image file. +Resolve Pragmatically +~~~~~~~~~~~~~~~~~~~~~ -.. note:: +You can resolve the image URL in your code using the ``getBrowserPath`` +method of the ``liip_imagine.cache.manager`` service. Assuming you already +have the service assigned to a variable called ``$imagineCacheManager``, +you would run: - Using the ``dev`` environment you might find that the images are not properly - rendered when using the template helper. This is likely caused by having - ``intercept_redirect`` enabled in your application configuration. To ensure - that the images are rendered disable this option: +.. code-block:: php - .. code-block:: yaml + $imagineCacheManager->getBrowserPath('/relative/path/to/image.jpg', 'my_thumb'); + +Often, you need to perform this operation in a controller. Assuming your +controller inherits from the base Symfony controller, you can take advantage +of the inherited ``get`` method to request the ``liip_imagine.cache.manager`` +service, from which you can call ``getBrowserPath`` on a relative image +path to get its resolved location. + +.. code-block:: php + + /** @var CacheManager */ + $imagineCacheManager = $this->get('liip_imagine.cache.manager'); - web_profiler: - intercept_redirects: false + /** @var string */ + $resolvedPath = $this->getBrowserPath('/relative/path/to/image.jpg', 'my_thumb'); diff --git a/Resources/doc/cache-manager.rst b/Resources/doc/cache-manager.rst index 989bcbac1..75d596545 100644 --- a/Resources/doc/cache-manager.rst +++ b/Resources/doc/cache-manager.rst @@ -1,7 +1,8 @@ -CacheManager -============ -Cache removal +Cache Manager +============= + +Cache Removal ------------- CacheManager allows to remove cache in various ways. diff --git a/Resources/doc/cache-resolver/amazons3.rst b/Resources/doc/cache-resolver/amazons3.rst index bc139dc83..227400d9f 100644 --- a/Resources/doc/cache-resolver/amazons3.rst +++ b/Resources/doc/cache-resolver/amazons3.rst @@ -1,39 +1,50 @@ -AmazonS3Resolver -================ -The AmazonS3Resolver requires the `aws-sdk-php`_ library. Open a command -console, enter your project directory and execute the following command to -download the latest stable version of this library: +Amazon S3 Resolver +================== + +.. _cache-resolver-amazon-s3: + +The ``AmazonS3Resolver`` resolver enables cache resolution using Amazon S3. + +Dependencies +------------ + +This cache resolver requires the `aws-sdk-php`_ library, which can be installed +by executing the following command in your project directory: .. code-block:: bash $ composer require aws/aws-sdk-php -This command requires you to have Composer installed globally, as explained -in the `installation chapter`_ of the Composer documentation. +.. note:: + + This command requires that `Composer`_ is installed globally, as explained in + their `installation documentation`_. -Afterwards, you only need to configure some information regarding your AWS -account and the bucket. +Configuration +------------- + +To begin, you must assign your Amazon key, secret, and bucket to their respective parameters. .. code-block:: yaml + # app/config/config.yml or app/config/parameters.yml + parameters: - amazon_s3.key: 'your-aws-key' - amazon_s3.secret: 'your-aws-secret' - amazon_s3.bucket: 'your-bucket.example.com' + amazon_s3.key : "your-aws-key" + amazon_s3.secret: "your-aws-secret" + amazon_s3.bucket: "your-bucket.example.com" -Now you can set up the services required: +Prerequisites +------------- + +Next, you must define the required services. .. code-block:: yaml + # app/config/services.yml + services: - acme.amazon_s3: - class: AmazonS3 - arguments: - - - key: %amazon_s3.key% - secret: %amazon_s3.secret% - # more S3 specific options, see \AmazonS3::__construct() acme.imagine.cache.resolver.amazon_s3: class: Liip\ImagineBundle\Imagine\Cache\Resolver\AmazonS3Resolver @@ -41,18 +52,52 @@ Now you can set up the services required: - "@acme.amazon_s3" - "%amazon_s3.bucket%" tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'amazon_s3' } + - { name: "liip_imagine.cache.resolver", resolver: "amazon_s3" } -Now you are ready to use the ``AmazonS3Resolver`` by configuring the bundle. -The following example will configure the resolver is default. + acme.amazon_s3: + class: AmazonS3 + arguments: + - + key : "%amazon_s3.key%" + secret: "%amazon_s3.secret%" + +Usage +----- + +After configuring ``AmazonS3Resolver``, you can set it as the default cache resolver +for ``LiipImagineBundle`` using the following configuration. + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + cache: amazon_s3 + + +Usage on a Specific Filter +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Alternatively, you can set ``AmazonS3Resolver`` as the cache resolver for a specific +filter set using the following configuration. .. code-block:: yaml + # app/config/config.yml + liip_imagine: - cache: 'amazon_s3' + filter_sets: + cache: ~ + my_thumb: + cache: amazon_s3 + filters: + # the filter list + +.. tip:: + + If you want to use other buckets for other images, simply alter the parameter + names and create additional services. -If you want to use other buckets for other images, simply alter the parameter -names and create additional services! Object URL Options ------------------ @@ -62,6 +107,8 @@ service, to alter those options you need. .. code-block:: yaml + # app/config/services.yml + services: acme.imagine.cache.resolver.amazon_s3: class: Liip\ImagineBundle\Imagine\Cache\Resolver\AmazonS3Resolver @@ -72,13 +119,15 @@ service, to alter those options you need. # This calls $service->setObjectUrlOption('https', true); - [ setObjectUrlOption, [ 'https', true ] ] tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'amazon_s3' } + - { name: "liip_imagine.cache.resolver", resolver: "amazon_s3" } You can also use the constructor of the resolver to directly inject multiple options. .. code-block:: yaml + # app/config/services.yml + services: acme.imagine.cache.resolver.amazon_s3: class: Liip\ImagineBundle\Imagine\Cache\Resolver\AmazonS3Resolver @@ -88,7 +137,9 @@ options. - "public-read" # AmazonS3::ACL_PUBLIC (default) - { https: true, torrent: true } tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'amazon_s3' } + - { name: "liip_imagine.cache.resolver", resolver: "amazon_s3" } + .. _`aws-sdk-php`: https://github.com/amazonwebservices/aws-sdk-for-php -.. _`installation chapter`: https://getcomposer.org/doc/00-intro.md +.. _`Composer`: https://getcomposer.org/ +.. _`installation documentation`: https://getcomposer.org/doc/00-intro.md diff --git a/Resources/doc/cache-resolver/aws_s3.rst b/Resources/doc/cache-resolver/aws_s3.rst index 5ac8115b8..d039792e5 100644 --- a/Resources/doc/cache-resolver/aws_s3.rst +++ b/Resources/doc/cache-resolver/aws_s3.rst @@ -1,76 +1,109 @@ -AwsS3Resolver -============= -The AwsS3Resolver requires the `aws-sdk-php`_ library. Open a command -console, enter your project directory and execute the following command to -download the latest stable version of this library: +AWS S3 Resolver +=============== + +.. _cache-resolver-aws-s3: + +The ``AwsS3Resolver`` resolver enables cache resolution using Amazon S3. + + +Dependencies +------------ + +This cache resolver requires the `aws-sdk-php`_ library, which can be installed +by executing the following command in your project directory: .. code-block:: bash $ composer require aws/aws-sdk-php -This command requires you to have Composer installed globally, as explained -in the `installation chapter`_ of the Composer documentation. -Afterwards, you only need to configure some information regarding your AWS -account and the bucket. +.. note:: + + This command requires that `Composer`_ is installed globally, as explained in + their `installation documentation`_. + + +Configuration +------------- + +To begin, you must assign your AWS key, secret, bucket, and region to their respective parameters. .. code-block:: yaml + # app/config/config.yml or app/config/parameters.yml + parameters: - amazon.s3.key: 'your-aws-key' - amazon.s3.secret: 'your-aws-secret' - amazon.s3.bucket: 'your-bucket.example.com' - amazon.s3.region: 'your-bucket-region' + amazon.s3.key : "your-aws-key" + amazon.s3.secret: "your-aws-secret" + amazon.s3.bucket: "your-bucket.example.com" + amazon.s3.region: "your-bucket-region" + -Create resolver using factory ------------------------------ +Prerequisites +------------- + +Create Resolver from a Factory +~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ .. code-block:: yaml + # app/config/config.yml + liip_imagine: resolvers: - profile_photos: - aws_s3: - client_config: - key: %amazon.s3.key% - secret: %amazon.s3.secret% - region: %amazon.s3.region% - bucket: %amazon.s3.cache_bucket% - get_options: - Scheme: 'https' - put_options: - CacheControl: 'max-age=86400' - -If you use `aws-sdk-php`_ library version >= 3.0.0 client config credentials -must be an associative array containing key and secret. + profile_photos: + aws_s3: + client_config: + credentials: + key : "%amazon.s3.key%" + secret: "%amazon.s3.secret%" + region: "%amazon.s3.region%" + bucket: "%amazon.s3.cache_bucket%" + get_options: + Scheme: https + put_options: + CacheControl: "max-age=86400" -.. code-block:: yaml - aws_s3: - client_config: - credentials: - key: %amazon.s3.key% - secret: %amazon.s3.secret% - region: %amazon.s3.region% +.. tip:: + + If using `aws-sdk-php`_ < ``3.0.0``, you must omit the ``credentials`` key and instead + place the ``key`` and ``secret`` keys at the same level as ``region`` and ``bucket``. + + .. code-block:: yaml + + # app/config/services.yml + + services: + aws_s3: + client_config: + key : "%amazon.s3.key%" + secret: "%amazon.s3.secret%" + region: "%amazon.s3.region%" + bucket: "%amazon.s3.cache_bucket%" + + # ... -Create resolver as a service ----------------------------- + +Create Resolver as a Service +~~~~~~~~~~~~~~~~~~~~~~~~~~~~ You have to set up the services required: .. code-block:: yaml + # app/config/services.yml + services: acme.amazon_s3: class: Aws\S3\S3Client factory_class: Aws\S3\S3Client - factory_method: factory + factory_method: factory arguments: - - key: %amazon.s3.key% - secret: %amazon.s3.secret% - region: %amazon.s3.region% + credentials: { key: "%amazon.s3.key%", secret: "%amazon.s3.secret%" } + region: "%amazon.s3.region%" acme.imagine.cache.resolver.amazon_s3: class: Liip\ImagineBundle\Imagine\Cache\Resolver\AwsS3Resolver @@ -78,63 +111,99 @@ You have to set up the services required: - "@acme.amazon_s3" - "%amazon.s3.bucket%" tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'profile_photos' } + - { name: "liip_imagine.cache.resolver", resolver: "amazon_s3" } -If you use `aws-sdk-php`_ library version >= 3.0.0 client config credentials -must be an associative array containing key and secret. -.. code-block:: yaml +.. tip:: + + If using `aws-sdk-php`_ < ``3.0.0``, you must omit the ``credentials`` key and instead + place the ``key`` and ``secret`` keys at the same level as ``region`` and ``bucket``. + + .. code-block:: yaml + + # app/config/services.yml + + services: + acme.amazon_s3: + # ... + arguments: + - + key: "%amazon.s3.key%" + secret: "%amazon.s3.secret%" + region: "%amazon.s3.region%" - acme.amazon_s3: - class: Aws\S3\S3Client - factory_class: Aws\S3\S3Client - factory_method: factory - arguments: - - - credentials: { key: %amazon.s3.key%, secret: %amazon.s3.secret% } - region: %amazon.s3.region% Usage ----- -Now you are ready to use the ``AwsS3Resolver`` by configuring the bundle. -The following example will configure the resolver is default. +After configuring ``AwsS3Resolver``, you can set it as the default cache resolver +for ``LiipImagineBundle`` using the following configuration. .. code-block:: yaml + # app/config/config.yml + liip_imagine: cache: profile_photos -If you want to use other buckets for other images, simply alter the parameter -names and create additional services! -Additional options +Usage on a Specific Filter +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Alternatively, you can set ``AmazonS3Resolver`` as the cache resolver for a specific +filter set using the following configuration. + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + filter_sets: + cache: ~ + my_thumb: + cache: profile_photos + filters: + # the filter list + +.. tip:: + + If you want to use other buckets for other images, simply alter the parameter + names and create additional services. + + +Additional Options ------------------ -You can use :doc:`Cache ` and :doc:`Proxy ` resolvers in chain with +You can use :ref:`Cache ` and :ref:`Proxy ` resolvers in chain with current. You just need to configure them with defined options. .. code-block:: yaml + # app/config/config.yml + liip_imagine: resolvers: profile_photos: aws_s3: - ... - proxies: ['http://one.domain.com', 'http://two.domain.com'] + #... + proxies: ["http://one.domain.com", "http://two.domain.com"] cache: true -If enabled both first one will be :doc:`Cache `, then :doc:`Proxy ` -and after all process delegates to AwsS3 resolver. + +If enabled both first one will be :ref:`Cache `, then +:ref:`Proxy ` and after all process delegates to AwsS3 resolver. + Object GET Options ------------------- +~~~~~~~~~~~~~~~~~~ In order to make use of the object GET options, you can simply add a call to the service, to alter those options you need. .. code-block:: yaml + # app/config/services.yml + services: acme.imagine.cache.resolver.amazon_s3: class: Liip\ImagineBundle\Imagine\Cache\Resolver\AwsS3Resolver @@ -143,14 +212,17 @@ service, to alter those options you need. - "%amazon_s3.bucket%" calls: # This calls $service->setGetOption('Scheme', 'https'); - - [ setGetOption, [ 'Scheme', 'https' ] ] + - [ setGetOption, [ Scheme, https ] ] tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'amazon_s3' } + - { name: "liip_imagine.cache.resolver", resolver: "amazon_s3" } + You can also use the constructor of the resolver to directly inject multiple options. .. code-block:: yaml + # app/config/services.yml + services: acme.imagine.cache.resolver.amazon_s3: class: Liip\ImagineBundle\Imagine\Cache\Resolver\AwsS3Resolver @@ -160,10 +232,11 @@ You can also use the constructor of the resolver to directly inject multiple opt - "public-read" # Aws\S3\Enum\CannedAcl::PUBLIC_READ (default) - { Scheme: https } tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'amazon_s3' } + - { name: "liip_imagine.cache.resolver", resolver: "amazon_s3" } + Object PUT Options ------------------- +~~~~~~~~~~~~~~~~~~ Similar to Object GET Options you can configure additional options to be passed to S3 when storing objects. This is useful, for example, to configure Cache- @@ -179,11 +252,14 @@ ignored, even if you configure it via ObjectOptions: * ``Body`` * ``ContentType`` + In order to make use of the object PUT options, you can simply add a call to the service, to alter those options you need. .. code-block:: yaml + # app/config/services.yml + services: acme.imagine.cache.resolver.amazon_s3: class: Liip\ImagineBundle\Imagine\Cache\Resolver\AwsS3Resolver @@ -192,14 +268,17 @@ service, to alter those options you need. - "%amazon_s3.bucket%" calls: # This calls $service->setPutOption('CacheControl', 'max-age=86400'); - - [ setPutOption, [ 'CacheControl', 'max-age=86400' ] ] + - [ setPutOption, [ CacheControl, "max-age=86400" ] ] tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'amazon_s3' } + - { name: "liip_imagine.cache.resolver", resolver: "amazon_s3" } + You can also use the constructor of the resolver to directly inject multiple options. .. code-block:: yaml + # app/config/services.yml + services: acme.imagine.cache.resolver.amazon_s3: class: Liip\ImagineBundle\Imagine\Cache\Resolver\AwsS3Resolver @@ -208,10 +287,12 @@ You can also use the constructor of the resolver to directly inject multiple opt - "%amazon_s3.bucket%" - "public-read" # Aws\S3\Enum\CannedAcl::PUBLIC_READ (default) - { Scheme: https } - - { CacheControl: 'max-age=86400' } + - { CacheControl: "max-age=86400" } tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'amazon_s3' } + - { name: "liip_imagine.cache.resolver", resolver: "amazon_s3" } + .. _`aws-sdk-php`: https://github.com/amazonwebservices/aws-sdk-for-php -.. _`installation chapter`: https://getcomposer.org/doc/00-intro.md +.. _`Composer`: https://getcomposer.org/ +.. _`installation documentation`: https://getcomposer.org/doc/00-intro.md .. _`S3 SDK documentation`: http://docs.aws.amazon.com/aws-sdk-php/latest/class-Aws.S3.S3Client.html#_putObject diff --git a/Resources/doc/cache-resolver/cache.rst b/Resources/doc/cache-resolver/cache.rst index 14bae1751..4eb13a588 100644 --- a/Resources/doc/cache-resolver/cache.rst +++ b/Resources/doc/cache-resolver/cache.rst @@ -1,24 +1,48 @@ -CacheResolver -============= -The ``CacheResolver`` requires the `Doctrine Cache`_ library. +Cache Resolver +============== -This resolver wraps another resolver around a ``Cache``. +.. _cache-resolver-cache: -Now you can set up the services required; by example using the ``AmazonS3Resolver``. +The ``CacheResolver`` cannot be used by itself. Instead, it is a "wrapper" for +another resolver. + + +Dependencies +------------ + +This cache resolver requires the `Doctrine Cache`_ library, which can be installed +by executing the following command in your project directory: + +.. code-block:: bash + + $ composer require doctrine/cache + +.. note:: + + This command requires that `Composer`_ is installed globally, as explained in + their `installation documentation`_. + +Configuration +------------- + +First, you need to setup the required services. In this example we're wrapping an +instance of ``AmazonS3Resolver`` inside this resolver. .. code-block:: yaml + # app/config/services.yml + services: acme.amazon_s3: - class: AmazonS3 + class : AmazonS3 arguments: - - key: %amazon_s3.key% - secret: %amazon_s3.secret% + key : "%amazon_s3.key%" + secret: "%amazon_s3.secret%" acme.imagine.cache.resolver.amazon_s3: - class: Liip\ImagineBundle\Imagine\Cache\Resolver\AmazonS3Resolver + class : Liip\ImagineBundle\Imagine\Cache\Resolver\AmazonS3Resolver arguments: - "@acme.amazon_s3" - "%amazon_s3.bucket%" @@ -26,39 +50,65 @@ Now you can set up the services required; by example using the ``AmazonS3Resolve memcache: class: Memcache calls: - - [ 'connect', [ '127.0.0.1', 11211 ] ] + - [ connect, [ "127.0.0.1", 11211 ] ] cache.memcache: class: Doctrine\Common\Cache\MemcacheCache calls: - - [ 'setMemcache', [ '@memcache' ] ] + - [ setMemcache, [ "@memcache" ] ] - # The actual acme.imagine.cache.resolver.amazon_s3.cache: - class: Liip\ImagineBundle\Imagine\Cache\Resolver\CacheResolver + class : Liip\ImagineBundle\Imagine\Cache\Resolver\CacheResolver arguments: - "@cache.memcache" - "@acme.imagine.cache.resolver.amazon_s3" - - - prefix: "amazon_s3" + - { prefix: "amazon_s3" } tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'cached_amazon_s3' } + - { name: "liip_imagine.cache.resolver", resolver: "cached_amazon_s3" } -There are currently three options available when configuring the ``CacheResolver``: +There are three options available: -* ``global_prefix`` A prefix for all keys within the cache. This is useful to +* ``global_prefix``: A prefix for all keys within the cache. This is useful to avoid colliding keys when using the same cache for different systems. -* ``prefix`` A "local" prefix for this wrapper. This is useful when re-using the +* ``prefix``: A "local" prefix for this wrapper. This is useful when re-using the same resolver for multiple filters. This mainly affects the clear method. -* ``index_key`` The name of the index key being used to save a list of created +* ``index_key``: The name of the index key being used to save a list of created cache keys regarding one image and filter pairing. -Now you are ready to use the ``CacheResolver`` by configuring the bundle. -The following example will configure the resolver is default. + +Usage +----- + +After configuring ``CacheResolver``, you can set it as the default cache resolver +for ``LiipImagineBundle`` using the following configuration. .. code-block:: yaml + # app/config/config.yml + liip_imagine: - cache: 'cached_amazon_s3' + cache: cached_amazon_s3 + + +Usage on a Specific Filter +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Alternatively, you can set ``CacheResolver`` as the cache resolver for a specific +filter set using the following configuration. + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + filter_sets: + cache: ~ + my_thumb: + cache: cached_amazon_s3 + filters: + # the filter list + -.. _`Doctrine Cache`: https://github.com/doctrine/cache \ No newline at end of file +.. _`Doctrine Cache`: https://github.com/doctrine/cache +.. _`Composer`: https://getcomposer.org/ +.. _`installation documentation`: https://getcomposer.org/doc/00-intro.md diff --git a/Resources/doc/cache-resolver/flysystem.rst b/Resources/doc/cache-resolver/flysystem.rst index c755bda0d..7d75fc455 100644 --- a/Resources/doc/cache-resolver/flysystem.rst +++ b/Resources/doc/cache-resolver/flysystem.rst @@ -1,27 +1,48 @@ -FlysystemResolver -================= -This resolver lets you load images onto `Flysystem`_ filesystem abstraction layer, -which can be used in Symfony projects by installing, for example, `OneupFlysystemBundle`_. +Flysystem Resolver +================== -Value of ``filesystem_service`` property must be a service, -which returns an instance of League\\Flysystem\\Filesystem. +.. _cache-resolver-flysystem: -For implementation using `OneupFlysystemBundle`_ look below. +The ``FlysystemResolver`` resolver enabled cache resolution using the `Flysystem`_ +filesystem abstraction layer. -Create resolver ---------------- +Dependencies +------------ + +This cache resolver has a soft dependency on `OneupFlysystemBundle`_, which +can be installed by executing the following command in your project directory: + +.. code-block:: bash + + $ composer require oneup/flysystem-bundle + +.. note:: + + This command requires that `Composer`_ is installed globally, as explained in + their `installation documentation`_. + +Configuration +------------- + +The value of the ``filesystem_service`` property must be a service that returns an +instance of ``League\\Flysystem\\Filesystem``. + +The following implementation uses `OneupFlysystemBundle`_. .. code-block:: yaml + # app/config/config.yml + liip_imagine: resolvers: profile_photos: flysystem: filesystem_service: oneup_flysystem.profile_photos_filesystem - root_url: http://images.example.com - cache_prefix: media/cache - visibility: public + root_url : http://images.example.com + cache_prefix : media/cache + visibility : public + oneup_flysystem: adapters: profile_photos: @@ -34,36 +55,53 @@ Create resolver There are several configuration options available: -* ``root_url`` - must be a valid url to the target system the flysystem adapter +* ``root_url``: must be a valid url to the target system the flysystem adapter points to. This is used to determine how the url should be generated upon request. Default value: ``null`` -* ``cache_prefix`` - this is used for the image path generation. This will be the +* ``cache_prefix``: this is used for the image path generation. This will be the prefix inside the given Flysystem. Default value: ``media/cache`` -* ``visibility`` - one of the two predefined flysystem visibility constants +* ``visibility``: one of the two predefined flysystem visibility constants (``AdapterInterface::VISIBILITY_PUBLIC`` [``public``] / ``AdapterInterface::VISIBILITY_PRIVATE`` [``private``]) The visibility is applied, when the objects are stored on a flysystem filesystem. You will most probably want to leave the default or explicitly set ``public``. Default value: ``public`` + Usage ----- +After configuring ``FlysystemResolver``, you can set it as the default cache resolver +for ``LiipImagineBundle`` using the following configuration. + .. code-block:: yaml + # app/config/config.yml + liip_imagine: cache: profile_photos -Usage on a specific filter --------------------------- + +Usage on a Specific Filter +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Alternatively, you can set it as the cache resolver for a specific filter set using +the following configuration. .. code-block:: yaml + # app/config/config.yml + liip_imagine: filter_sets: cache: ~ my_thumb: cache: profile_photos - quality: 75 filters: - thumbnail: { size: [120, 90], mode: outbound } + # the filter list + + +.. _`Flysystem`: https://github.com/thephpleague/flysystem +.. _`OneupFlysystemBundle`: https://github.com/1up-lab/OneupFlysystemBundle +.. _`Composer`: https://getcomposer.org/ +.. _`installation documentation`: https://getcomposer.org/doc/00-intro.md diff --git a/Resources/doc/cache-resolver/proxy.rst b/Resources/doc/cache-resolver/proxy.rst index 4040a5d71..1b65dbfa0 100644 --- a/Resources/doc/cache-resolver/proxy.rst +++ b/Resources/doc/cache-resolver/proxy.rst @@ -1,30 +1,72 @@ + ProxyResolver ============= -The ProxyResolver is a ``decorator`` for every other Resolver +.. _cache-resolver-proxy: + +The ``ProxyResolver`` cannot be used by itself. Instead, it is a "decorator" for +another resolver. It add the ability to use "Proxy Hosts" for your assets. If no +"Proxy Domains" are set, it behaves like the underlying cache resolver. -This Resolver adds the possibility to use Proxy Hosts for your Assets. If no -Proxy Domains are set, it behaves like the underlying ``Resolver``. +Prerequisites +------------- -Set Proxy Domains ------------------ +Create Service +~~~~~~~~~~~~~~ -In order to use this Resolver you must create a Service and inject some domains -and your underlying Resolver +To use this cache resolver, you must first define the cache resolver it will decorate. +In this example, we will use the :ref:`AWS Cache Resolver `. + +Next, we need to define a service for this cache resolver and inject an array of domains +and the cache resolver service to decorate. .. code-block:: yaml + # app/config/services.yml + services: acme.imagine.cache.resolver.proxy: class: Liip\ImagineBundle\Imagine\Cache\Resolver\ProxyResolver arguments: - "@acme.imagine.cache.resolver.amazon_s3" - - [ 'http://images0.domain.com', 'http://images1.domain.com','http://images2.domain.com' ] + - [ "http://images0.domain.com", "http://images1.domain.com", "http://images2.domain.com" ] tags: - - { name: 'liip_imagine.cache.resolver', resolver: 'proxy' } + - { name: "liip_imagine.cache.resolver", resolver: "proxy" } + +With this configuration, the cache resolver will generate paths such as +``//images0.domain.com/.../image.jpg``, ``//images1.domain.com/.../image.jpg``, and +``//images2.domain.com/.../image.jpg`` (instead of the original path +returned from the decorated cache resolver, in this example using AWS, +``//bucket.s3.awsamazoncloud.com/.../image.jpg``). + +Usage +----- + +After configuring ``ProxyResolver``, you can set it as the default cache resolver +for ``LiipImagineBundle`` using the following configuration. + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + cache: proxy + + +Usage on a Specific Filter +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Alternatively, you can set ``ProxyResolver`` as the cache resolver for a specific +filter set using the following configuration. + +.. code-block:: yaml + # app/config/config.yml -Now your Resolver would generate ``http://images0.domain.com/thumbs/article_thumb/foo.jpg`` -instead of the original path from the underlying Resolver -``bucket.s3.awsamazoncloud.com/thumbs/article_thumb/foo.jpg`` for every relevant -Action. + liip_imagine: + filter_sets: + cache: ~ + my_thumb: + cache: proxy + filters: + # the filter list diff --git a/Resources/doc/cache-resolver/web_path.rst b/Resources/doc/cache-resolver/web_path.rst index f263f8b80..fa18bade2 100644 --- a/Resources/doc/cache-resolver/web_path.rst +++ b/Resources/doc/cache-resolver/web_path.rst @@ -1,8 +1,15 @@ -WebPathResolver -=============== -Create resolver ---------------- +Web Path Resolver +================= + +.. _cache-resolver-web-path: + +The ``WebPathResolver`` resolver enabled cache resolution using the +web path of your application. + + +Configuration +------------- .. code-block:: yaml @@ -10,8 +17,8 @@ Create resolver resolvers: profile_photos: web_path: - web_root: %kernel.root_dir%/../web - cache_prefix: media/cache + web_root: "%kernel.root_dir%/../web" + cache_prefix: "media/cache" There are several configuration options available: @@ -25,10 +32,35 @@ There are several configuration options available: the images would be written to the ``web/media/cache/`` directory. Default value: ``/media/cache`` + Usage ----- +After configuring ``WebPathResolver``, you can set it as the default cache resolver +for ``LiipImagineBundle`` using the following configuration. + .. code-block:: yaml + # app/config/config.yml + liip_imagine: cache: profile_photos + + +Usage on a Specific Filter +~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Alternatively, you can set ``WebPathResolver`` as the cache resolver for a specific +filter set using the following configuration. + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + filter_sets: + cache: ~ + my_thumb: + cache: profile_photos + filters: + # the filter list diff --git a/Resources/doc/cache-resolvers.rst b/Resources/doc/cache-resolvers.rst index bf70d02d6..fa2290fff 100644 --- a/Resources/doc/cache-resolvers.rst +++ b/Resources/doc/cache-resolvers.rst @@ -1,57 +1,169 @@ -Built-In CacheResolver -====================== + +Cache Resolvers +=============== + +A number of built-in cache resolvers are available: .. toctree:: :maxdepth: 1 + :glob: + + cache-resolver/* + + +Set the Default Cache Resolver +------------------------------ + +The default cache is the :ref:`web path cache resolver `, +which caches images under ``/media/cache/`` within your application web root path. + +You can specify the cache resolver to use per individual `filter_sets` or globally. +To set the default cache resolver globally, use: + +.. code-block:: yaml + + # app/config/config.yml - cache-resolver/web_path - cache-resolver/amazons3 - cache-resolver/aws_s3 - cache-resolver/cache - cache-resolver/proxy + liip_imagine: + cache: your_resolver -Changing the default cache resolver ------------------------------------ -The default cache is a web path cache that caches images under -``{web}/media/cache/``. You can specify the cache to use per individual -filter_sets. To change the defaults, you can either change the top level -``cache`` option to the name of the cache resolver you want to use by default, -or redefine the default cache resolver by explicitly defining a resolver called -``default``: +To change the default configuration, you can redefine the default cache resolver +by explicitly defining a resolver called ``default``: .. code-block:: yaml + # app/config/config.yml + liip_imagine: resolvers: default: web_path: cache_prefix: custom_path -Custom cache resolver ---------------------- +To change the cache resolver for a specific ``filter_set``, use the following configuration. -The ImagineBundle allows you to add your custom cache resolver classes. The only -requirement is that each cache resolver loader implement the following interface: -``Liip\ImagineBundle\Imagine\Cache\Resolver\ResolverInterface``. - -To tell the bundle about your new cache resolver, register it in the service -container and apply the ``liip_imagine.cache.resolver`` tag to it (example here -in XML): +.. code-block:: yaml -.. code-block:: xml + # app/config/config.yml - - - - - + liip_imagine: + filter_sets: + cache: ~ + my_thumb: + cache: your_resolver + filters: + # the filter list -For more information on the service container, see the `Symfony Service Container`_ -documentation. +Custom Cache Resolver +--------------------- -You can set your custom cache resolver by adding it to the your configuration as -the new default resolver as follows: +You can easily define your own, custom cache resolvers to handle cache resolution +using any imaginable backend. Creating a custom cache resolver begins by creating +a class that implements the ``ResolverInterface``, as shown below. + +.. code-block:: php + + interface ResolverInterface + { + public function isStored($path, $filter); + public function resolve($path, $filter); + public function store(BinaryInterface $binary, $path, $filter); + public function remove(array $paths, array $filters); + } + +The following is a template for creating your own cache resolver. You must provide +implementations for all methods to create a valid cache resolver. + +.. code-block:: php + + + + + + + + + +.. note:: + + For more information on the Service Container, reference the official + `Symfony Service Container documentation`_. + +Now your custom cache resolver can be set as the global default +using the name defined in the ``resolver`` attribute of the ``tags`` key. .. code-block:: yaml @@ -67,10 +179,6 @@ filter set: filter_sets: my_special_style: cache: my_custom_cache - filters: - my_custom_filter: { } -For an example of a cache resolver implementation, refer to -``Liip\ImagineBundle\Imagine\Cache\Resolver\WebPathResolver``. -.. _`Symfony Service Container`: http://symfony.com/doc/current/book/service_container.html +.. _`Symfony Service Container documentation`: http://symfony.com/doc/current/book/service_container.html diff --git a/Resources/doc/commands.rst b/Resources/doc/commands.rst index d985b7184..4e376b4cf 100644 --- a/Resources/doc/commands.rst +++ b/Resources/doc/commands.rst @@ -1,7 +1,8 @@ -Commands -======== -Remove cache +Console Commands +================ + +Remove Cache ------------ All cache for a given paths will be removed: @@ -29,7 +30,8 @@ without parameters: $ php app/console liip:imagine:cache:remove -Resolve cache + +Resolve Cache ------------- .. code-block:: bash diff --git a/Resources/doc/configuration.rst b/Resources/doc/configuration.rst index 289665322..288b25283 100644 --- a/Resources/doc/configuration.rst +++ b/Resources/doc/configuration.rst @@ -1,3 +1,4 @@ + Configuration ============= @@ -87,4 +88,5 @@ Each filter set that you specify has the following options: * ``animated`` - support for resizing animated gif (currently not supported by Imagine (PR pending)) + .. _`PHP Manual`: http://php.net/imagepng diff --git a/Resources/doc/data-loader/filesystem.rst b/Resources/doc/data-loader/filesystem.rst index 8a03b01c7..8bfff9dae 100644 --- a/Resources/doc/data-loader/filesystem.rst +++ b/Resources/doc/data-loader/filesystem.rst @@ -1,23 +1,38 @@ -Filesystem -========== -Using factory +File System Loader +================== + +.. _data-loaders-filesystem: + +The ``FileSystem`` data loader allows for loading images from local file system paths. + +.. tip:: + + If you don't configure anything, this loader is used by default. + + +Configuration ------------- +To set this loader for a specific context called ``profile_photos``, use: + .. code-block:: yaml + # app/config/config.yml + liip_imagine: loaders: profile_photos: filesystem: ~ -If you don't configure anything, this loader is set by default. You can -also configure a root dir where to look for the origin images: +You can configure the ``data_root``, used as the root path to search for images: .. code-block:: yaml + # app/config/config.yml + liip_imagine: loaders: profile_photos: filesystem: - data_root: %kernel.root_dir%/../web + data_root: "%kernel.root_dir%/../web" diff --git a/Resources/doc/data-loader/flysystem.rst b/Resources/doc/data-loader/flysystem.rst index b41706d6d..ffd48d484 100644 --- a/Resources/doc/data-loader/flysystem.rst +++ b/Resources/doc/data-loader/flysystem.rst @@ -1,19 +1,37 @@ -FlysystemLoader -=============== -This loader lets you load images from `Flysystem`_ filesystem abstraction layer, -which can be used in Symfony projects by installing, for example, `OneupFlysystemBundle`_. +FlySystem Loader +================ -Value of ``filesystem_service`` property must be a service, -which returns an instance of League\\Flysystem\\Filesystem. +.. _data-loaders-flysystem: -For implementation using `OneupFlysystemBundle`_ look below. +The ``FlysystemLoader`` lets you load images using the `Flysystem`_ filesystem abstraction +layer. -Using factory +Dependencies +------------ + +This cache resolver has a soft dependency on `OneupFlysystemBundle`_, which +can be installed by executing the following command in your project directory: + +.. code-block:: bash + + $ composer require oneup/flysystem-bundle + +.. note:: + + This command requires that `Composer`_ is installed globally, as explained in + their `installation documentation`_. + + +Configuration ------------- +Using `OneupFlysystemBundle`_, a basic configuration might look like the following. + .. code-block:: yaml + # app/config/config.yml + liip_imagine: loaders: profile_photos: @@ -30,6 +48,13 @@ Using factory profile_photos: adapter: profile_photos +.. note:: + + The value of ``filesystem_service`` must be a service id that returns an instance + of ``League\\Flysystem\\Filesystem``. + .. _`Flysystem`: https://github.com/thephpleague/flysystem .. _`OneupFlysystemBundle`: https://github.com/1up-lab/OneupFlysystemBundle +.. _`Composer`: https://getcomposer.org/ +.. _`installation documentation`: https://getcomposer.org/doc/00-intro.md diff --git a/Resources/doc/data-loader/gridfs.rst b/Resources/doc/data-loader/gridfs.rst index dd31c5db7..c7c97fbf1 100644 --- a/Resources/doc/data-loader/gridfs.rst +++ b/Resources/doc/data-loader/gridfs.rst @@ -1,10 +1,18 @@ -GridFSLoader -============ -Load your images from `MongoDB GridFS`_. +GridFS Loader +============= + +.. _data-loaders-grid-fs: + +The ``GridFSLoader`` allows you to load your images from `MongoDB GridFS`_. + +Configuration +------------- .. code-block:: yaml + # app/config/config.yml + liip_imagine: filter_sets: my_special_style: @@ -12,20 +20,48 @@ Load your images from `MongoDB GridFS`_. filters: my_custom_filter: { } -Add loader to your services: +Define a service for the loader: + +.. configuration-block:: + + .. code-block:: yaml + + # app/config/services.yml + + services: + liip_imagine.binary.loader.grid_fs: + class: Liip\ImagineBundle\Binary\Loader\GridFSLoader + arguments: + - "@doctrine.odm.mongodb.document_manager" + - Application\ImageBundle\Document\Image + tags: + - { name: "liip_imagine.binary.loader", loader: grid_fs } + + .. code-block:: xml + + + + + + + Application\ImageBundle\Document\Image + + + +Usage +----- + +Reference the image by its ``id`` when piping to the template helper: + +.. configuration-block:: -.. code-block:: xml + .. code-block:: html+twig - - - - Application\ImageBundle\Document\Image - + -Reference the image by its id: + .. code-block:: html+php -.. code-block:: jinja + - .. _`MongoDB GridFS`: http://docs.mongodb.org/manual/applications/gridfs/ diff --git a/Resources/doc/data-loader/stream.rst b/Resources/doc/data-loader/stream.rst index 70898fc29..b20e99ba7 100644 --- a/Resources/doc/data-loader/stream.rst +++ b/Resources/doc/data-loader/stream.rst @@ -1,7 +1,13 @@ -StreamLoader -============ -Using factory +Stream Loader +============= + +.. _data-loaders-stream: + +The ``StreamLoader`` allows you to load images using PHP Streams. + + +Configuration ------------- .. code-block:: yaml @@ -12,8 +18,9 @@ Using factory stream: wrapper: gaufrette://profile_photos + Custom ------- +~~~~~~ The ``Liip\ImagineBundle\Binary\Loader\StreamLoader`` allows to read images from any stream (http, ftp, and others…) registered thus allowing you to serve your images from @@ -21,14 +28,16 @@ literally anywhere. The example service definition shows how to use a stream wrapped by the `Gaufrette`_ filesystem abstraction layer. In order to have this example -working, you need to register the stream wrapper first, refer to the `Gaufrette -README`_ on how to do this. +working, you need to register the stream wrapper first, refer to the `Gaufrette README`_ +on how to do this. If you are using the `KnpGaufretteBundle`_ you can make use of the `StreamWrapper configuration`_ to register the filesystems. .. code-block:: yaml + # app/config/services.yml + services: acme.liip_imagine.binary.loader.stream.profile_photos: class: "%liip_imagine.binary.loader.stream.class%" @@ -37,6 +46,7 @@ If you are using the `KnpGaufretteBundle`_ you can make use of the tags: - { name: 'liip_imagine.binary.loader', loader: 'stream.profile_photos' } + Usage ----- @@ -45,10 +55,13 @@ The following example will configure the resolver as default. .. code-block:: yaml + # app/config/config.yml + liip_imagine: data_loader: stream.profile_photos + +.. _`StreamWrapper configuration`: https://github.com/KnpLabs/KnpGaufretteBundle#stream-wrapper .. _`Gaufrette`: https://github.com/KnpLabs/Gaufrette .. _`Gaufrette README`: https://github.com/KnpLabs/Gaufrette/blob/master/README.markdown .. _`KnpGaufretteBundle`: https://github.com/KnpLabs/KnpGaufretteBundle -.. _`StreamWrapper configuration`: https://github.com/KnpLabs/KnpGaufretteBundle#stream-wrapper diff --git a/Resources/doc/data-loaders.rst b/Resources/doc/data-loaders.rst index f05de8a8b..649a45041 100644 --- a/Resources/doc/data-loaders.rst +++ b/Resources/doc/data-loaders.rst @@ -1,62 +1,130 @@ -Built-In DataLoader -=================== + +Data Loaders +============ + +A number of built-in data loaders are available: .. toctree:: :maxdepth: 1 + :glob: + + data-loader/* + + +Other Data Loaders +------------------ - data-loader/filesystem - data-loader/gridfs - data-loader/stream - data-loader/flysystem +* `Doctrine PHPCR-ODM`_: You can include the ``CmfMediaBundle`` alone if you just + want to use the images but no other Symfony CMF features. -Other data loaders + +Custom Data Loader ------------------ -* `Doctrine PHPCR-ODM`_: you can include the CmfMediaBundle alone if you just - want to use the images but no other CMF features. +.. _data-loaders-custom: + +You can easily define your own, custom data loaders to allow you to retrieve you +image data from any imaginable backend. Creating a custom data loader begins by creating +a class that implements the ``LoaderInterface``, as shown below. + +.. code-block:: php + + interface LoaderInterface + { + public function find($path); + } + +As defined in ``LoaderInterface``, the only required method is one named ``find``, +which is provided a relative image path as its singular parameter, and +subsequently provides an instance of ``BinaryInterface`` in return. + +The following is a template for creating your own data loader. You must provide +the implementation for the ``find`` method to create a valid data loader. -Custom image loaders --------------------- +.. code-block:: php -The ImagineBundle allows you to add your custom image loader classes. The only -requirement is that each data loader implements the following interface: -``Liip\ImagineBundle\Binary\Loader\LoaderInterface``. + namespace AppBundle\Imagine\Binary\Loader; -To tell the bundle about your new data loader, register it in the service -container and apply the ``liip_imagine.binary.loader`` tag to it (example here -in XML): + use Liip\ImagineBundle\Binary\BinaryInterface; + use Liip\ImagineBundle\Binary\Loader\LoaderInterface; + use Liip\ImagineBundle\Model\Binary; -.. code-block:: xml + class MyCustomDataLoader implements LoaderInterface + { + /** + * @param mixed $path + * + * @return BinaryInterface + */ + public function find($path) + { + $data = /** @todo: implement logic to read image data */ + $mime = /** @todo: implement logic to determine image mime-type */ - - - - + // return binary instance with data + return new Binary($data, $mime); + } + } -For more information on the service container, see the `Symfony Service Container`_ -documentation. +Once you have defined your custom data loader, you must define it as a service and tag it +with ``liip_imagine.binary.loader``. + +.. note:: + + For more information on the Service Container, reference the official + `Symfony Service Container documentation`_. + +To register ``AppBundle\Imagine\Binary\Loader\MyCustomDataLoader`` with the name +``my_custom_data_loader``, you would use the following configuration. + +.. configuration-block:: + + .. code-block:: yaml + + # app/config/services.yml + + services: + imagine.data.loader.my_custom: + class: AppBundle\Imagine\Binary\Loader\MyCustomDataLoader + arguments: + - "@liip_imagine" + - "%liip_imagine.formats%" + tags: + - { name: "liip_imagine.data.loader", loader: my_custom_data_loader } + + .. code-block:: xml + + + + + + + + You can set your custom data loader by adding it to the configuration as the new default -loader as follows: +loader: .. code-block:: yaml + # app/config/config.yml + liip_imagine: - data_loader: my_custom_data + data_loader: my_custom_data_loader -Alternatively you can only set the custom data loader for just a specific filter set: +Alternatively, you can only set the custom data loader for just a specific filter set: .. code-block:: yaml + # app/config/config.yml + liip_imagine: filter_sets: my_special_style: - data_loader: my_custom_data + data_loader: my_custom_data_loader filters: - my_custom_filter: { } + # your filters -For an example of a data loader implementation, refer to -``Liip\ImagineBundle\Binary\Loader\FileSystemLoader``. +.. _`Symfony Service Container documentation`: http://symfony.com/doc/current/book/service_container.html .. _`Doctrine PHPCR-ODM`: http://symfony.com/doc/master/cmf/bundles/media.html#liipimagine -.. _`Symfony Service Container`: http://symfony.com/doc/current/book/service_container.html diff --git a/Resources/doc/filters.rst b/Resources/doc/filters.rst index 7fb92a6c4..6bd0b48d4 100644 --- a/Resources/doc/filters.rst +++ b/Resources/doc/filters.rst @@ -1,14 +1,37 @@ + Filters ======= +Filters perform image transformation operations in the sequence they are defined +for the respective filter set. + + Built-in Filters ---------------- -The ``thumbnail`` filter -~~~~~~~~~~~~~~~~~~~~~~~~ +A number of built-in filters are provided by default. + +* :ref:`Thumbnail ` +* :ref:`Relative Resize ` +* :ref:`Scale ` +* :ref:`Up-Scale ` +* :ref:`Down-Scale ` +* :ref:`Crop ` +* :ref:`Strip ` +* :ref:`Background ` +* :ref:`Watermark ` +* :ref:`Auto Rotate ` +* :ref:`Rotate ` +* :ref:`Interlace ` +* :ref:`Grayscale ` + -The thumbnail filter, as the name implies, performs a thumbnail transformation -on your image. +Thumbnail +~~~~~~~~~ + +.. _filter-thumbnail: + +The ``thumbnail`` filter performs a thumbnail transformation on your image. The ``mode`` can be either ``outbound`` or ``inset``. Option ``inset`` does a relative resize, where the height and the width will not exceed the values in @@ -16,7 +39,7 @@ the configuration. Option ``outbound`` does a relative resize, but the image gets cropped if width and height are not the same. Given an input image sized 50x40 (width x height), consider the following -annotated configuration examples: +annotated configuration examples. .. code-block:: yaml @@ -38,15 +61,18 @@ example above will be expanded to the requested size by interpolation of its content. Without this option, a smaller image will be left as it. This means you may get images that are smaller than the specified dimensions. -The ``relative_resize`` filter -~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ + +Relative Resize +~~~~~~~~~~~~~~~ + +.. _filter-relative-resize: The ``relative_resize`` filter may be used to ``heighten``, ``widen``, ``increase`` or ``scale`` an image with respect to its existing dimensions. These options directly correspond to methods on Imagine's ``BoxInterface``. Given an input image sized 50x40 (width, height), consider the following -annotated configuration examples: +annotated configuration examples. .. code-block:: yaml @@ -65,11 +91,14 @@ annotated configuration examples: filters: relative_resize: { scale: 2.5 } # Transforms 50x40 to 125x100 -The ``scale`` filter -~~~~~~~~~~~~~~~~~~~~~~ -It performs an upscale or downscale transformation on your image to increase its size to the -given dimensions or ratio: +Scale +~~~~~ + +.. _filter-scale: + +The ``scale`` filter performs an upscale or downscale transformation on your +image to increase its size to the given dimensions or ratio. .. code-block:: yaml @@ -80,11 +109,13 @@ given dimensions or ratio: scale: { dim: [600, 750] } #or { to: 1.56 } -> Upscales to [936, 1170] | { to: 0.66 } -> Downscales to [396, 495] -The ``upscale`` filter -~~~~~~~~~~~~~~~~~~~~~~ +Up-Scale +~~~~~~~~ -It performs an upscale transformation on your image to increase its size to the -given dimensions or ratio: +.. _filter-up-scale: + +The ``upscale`` filter performs an upscale transformation on your image to increase its size to the +given dimensions or ratio. .. code-block:: yaml @@ -94,10 +125,13 @@ given dimensions or ratio: filters: upscale: { min: [800, 600] } #or { by: 0.7 } -> Upscales to [1360, 1020] -The ``downscale`` filter -~~~~~~~~~~~~~~~~~~~~~~~~ -It performs a downscale transformation on your image to reduce its size to the +Down-Scale +~~~~~~~~~~ + +.. _filter-down-scale: + +The ``downscale`` filter performs a downscale transformation on your image to reduce its size to the given dimensions or ratio: .. code-block:: yaml @@ -108,10 +142,13 @@ given dimensions or ratio: filters: downscale: { max: [1980, 1280] } #or { by: 0.6 } -> Downscales to [792, 512] -The ``crop`` filter -~~~~~~~~~~~~~~~~~~~ -It performs a crop transformation on your image. The ``start`` option defines +Crop +~~~~ + +.. _filter-crop: + +The ``crop`` filter performs a crop transformation on your image. The ``start`` option defines the coordinates of the left-top pixel where the crop begins (the ``[0, 0]`` coordinates correspond to the top leftmost pixel of the original image). The ``size`` option defines in pixels the width and height (in this order) of the @@ -125,12 +162,15 @@ area cropped: filters: crop: { start: [10, 20], size: [120, 90] } -The ``strip`` filter -~~~~~~~~~~~~~~~~~~~~ -It removes all profiles and comments from your image to reduce its file size +Strip +~~~~~ + +.. _filter-strip: + +The ``strip`` filter removes all profiles and comments from your image to reduce its file size without degrading its quality. This filter provides no configuration options, -so you just need to enable it as follows: +so you just need to enable it. .. code-block:: yaml @@ -140,10 +180,13 @@ so you just need to enable it as follows: filters: strip: ~ -The ``background`` filter -~~~~~~~~~~~~~~~~~~~~~~~~~ -It sets a background color for the image. The default color is white (``#FFF``): +Background +~~~~~~~~~~ + +.. _filter-background: + +The ``background`` filter adds a background color for the image. The default color is white (``#FFF``). .. code-block:: yaml @@ -155,7 +198,7 @@ It sets a background color for the image. The default color is white (``#FFF``): By default, the background color is only visible through the transparent sections of the image (if any). However, if you provide a ``size`` option, a new image is -created (with the given size and color) and the original image is placed on top: +created (with the given size and color) and the original image is placed on top. .. code-block:: yaml @@ -165,11 +208,14 @@ created (with the given size and color) and the original image is placed on top: filters: background: { size: [1026, 684], position: center, color: '#fff' } -The ``watermark`` filter -~~~~~~~~~~~~~~~~~~~~~~~~ -The watermark filter pastes a second image onto your image while keeping its -ratio. Configuration looks like this: +Watermark +~~~~~~~~~ + +.. _filter-watermark: + +The ``watermark`` filter pastes a second image onto your image while keeping its +ratio. .. code-block:: yaml @@ -191,13 +237,16 @@ ratio. Configuration looks like this: filters like ``crop`` after it is possible that watermark image will be cropped. -The ``auto_rotate`` filter -~~~~~~~~~~~~~~~~~~~~~~~~~~ -It rotates the image automatically to display it as correctly as possible. The +Auto Rotate +~~~~~~~~~~~ + +.. _filter-auto-rotate: + +The ``auto_rotate`` filter rotates the image automatically to display it as correctly as possible. The rotation to apply is obtained through the metadata stored in the EXIF data of the original image. This filter provides no configuration options, so you just -need to enable it as follows: +need to enable it as follows. .. code-block:: yaml @@ -207,15 +256,18 @@ need to enable it as follows: filters: auto_rotate: ~ -.. note:: +.. tip:: This filter should be called as early as possible to get better results. -The ``rotate`` filter -~~~~~~~~~~~~~~~~~~~~~ -It rotates the image based on specified angle (in degrees). The value of the -``angle`` configuration option must be a positive integer or float number: +Rotate +~~~~~~ + +.. _filter-rotate: + +The ``rotate`` filter rotates the image based on specified angle (in degrees). The value of the +``angle`` configuration option must be a positive integer or float number. .. code-block:: yaml @@ -225,10 +277,13 @@ It rotates the image based on specified angle (in degrees). The value of the filters: rotate: { angle: 90 } -The ``interlace`` filter -~~~~~~~~~~~~~~~~~~~~~~~~ -It modifies the way the image is loaded progressively: +Interlace +~~~~~~~~~ + +.. _filter-interlace: + +The ``interlace`` filter modifies the way the image is loaded progressively. .. code-block:: yaml @@ -240,10 +295,13 @@ It modifies the way the image is loaded progressively: # mode can be one of: 'none', 'line', 'plane' and 'partition' mode: line -The ``grayscale`` filter -~~~~~~~~~~~~~~~~~~~~~~~~ -It modifies the image colors by calculating the gray-value based on RGB: +Grayscale +~~~~~~~~~ + +.. _filter-grayscale: + +The ``grayscale`` filter modifies the image colors by calculating the gray-value based on RGB. .. code-block:: yaml @@ -253,54 +311,104 @@ It modifies the image colors by calculating the gray-value based on RGB: filters: grayscale: ~ -Load your Custom Filters ------------------------- -The ImagineBundle allows you to load your own custom filter classes. The only -requirement is that each filter loader implements the following interface: -``Liip\ImagineBundle\Imagine\Filter\Loader\LoaderInterface``. +Custom Filters +-------------- + +.. _filter-custom: + +You can easily define your own, custom filters to perform any image +transformation operations required. Creating a custom filter begins +by creating a class that implements the ``Liip\ImagineBundle\Imagine\Filter\Loader\LoaderInterface`` +interface, as shown below. + +.. code-block:: php + + interface LoaderInterface + { + public function load(ImageInterface $image, array $options = array()); + } + +As defined in ``LoaderInterface``, the only required method is one named ``load``, +which is provided an instance of ``ImageInterface`` and an array of options, and +subsequently provides an instance of ``ImageInterface`` in return. + +The following is a template for creating your own filter. You must provide +the implementation for the ``load`` method to create a valid filter. -To tell the bundle about your new filter loader, register it in the service -container and apply the ``liip_imagine.filter.loader`` tag to it (example here -in XML): +.. code-block:: php + + namespace AppBundle\Imagine\Filter\Loader; + + use Imagine\Image\ImageInterface; + use Liip\ImagineBundle\Imagine\Filter\Loader\LoaderInterface; + + class MyCustomFilter implements LoaderInterface + { + /** + * @param ImageInterface $image + * @param array $options + * + * @return ImageInterface + */ + public function load(ImageInterface $image, array $options = array()) + { + /** @todo: implement */ + + // return the image + return $image; + } + } + +Once you have defined your custom filter, you must define it as a service and tag it +with ``liip_imagine.filter.loader``. + +.. note:: + + For more information on the Service Container, reference the official + `Symfony Service Container documentation`_. + +To register ``AppBundle\Imagine\Filter\Loader\MyCustomFilter`` with the name +``my_custom_filter``, you would use the following configuration. .. configuration-block:: .. code-block:: yaml # app/config/services.yml - app.filter.my_custom_filter: - class: AppBundle\Imagine\Filter\Loader\MyCustomFilterLoader - tags: - - { name: 'liip_imagine.filter.loader', loader: 'my_custom_filter' } + + services: + app.filter.my_custom_filter: + class: AppBundle\Imagine\Filter\Loader\MyCustomFilter + tags: + - { name: "liip_imagine.filter.loader", loader: my_custom_filter } .. code-block:: xml - + + -For more information on the service container, see the `Symfony Service Container`_ -documentation. - -You can now reference and use your custom filter when defining filter sets you'd -like to apply in your configuration: +You can now reference and use your custom filter when defining filter sets in your configuration: .. code-block:: yaml + # app/config/config.yml + liip_imagine: filter_sets: my_special_style: filters: my_custom_filter: { } -For an example of a filter loader implementation, refer to -``Liip\ImagineBundle\Imagine\Filter\Loader\ThumbnailFilterLoader``. Dynamic filters --------------- +.. _filter-dynamic: + With a custom controller action it is possible to dynamically modify the configuration that will be applied to the image. Inside the controller you can access ``FilterManager`` instance, pass configuration as third parameter of @@ -330,198 +438,10 @@ A simple example showing how to change the filter configuration dynamically. return new RedirectResponse($this->cacheManager->resolve($path, $filter), Response::HTTP_MOVED_PERMANENTLY); } -.. note:: +.. tip:: The constant ``Response::HTTP_MOVED_PERMANENTLY`` was introduced in Symfony 2.4. Developers using older versions of Symfony, please replace the constant by ``301``. -Post-Processors ---------------- - -Filters allow modifying the image, but in order to modify the resulting binary -file created by filters, you can use post-processors. Post-processors must -implement ``Liip\ImagineBundle\Imagine\Filter\PostProcessor\PostProcessorInterface``. - -``PostProcessorInterface::process`` method receives ``BinaryInterface`` - -basically, the file containing an image after all filters have been applied. It -should return the ``BinaryInterface`` as well. - -Post-Processors, for this reason, may be safely chained. This is true even if they -operate on different mime-types, meaning that they are perfect for image-specific -optimisation techniques. A number of optimisers, lossy and loss-less, are provided -by default. - -To tell the bundle about your post-processor, register it in the service -container and apply the ``liip_imagine.filter.post_processor`` tag to it: - -.. configuration-block:: - - .. code-block:: yaml - - # app/config/services.yml - app.post_processor.my_custom_post_processor: - class: AppBundle\Imagine\Filter\PostProcessor\MyCustomPostProcessor - tags: - - { name: 'liip_imagine.filter.post_processor', post_processor: 'my_custom_post_processor' } - - .. code-block:: xml - - - - - - -For more information on the service container, see the `Symfony Service Container`_ -documentation. - -You can now reference and use your custom filter when defining filter sets you'd -like to apply in your configuration: - -.. code-block:: yaml - - liip_imagine: - filter_sets: - my_special_style: - post_processors: - my_custom_post_processor: { } - -For an example of a post processor implementation, refer to -``Liip\ImagineBundle\Imagine\Filter\PostProcessor\JpegOptimPostProcessor``. - -The ``JpegOptimPostProcessor`` can be used to provide lossless JPEG -optimization, which is good for you website loading speed. Parameters to configure -stripping of comment and exif data, max quality and progressive rendering may be -passed in optionally. In order to add lossless JPEG optimization to your filters, -use the following configuration: - -.. code-block:: yaml - - liip_imagine: - filter_sets: - my_thumb: - filters: - thumbnail: { size: [150, 150], mode: outbound } - post_processors: - jpegoptim: { strip_all: true, max: 70, progressive: true } - -Make sure that jpegoptim binary is installed on the system. If path to jpegoptim -binary is different from ``/usr/bin/jpegoptim``, adjust the path by overriding -parameters, for example: - -.. code-block:: yaml - - parameters: - liip_imagine.jpegoptim.binary: /usr/local/bin/jpegoptim - -.. _`Symfony Service Container`: http://symfony.com/doc/current/book/service_container.html - -It is also possible to configure other defaults for the conversion process via parameters, -for example: - -.. code-block:: yaml - - parameters: - # When true, this passes down --strip-all to jpegoptim, which strips all markers from the output jpeg. - liip_imagine.jpegoptim.stripAll: true - - # Sets the maxiumum image quality factor. - liip_imagine.jpegoptim.max: null - - # When true, --all-progressive is passed to jpegoptim, which results in the output being a progressive jpeg. - liip_imagine.jpegoptim.progressive: true - - # The directory where temporary file will be written. By default it's empty, and computed using `sys_get_temp_dir()` - # You can set it to `/run/shm` or something similar for writing temporary files in-memory, for decrease of disk load - liip_imagine.jpegoptim.tempDir: "" - -.. _`Symfony Service Container`: http://symfony.com/doc/current/book/service_container.html - - -The ``OptiPngPostProcessor`` is also available by default and can be used just as jpegoptim. -Make sure that optipng binary is installed on the system and change the -``liip_imagine.optipng.binary`` in parameters if needed. - -.. code-block:: yaml - - parameters: - liip_imagine.optipng.binary: /usr/local/bin/optipng - -.. _`Symfony Service Container`: http://symfony.com/doc/current/book/service_container.html - -It is also possible to configure other defaults for the conversion process via parameters, -for example: - -.. code-block:: yaml - - parameters: - # When true, this passes down --strip=all to optipng, which removes all metadata from the output image. - liip_imagine.optipng.stripAll: true - - # The optimisation level to be used by optipng. Defaults to 7. - liip_imagine.optipng.level: 7 - - # The directory where temporary file will be written. By default is empty, and computed using `sys_get_temp_dir()` - # You can set it to `/run/shm` or something similar for writing temporary files in-memory, for decrease of disk load - liip_imagine.optipng.tempDir: "" - -.. _`Symfony Service Container`: http://symfony.com/doc/current/book/service_container.html - - -The ``MozJpegPostProcessor`` can be used to provide safe lossy JPEG optimization. -Optionally, a quality parameter may be passed down to each instance. -More parameters may surface in the future. - -.. code-block:: yaml - - liip_imagine: - filter_sets: - my_thumb: - filters: - thumbnail: { size: [150, 150], mode: outbound } - post_processors: - mozjpeg: {} - my_other_thumb: - filters: - thumbnail: { size: [150, 150], mode: outbound } - post_processors: - mozjpeg: { quality: 90 } - -Make sure that you have installed the mozjpeg tools on your system, and please adjust the -``liip_imagine.mozjpeg.binary`` in parameters if needed. - -.. code-block:: yaml - - parameters: - liip_imagine.mozjpeg.binary: /opt/mozjpeg/bin/cjpeg - -.. _`Symfony Service Container`: http://symfony.com/doc/current/book/service_container.html - - -The ``PngquantPostProcessor`` can be used to provide safe lossy PNG optimization. -Optionally, a quality parameter may be passed down to each instance. -More parameters may surface in the future. - -.. code-block:: yaml - - liip_imagine: - filter_sets: - my_thumb: - filters: - thumbnail: { size: [150, 150], mode: outbound } - post_processors: - pngquant: {} - my_other_thumb: - filters: - thumbnail: { size: [150, 150], mode: outbound } - post_processors: - pngquant: { quality: "80-100" } - -Make sure that you have installed a recent version (at least 2.3) of pngquant on your system, and please adjust the -``liip_imagine.pngquant.binary`` in parameters if needed. - -.. code-block:: yaml - - parameters: - liip_imagine.pngquant.binary: /usr/bin/pngquant -.. _`Symfony Service Container`: http://symfony.com/doc/current/book/service_container.html +.. _`Symfony Service Container documentation`: http://symfony.com/doc/current/book/service_container.html diff --git a/Resources/doc/index.rst b/Resources/doc/index.rst index 4df5e0762..53696d3ae 100644 --- a/Resources/doc/index.rst +++ b/Resources/doc/index.rst @@ -1,9 +1,33 @@ + LiipImagineBundle ================= -This bundle allows to alter images in certain ways, such as scaling or rotating -them, creating thumbnails, adding watermarks, etc. In order to not hurt application -performance, altered images can be cached locally and in Amazon S3 service. +Overview +-------- + +The `LiipImagineBundle`_ package provides an *image manipulation abstraction toolkit* +for Symfony-based projects. Features include: + +* :doc:`Filter Sets `: Using any Symfony-supported configuration language + (such as YML and XML), you can create *filter set* definitions that specify + transformation routines. These include a set of *filters* and *post-processors*, as + well as other, optional parameters. +* :doc:`Filters `: A number of built-in filters are provided, allowing for an array of + common image transformations. Examples include :ref:`thumbnail `, + :ref:`scale `, :ref:`crop `, :ref:`strip `, + and :ref:`watermark `, and many more. Additionally, + :ref:`custom filters ` are supported. +* :doc:`Post-Processors `: A number of build-in post-processors are provided, + allowing for the modification of the resulting binary file created by filters. Examples include + :ref:`JpegOptim `, :ref:`OptiPNG `, + :ref:`MozJpeg `, and :ref:`PngQuant `. Additionally, + :ref:`custom post-processors ` are supported. + + +Chapters +-------- + +To learn more about this bundle, explore the available chapters: .. toctree:: :maxdepth: 1 @@ -12,8 +36,12 @@ performance, altered images can be cached locally and in Amazon S3 service. introduction basic-usage filters + post-processors configuration data-loaders cache-resolvers cache-manager commands + + +.. _`LiipImagineBundle`: https://github.com/liip/LiipImagineBundle diff --git a/Resources/doc/installation.rst b/Resources/doc/installation.rst index 8bb366dc2..e61bd0252 100644 --- a/Resources/doc/installation.rst +++ b/Resources/doc/installation.rst @@ -1,28 +1,33 @@ + Installation ============ Step 1: Download the Bundle --------------------------- -Open a command console, enter your project directory and execute the -following command to download the latest stable version of this bundle: +Open a command console, enter your project directory, and execute the +following command to download the latest stable version of this bundle +and add it as a dependency to your project: .. code-block:: bash $ composer require liip/imagine-bundle -This command requires you to have Composer installed globally, as explained -in the `installation chapter`_ of the Composer documentation. +This command requires that `Composer`_ is installed globally, as explained in +the `installation documentation`_ for Composer. + Step 2: Enable the Bundle ------------------------- -Then, enable the bundle by adding the following line in the ``app/AppKernel.php`` -file of your project: +Then, enable the bundle by adding ``new Liip\ImagineBundle\LiipImagineBundle()`` +to the bundles array of the ``registerBundles`` method in your project's +``app/AppKernel.php`` file: .. code-block:: php - # app/config/routing.yml - _liip_imagine: - resource: "@LiipImagineBundle/Resources/config/routing.xml" +Congratulations; you are ready to rock your images! -Congratulations! You're ready to rock your images! -.. _`installation chapter`: https://getcomposer.org/doc/00-intro.md +.. _`installation documentation`: https://getcomposer.org/doc/00-intro.md +.. _`Composer`: https://getcomposer.org/ diff --git a/Resources/doc/introduction.rst b/Resources/doc/introduction.rst index d4e9c4d9a..bf6013364 100644 --- a/Resources/doc/introduction.rst +++ b/Resources/doc/introduction.rst @@ -1,5 +1,6 @@ -LiipImagineBundle -================= + +Introduction +============ Basic Data Flow --------------- @@ -8,6 +9,7 @@ The core feature of this bundle is to provide a way to alter images in certain ways and cache the altered versions. There are several components involved to get this done. + Retrieving the original image ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ @@ -31,6 +33,7 @@ The most important parts about those ``DataLoader``: Check out the :doc:`chapter about data loaders ` to learn more about them. + Apply filters on the original image ~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~ @@ -51,6 +54,7 @@ customize the outcome. Check out the :doc:`chapter about filters ` to learn more about them. + Cache the filtered image ~~~~~~~~~~~~~~~~~~~~~~~~ @@ -67,8 +71,7 @@ image in the web directory as a static file, so the web server won't call the application stack anymore on those images. The images will be created upon first request and will remain in their static cached version until removed. -A ``CacheResolver`` implements the -``Liip\ImagineBundle\Imagine\Cache\Resolver\ResolverInterface``. +A ``CacheResolver`` implements the ``Liip\ImagineBundle\Imagine\Cache\Resolver\ResolverInterface``. It handles the so-called ``path``, which is the identifier you use, when addressing the original image, e.g. in your template. This path relates to the diff --git a/Resources/doc/post-processors.rst b/Resources/doc/post-processors.rst new file mode 100644 index 000000000..d70dcb9b8 --- /dev/null +++ b/Resources/doc/post-processors.rst @@ -0,0 +1,434 @@ + +Post-Processors +=============== + +We already know that :doc:`filters ` perform image transformation. This may +leave you wondering how post-processors fit into the runtime. To help illustrate the +difference between filters and post-processors, it is important to highlight the following. + +* Filters modify the **image**. +* Post-processors modify the **image binary**. + +After all filters have run, the result is an image binary. This is then provided to, +processed by, and returned from all configured post-processors. + +.. tip:: + + Post-Processors can be safely chained, even if they operate on different mime-types. + This makes them perfect for image-specific optimisation techniques. + + +Built-in Post-Processors +------------------------ + +A number of built-in post-processors are provided by default. The following are of the +"image optimizer" classification. They are intended to reduce image file size, and +therefor improve the load performance of your application's assets. + +* `JpegOptim`_ +* `OptiPng`_ +* `MozJpeg`_ +* `PngQuant`_ + + +JpegOptim +~~~~~~~~~ + +.. _post-processor-jpegoptim: + +The ``JpegOptimPostProcessor`` is a built-in post-processor that performs a number of +*lossless* optimizations on *JPEG* encoded images. + +To add this post-processor to the filter set created in the +:ref:`thumbnail usage example ` use: + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + filter_sets: + my_thumb: + filters: + thumbnail: { size: [120, 90], mode: outbound } + background: { size: [124, 94], position: center, color: '#000' } + post_processors: + jpegoptim: { strip_all: true, max: 70, progressive: true } + +This configuration enables metadata stripping and progressive JPEG encoding, and sets +a maximum quality factor of 70 for the resulting image binary. + +.. note:: + + The default executable path is ``/usr/bin/jpegoptim``. If installed elsewhere + on your system, you must set the ``liip_imagine.jpegoptim.binary`` parameter accordingly. + + .. code-block:: yaml + + # app/config/config.yml + + parameters: + liip_imagine.jpegoptim.binary : /your/custom/path/to/jpegoptim + + +Options +******* + +This post-processor offers a number of configuration options: + +* ``strip_all``: Removes all comments, EXIF markers, and other image metadata. +* ``max``: Sets the maximum image quality factor. +* ``progressive``: Ensures the image uses progressive encoding. + + +Parameters +********** + +Overwriting any of these parameters will change the post-processor's default behavior: + +* ``liip_imagine.jpegoptim.stripAll`` (default ``true``): + Removes all comments, EXIF markers, and other metadata from the image binary. + +* ``liip_imagine.jpegoptim.max`` (default ``null``): + Assigns the maximum quality factor for the image binary. + +* ``liip_imagine.jpegoptim.progressive`` (default ``true``): + Ensures that progressive encoding is enabled for the image binary. + +* ``liip_imagine.jpegoptim.binary`` (default ``string:/usr/bin/jpegoptim``): + Sets the location of the ``jpegoptim`` executable. + +* ``liip_imagine.jpegoptim.tempDir`` (default ````): + Sets the directory to store temporary files. + +.. tip:: + + The value of ``liip_imagine.jpegoptim.tempDir`` can be set to an in-memory mount point + on supported operating systems, such as ``/run/shm`` on Linux. This will decrease disk + load and may increase performance. + +OptiPng +~~~~~~~ + +.. _post-processor-optipng: + +The ``OptiPngPostProcessor`` is a built-in post-processor that performs a number of +*lossless* optimizations on *PNG* encoded images. + +To add this post-processor to the filter set created in the +:ref:`thumbnail usage example ` use: + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + filter_sets: + my_thumb: + filters: + thumbnail: { size: [120, 90], mode: outbound } + background: { size: [124, 94], position: center, color: '#000' } + post_processors: + optipng: { strip_all: true, level: 5 } + +This configuration enables metadata stripping, and sets a maximum optimization factor of 5 +for the resulting image binary. + +.. note:: + + The default executable path is ``/usr/bin/optipng``. If installed elsewhere + on your system, you must set the ``liip_imagine.optipng.binary`` parameter accordingly. + + .. code-block:: yaml + + # app/config/config.yml + + parameters: + liip_imagine.optipng.binary : /your/custom/path/to/optipng + + +Options +******* + +This post-processor offers a number of configuration options: + +* ``strip_all``: Removes all comments, EXIF markers, and other image metadata. +* ``level``: Sets the image optimization factor. + + +Parameters +********** + +Overwriting any of these parameters will change the post-processor's default behavior: + +* ``liip_imagine.optipng.stripAll`` (default ``true``): + Removes all comments, EXIF markers, and other metadata from the image binary. + +* ``liip_imagine.optipng.level`` (default ``7``): + Assigns the maximum optimization factor for the image binary. + +* ``liip_imagine.optipng.binary`` (default ``string:/usr/bin/optipng``): + Sets the location of the ``optipng`` executable. + +* ``liip_imagine.optipng.tempDir`` (default ````): + Sets the directory to store temporary files. + +.. tip:: + + The value of ``liip_imagine.optipng.tempDir`` can be set to an in-memory mount point + on supported operating systems, such as ``/run/shm`` on Linux. This will decrease disk + load and may increase performance. + + +MozJpeg +~~~~~~~ + +.. _post-processor-mozjpeg: + +The ``MozJpegPostProcessor`` is a built-in post-processor that performs a number of +*safe, lossy* optimizations on *JPEG* encoded images. + +To add this post-processor to the filter set created in the +:ref:`thumbnail usage example ` use: + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + filter_sets: + my_thumb: + filters: + thumbnail: { size: [120, 90], mode: outbound } + background: { size: [124, 94], position: center, color: '#000' } + post_processors: + mozjpeg: { quality: 80 } + +This configuration sets a maximum quality factor of 70 for the resulting image binary. + +.. note:: + + The default executable path is ``/opt/mozjpeg/bin/cjpeg``. If installed elsewhere + on your system, you must set the ``liip_imagine.mozjpeg.binary`` parameter accordingly. + + .. code-block:: yaml + + # app/config/config.yml + + parameters: + liip_imagine.mozjpeg.binary : /your/custom/path/to/cjpeg + + +Options +******* + +This post-processor offers the following configuration option: + +* ``quality``: Sets the image quality factor. + + +Parameters +********** + +Overwriting any of these parameters will change the post-processor's default behavior: + +* ``liip_imagine.mozjpeg.binary`` (default ``/opt/mozjpeg/bin/cjpeg``): + Sets the location of the ``cjpeg`` executable. + + +PngQuant +~~~~~~~~ + +.. _post-processor-pngquant: + +The ``PngquantPostProcessor`` is a built-in post-processor that performs a number of +*safe, lossy* optimizations on *PNG* encoded images. + +To add this post-processor to the filter set created in the +:ref:`thumbnail usage example ` use: + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine: + filter_sets: + my_thumb: + filters: + thumbnail: { size: [120, 90], mode: outbound } + background: { size: [124, 94], position: center, color: '#000' } + post_processors: + pngquant: { quality: "75-85" } + +This configuration sets a quality factor range of 75 to 80 for the resulting image binary. + +.. note:: + + The default executable path is ``/usr/bin/pngquant``. If installed elsewhere + on your system, you must set the ``liip_imagine.pngquant.binary`` parameter accordingly. + + .. code-block:: yaml + + # app/config/config.yml + + parameters: + liip_imagine.pngquant.binary : /your/custom/path/to/pngquant + + +Options +******* + +This post-processor offers the following configuration option: + +* ``quality``: Sets the image quality factor range. + + +Parameters +********** + +Overwriting any of these parameters will change the post-processor's default behavior: + +* ``liip_imagine.pngquant.binary`` (default ``/usr/bin/pngquant``): + Sets the location of the ``pnquant`` executable. + + +Custom Post-Processors +---------------------- + +.. _post-processors-custom: + +Just like filters, you can easily define your own, custom post-processors to +perform any image binary operations required. Creating a custom post-processor +begins by creating a class that implements the ``PostProcessorInterface`` +interface, as shown below. + +.. code-block:: php + + interface PostProcessorInterface + { + public function process(BinaryInterface $binary); + } + +As defined in ``PostProcessorInterface``, the only required method is one named ``process``, +which is provided an instance of ``BinaryInterface`` as its singular parameter, and +subsequently provides an instance of ``BinaryInterface`` in return. + +.. tip:: + + You may optionally implement ``ConfigurablePostProcessorInterface`` in your + post-processor to allow it to be configurable. + +The following is a template for creating your own post-processor that calls an executable. +You must set the ``EXECUTABLE_PATH`` class constant to the absolute path of the desired +executable. You may also want to change ``array('image/png')`` to the supported mime types +for your custom post-processor. + +.. code-block:: php + + namespace AppBundle\Imagine\Filter\PostProcessor; + + use Liip\ImagineBundle\Binary\BinaryInterface; + use Liip\ImagineBundle\Model\Binary; + use Liip\ImagineBundle\Imagine\Filter\PostProcessor\PostProcessorInterface; + use Symfony\Component\Process\Exception\ProcessFailedException; + use Symfony\Component\Process\ProcessBuilder; + + class MyCustomPostProcessor implements PostProcessorInterface + { + const EXECUTABLE_PATH = '/path/to/your/executable'; + + /** + * @param BinaryInterface $binary + * + * @return BinaryInterface + */ + public function process(BinaryInterface $binary) + { + // ensure the passed binary is a png + if (!in_array(strtolower($binary->getMimeType()), array('image/png'))) { + return $binary; + } + + // create a temporary input file + if (false === $input = tempnam($path = sys_get_temp_dir(), 'custom_')) { + throw new \Exception(sprintf('Error created tmp file in "%s".', $path)); + } + + // populate temporary file with passed file contents + file_put_contents($input, $binary->getContent()); + + // create a process builder, add the input file as argument + $pb = new ProcessBuilder(array(self::EXECUTABLE_PATH)); + $pb->add($input); + + // get a process instance and run it + $process = $pb->getProcess(); + $process->run(); + + // error out if command returned non-zero + if (0 !== $process->getExitCode()) { + unlink($input); + throw new ProcessFailedException($process); + } + + // retrieve the result + $result = new Binary( + file_get_contents($input), + $binary->getMimeType(), + $binary->getFormat() + ); + + // remove temporary file + unlink($input); + + // return the result + return $result; + } + } + +Once you have defined your custom post-processor, you must define it as a service and tag it +with ``liip_imagine.filter.post_processor``. + +.. note:: + + For more information on the Service Container, reference the official + `Symfony Service Container documentation`_. + +To register ``AppBundle\Imagine\Filter\PostProcessor\MyCustomPostProcessor`` with the name +``my_custom_post_processor``, you would use the following configuration. + +.. configuration-block:: + + .. code-block:: yaml + + # app/config/services.yml + + app.post_processor.my_custom_post_processor : + class : AppBundle\Imagine\Filter\PostProcessor\MyCustomPostProcessor + tags : + - { name : 'liip_imagine.filter.post_processor', post_processor : 'my_custom_post_processor' } + + .. code-block:: xml + + + + + + + +Now your custom post-processor can be referenced in a filter set using the name +assigned via the ``post_processor`` tag attribute above (in this example, +``my_custom_post_processor``). + +.. code-block:: yaml + + # app/config/config.yml + + liip_imagine : + filter_sets : + my_special_style : + post_processors : + my_custom_post_processor : { } + + +.. _`Symfony Service Container documentation`: http://symfony.com/doc/current/book/service_container.html diff --git a/Resources/meta/LICENSE b/Resources/meta/LICENSE deleted file mode 100644 index a051eb0e2..000000000 --- a/Resources/meta/LICENSE +++ /dev/null @@ -1,19 +0,0 @@ -Copyright (c) 2010-2011 Liip - -Permission is hereby granted, free of charge, to any person obtaining a copy -of this software and associated documentation files (the "Software"), to deal -in the Software without restriction, including without limitation the rights -to use, copy, modify, merge, publish, distribute, sublicense, and/or sell -copies of the Software, and to permit persons to whom the Software is furnished -to do so, subject to the following conditions: - -The above copyright notice and this permission notice shall be included in all -copies or substantial portions of the Software. - -THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR -IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, -FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE -AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER -LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, -OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN -THE SOFTWARE. diff --git a/Templating/Helper/ImagineHelper.php b/Templating/Helper/ImagineHelper.php index 3441965d5..acc930d6f 100644 --- a/Templating/Helper/ImagineHelper.php +++ b/Templating/Helper/ImagineHelper.php @@ -1,5 +1,14 @@