View Javadoc
1   /**
2    * This file Copyright (c) 2015-2016 Magnolia International
3    * Ltd.  (http://www.magnolia-cms.com). All rights reserved.
4    *
5    *
6    * This file is dual-licensed under both the Magnolia
7    * Network Agreement and the GNU General Public License.
8    * You may elect to use one or the other of these licenses.
9    *
10   * This file is distributed in the hope that it will be
11   * useful, but AS-IS and WITHOUT ANY WARRANTY; without even the
12   * implied warranty of MERCHANTABILITY or FITNESS FOR A
13   * PARTICULAR PURPOSE, TITLE, or NONINFRINGEMENT.
14   * Redistribution, except as permitted by whichever of the GPL
15   * or MNA you select, is prohibited.
16   *
17   * 1. For the GPL license (GPL), you can redistribute and/or
18   * modify this file under the terms of the GNU General
19   * Public License, Version 3, as published by the Free Software
20   * Foundation.  You should have received a copy of the GNU
21   * General Public License, Version 3 along with this program;
22   * if not, write to the Free Software Foundation, Inc., 51
23   * Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
24   *
25   * 2. For the Magnolia Network Agreement (MNA), this file
26   * and the accompanying materials are made available under the
27   * terms of the MNA which accompanies this distribution, and
28   * is available at http://www.magnolia-cms.com/mna.html
29   *
30   * Any modifications to this file must keep this entire header
31   * intact.
32   *
33   */
34  package info.magnolia.imaging;
35  
36  import info.magnolia.module.ModuleLifecycle;
37  import info.magnolia.module.ModuleLifecycleContext;
38  
39  import java.util.ArrayList;
40  import java.util.Arrays;
41  import java.util.Collection;
42  import java.util.Iterator;
43  import java.util.LinkedHashMap;
44  import java.util.List;
45  import java.util.Map;
46  import java.util.TreeSet;
47  
48  import javax.imageio.ImageIO;
49  import javax.imageio.spi.IIORegistry;
50  import javax.imageio.spi.ServiceRegistry;
51  
52  import org.apache.commons.collections4.CollectionUtils;
53  import org.apache.commons.collections4.Transformer;
54  import org.apache.commons.lang3.StringUtils;
55  import org.slf4j.Logger;
56  import org.slf4j.LoggerFactory;
57  
58  /**
59   * Keeps the configuration of the imaging module.
60   */
61  public class ImagingModule implements ModuleLifecycle {
62  
63      private static final Logger log = LoggerFactory.getLogger(ImagingModule.class);
64  
65      public static final String IMAGING = "imaging";
66  
67      private Map<String, ImageGenerator> generators = new LinkedHashMap<String, ImageGenerator>();
68      private boolean storeGeneratedImages = true;
69  
70      public void setGenerators(Map<String, ImageGenerator> generators) {
71          this.generators = generators;
72      }
73  
74      public Map<String, ImageGenerator> getGenerators() {
75          return generators;
76      }
77  
78      public boolean isStoreGeneratedImages() {
79          return storeGeneratedImages;
80      }
81  
82      public void setStoreGeneratedImages(boolean storeGeneratedImages) {
83          this.storeGeneratedImages = storeGeneratedImages;
84      }
85  
86      /**
87       * Scan for additional plugins when module starts.
88       *
89       * @see ImageIO#scanForPlugins()
90       */
91      @Override
92      public void start(ModuleLifecycleContext moduleLifecycleContext) {
93          ImageIO.scanForPlugins();
94  
95          // Outputting supported formats was previously done by the now extinct module info.magnolia:magnolia-module-imagingtools
96          if (log.isDebugEnabled()) {
97              log.debug("This lists the formats currently available to the javax.imageio package, as installed on this system.");
98              log.debug("Supported input formats:     {}", StringUtils.join(filter(ImageIO.getReaderFormatNames()), ", "));
99              log.debug("Supported input mime types:  {}", StringUtils.join(filter(ImageIO.getReaderMIMETypes()), ", "));
100             log.debug("Supported output formats:    {}", StringUtils.join(filter(ImageIO.getWriterFormatNames()), ", "));
101             log.debug("Supported output mime types: {}", StringUtils.join(filter(ImageIO.getWriterMIMETypes()), ", "));
102         }
103     }
104 
105     /**
106      * De-registers all plugins which have the {@link Thread#getContextClassLoader() current thread's context class loader}
107      * as its class loader when the module is stopped, to avoid class/resource leak.
108      *
109      * This is taken from com.twelvemonkeys.servlet.image.IIOProviderContextListener and the details are explained here:
110      * https://github.com/haraldk/TwelveMonkeys#deploying-the-plugins-in-a-web-app .
111      */
112     @Override
113     public void stop(ModuleLifecycleContext moduleLifecycleContext) {
114 
115         // De-register any locally registered IIO plugins. Relies on each web app having its own context class loader.
116         final IIORegistry registry = IIORegistry.getDefaultInstance();
117         final LocalFilter localFilter = new LocalFilter(Thread.currentThread().getContextClassLoader());
118 
119         Iterator<Class<?>> categories = registry.getCategories();
120 
121         while (categories.hasNext()) {
122             Class<?> category = categories.next();
123             Iterator<?> providers = registry.getServiceProviders(category, localFilter, false);
124 
125             // Copy the providers, as de-registering while iterating over providers will lead to ConcurrentModificationExceptions.
126             List<Object> providersCopy = new ArrayList<>();
127             while (providers.hasNext()) {
128                 providersCopy.add(providers.next());
129             }
130 
131             for (Object provider : providersCopy) {
132                 registry.deregisterServiceProvider(provider);
133                 log.debug("Unregistered locally installed provider class: {}", provider.getClass());
134             }
135         }
136     }
137 
138     /**
139      * Filters the providers based on their class loader.
140      */
141     static class LocalFilter implements ServiceRegistry.Filter {
142         private final ClassLoader loader;
143 
144         public LocalFilter(ClassLoader loader) {
145             this.loader = loader;
146         }
147 
148         @Override
149         public boolean filter(Object provider) {
150             return provider.getClass().getClassLoader() == loader;
151         }
152     }
153 
154     /**
155      * Removes duplicates and returns a collection of all entries in lowercase.
156      */
157     private Collection<String> filter(String... formats) {
158         final TreeSet<String> set = new TreeSet<>(String.CASE_INSENSITIVE_ORDER);
159         set.addAll(Arrays.asList(formats));
160         CollectionUtils.transform(set, new Transformer() {
161             @Override
162             public Object transform(Object input) {
163                 return ((String) input).toLowerCase();
164             }
165         });
166         return set;
167     }
168 }