001// Licensed under the Apache License, Version 2.0 (the "License");
002// you may not use this file except in compliance with the License.
003// You may obtain a copy of the License at
004//
005// http://www.apache.org/licenses/LICENSE-2.0
006//
007// Unless required by applicable law or agreed to in writing, software
008// distributed under the License is distributed on an "AS IS" BASIS,
009// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
010// See the License for the specific language governing permissions and
011// limitations under the License.
012
013package org.apache.tapestry5.modules;
014
015import java.util.List;
016import java.util.Map;
017
018import org.apache.tapestry5.SymbolConstants;
019import org.apache.tapestry5.internal.AssetConstants;
020import org.apache.tapestry5.internal.InternalConstants;
021import org.apache.tapestry5.internal.services.AssetSourceImpl;
022import org.apache.tapestry5.internal.services.ClasspathAssetAliasManagerImpl;
023import org.apache.tapestry5.internal.services.ClasspathAssetFactory;
024import org.apache.tapestry5.internal.services.ContextAssetFactory;
025import org.apache.tapestry5.internal.services.ExternalUrlAssetFactory;
026import org.apache.tapestry5.internal.services.IdentityAssetPathConverter;
027import org.apache.tapestry5.internal.services.RequestConstants;
028import org.apache.tapestry5.internal.services.ResourceStreamer;
029import org.apache.tapestry5.internal.services.assets.AssetChecksumGeneratorImpl;
030import org.apache.tapestry5.internal.services.assets.AssetPathConstructorImpl;
031import org.apache.tapestry5.internal.services.assets.CSSURLRewriter;
032import org.apache.tapestry5.internal.services.assets.ClasspathAssetRequestHandler;
033import org.apache.tapestry5.internal.services.assets.CompressionAnalyzerImpl;
034import org.apache.tapestry5.internal.services.assets.ContentTypeAnalyzerImpl;
035import org.apache.tapestry5.internal.services.assets.ContextAssetRequestHandler;
036import org.apache.tapestry5.internal.services.assets.JavaScriptStackAssembler;
037import org.apache.tapestry5.internal.services.assets.JavaScriptStackAssemblerImpl;
038import org.apache.tapestry5.internal.services.assets.JavaScriptStackMinimizeDisabler;
039import org.apache.tapestry5.internal.services.assets.MasterResourceMinimizer;
040import org.apache.tapestry5.internal.services.assets.ResourceChangeTracker;
041import org.apache.tapestry5.internal.services.assets.ResourceChangeTrackerImpl;
042import org.apache.tapestry5.internal.services.assets.SRSCachingInterceptor;
043import org.apache.tapestry5.internal.services.assets.SRSCompressedCachingInterceptor;
044import org.apache.tapestry5.internal.services.assets.SRSCompressingInterceptor;
045import org.apache.tapestry5.internal.services.assets.SRSMinimizingInterceptor;
046import org.apache.tapestry5.internal.services.assets.StackAssetRequestHandler;
047import org.apache.tapestry5.internal.services.assets.StreamableResourceSourceImpl;
048import org.apache.tapestry5.internal.services.assets.UTF8ForTextAssets;
049import org.apache.tapestry5.internal.services.messages.ClientLocalizationMessageResource;
050import org.apache.tapestry5.ioc.MappedConfiguration;
051import org.apache.tapestry5.ioc.OperationTracker;
052import org.apache.tapestry5.ioc.OrderedConfiguration;
053import org.apache.tapestry5.ioc.Resource;
054import org.apache.tapestry5.ioc.ServiceBinder;
055import org.apache.tapestry5.ioc.annotations.Autobuild;
056import org.apache.tapestry5.ioc.annotations.Contribute;
057import org.apache.tapestry5.ioc.annotations.Decorate;
058import org.apache.tapestry5.ioc.annotations.Marker;
059import org.apache.tapestry5.ioc.annotations.Order;
060import org.apache.tapestry5.ioc.annotations.Primary;
061import org.apache.tapestry5.ioc.annotations.Symbol;
062import org.apache.tapestry5.ioc.services.ChainBuilder;
063import org.apache.tapestry5.ioc.services.FactoryDefaults;
064import org.apache.tapestry5.ioc.services.SymbolProvider;
065import org.apache.tapestry5.services.ApplicationGlobals;
066import org.apache.tapestry5.services.AssetFactory;
067import org.apache.tapestry5.services.AssetPathConverter;
068import org.apache.tapestry5.services.AssetRequestDispatcher;
069import org.apache.tapestry5.services.AssetSource;
070import org.apache.tapestry5.services.ClasspathAssetAliasManager;
071import org.apache.tapestry5.services.ClasspathAssetProtectionRule;
072import org.apache.tapestry5.services.ClasspathProvider;
073import org.apache.tapestry5.services.ComponentClassResolver;
074import org.apache.tapestry5.services.ContextProvider;
075import org.apache.tapestry5.services.Core;
076import org.apache.tapestry5.services.Dispatcher;
077import org.apache.tapestry5.services.Request;
078import org.apache.tapestry5.services.ResponseCompressionAnalyzer;
079import org.apache.tapestry5.services.assets.AssetChecksumGenerator;
080import org.apache.tapestry5.services.assets.AssetPathConstructor;
081import org.apache.tapestry5.services.assets.AssetRequestHandler;
082import org.apache.tapestry5.services.assets.CompressionAnalyzer;
083import org.apache.tapestry5.services.assets.ContentTypeAnalyzer;
084import org.apache.tapestry5.services.assets.ResourceMinimizer;
085import org.apache.tapestry5.services.assets.StreamableResourceSource;
086import org.apache.tapestry5.services.javascript.JavaScriptStackSource;
087import org.apache.tapestry5.services.messages.ComponentMessagesSource;
088
089/**
090 * @since 5.3
091 */
092@Marker(Core.class)
093public class AssetsModule
094{
095    public static void bind(ServiceBinder binder)
096    {
097        binder.bind(AssetFactory.class, ClasspathAssetFactory.class).withSimpleId();
098        binder.bind(AssetPathConverter.class, IdentityAssetPathConverter.class);
099        binder.bind(AssetPathConstructor.class, AssetPathConstructorImpl.class);
100        binder.bind(ClasspathAssetAliasManager.class, ClasspathAssetAliasManagerImpl.class);
101        binder.bind(AssetSource.class, AssetSourceImpl.class);
102        binder.bind(StreamableResourceSource.class, StreamableResourceSourceImpl.class);
103        binder.bind(CompressionAnalyzer.class, CompressionAnalyzerImpl.class);
104        binder.bind(ContentTypeAnalyzer.class, ContentTypeAnalyzerImpl.class);
105        binder.bind(ResourceChangeTracker.class, ResourceChangeTrackerImpl.class);
106        binder.bind(ResourceMinimizer.class, MasterResourceMinimizer.class);
107        binder.bind(AssetChecksumGenerator.class, AssetChecksumGeneratorImpl.class);
108        binder.bind(JavaScriptStackAssembler.class, JavaScriptStackAssemblerImpl.class);
109    }
110
111    @Contribute(AssetSource.class)
112    public void configureStandardAssetFactories(MappedConfiguration<String, AssetFactory> configuration,
113                                                @ContextProvider
114                                                AssetFactory contextAssetFactory,
115
116                                                @ClasspathProvider
117                                                AssetFactory classpathAssetFactory)
118    {
119        configuration.add(AssetConstants.CONTEXT, contextAssetFactory);
120        configuration.add(AssetConstants.CLASSPATH, classpathAssetFactory);
121        configuration.add(AssetConstants.HTTP, new ExternalUrlAssetFactory(AssetConstants.HTTP));
122        configuration.add(AssetConstants.HTTPS, new ExternalUrlAssetFactory(AssetConstants.HTTPS));
123        configuration.add(AssetConstants.FTP, new ExternalUrlAssetFactory(AssetConstants.FTP));
124        configuration.add(AssetConstants.PROTOCOL_RELATIVE, new ExternalUrlAssetFactory(AssetConstants.PROTOCOL_RELATIVE));
125    }
126
127
128    @Contribute(SymbolProvider.class)
129    @FactoryDefaults
130    public static void setupSymbols(MappedConfiguration<String, Object> configuration)
131    {
132        // Minification may be enabled in production mode, but unless a minimizer is provided, nothing
133        // will change.
134        configuration.add(SymbolConstants.MINIFICATION_ENABLED, SymbolConstants.PRODUCTION_MODE_VALUE);
135        configuration.add(SymbolConstants.GZIP_COMPRESSION_ENABLED, true);
136        configuration.add(SymbolConstants.COMBINE_SCRIPTS, SymbolConstants.PRODUCTION_MODE_VALUE);
137        configuration.add(SymbolConstants.ASSET_URL_FULL_QUALIFIED, false);
138
139        configuration.add(SymbolConstants.ASSET_PATH_PREFIX, "assets");
140
141        configuration.add(SymbolConstants.BOOTSTRAP_ROOT, "${tapestry.asset.root}/bootstrap");
142        configuration.add(SymbolConstants.FONT_AWESOME_ROOT, "${tapestry.asset.root}/font_awesome");
143
144        configuration.add("tapestry.asset.root", "classpath:META-INF/assets/tapestry5");
145        configuration.add(SymbolConstants.OMIT_EXPIRATION_CACHE_CONTROL_HEADER, "max-age=60,must-revalidate");
146    }
147
148    // The use of decorators is to allow third-parties to get their own extensions
149    // into the pipeline.
150
151    @Decorate(id = "GZipCompression", serviceInterface = StreamableResourceSource.class)
152    public StreamableResourceSource enableCompression(StreamableResourceSource delegate,
153                                                      @Symbol(SymbolConstants.GZIP_COMPRESSION_ENABLED)
154                                                      boolean gzipEnabled, @Symbol(SymbolConstants.MIN_GZIP_SIZE)
155                                                      int compressionCutoff,
156                                                      AssetChecksumGenerator checksumGenerator)
157    {
158        return gzipEnabled
159                ? new SRSCompressingInterceptor(delegate, compressionCutoff, checksumGenerator)
160                : null;
161    }
162
163    @Decorate(id = "CacheCompressed", serviceInterface = StreamableResourceSource.class)
164    @Order("before:GZIpCompression")
165    public StreamableResourceSource enableCompressedCaching(StreamableResourceSource delegate,
166                                                            @Symbol(SymbolConstants.GZIP_COMPRESSION_ENABLED)
167                                                            boolean gzipEnabled, ResourceChangeTracker tracker)
168    {
169        return gzipEnabled
170                ? new SRSCompressedCachingInterceptor(delegate, tracker)
171                : null;
172    }
173
174    @Decorate(id = "Cache", serviceInterface = StreamableResourceSource.class)
175    @Order("after:GZipCompression")
176    public StreamableResourceSource enableUncompressedCaching(StreamableResourceSource delegate,
177                                                              ResourceChangeTracker tracker)
178    {
179        return new SRSCachingInterceptor(delegate, tracker);
180    }
181
182    // Goes after cache, to ensure that what we are caching is the minified version.
183    @Decorate(id = "Minification", serviceInterface = StreamableResourceSource.class)
184    @Order("after:Cache,TextUTF8")
185    public StreamableResourceSource enableMinification(StreamableResourceSource delegate, ResourceMinimizer minimizer,
186                                                       @Symbol(SymbolConstants.MINIFICATION_ENABLED)
187                                                       boolean enabled)
188    {
189        return enabled
190                ? new SRSMinimizingInterceptor(delegate, minimizer)
191                : null;
192    }
193
194    // Ordering this after minification means that the URL replacement happens first;
195    // then the minification, then the uncompressed caching, then compression, then compressed
196    // cache.
197    @Decorate(id = "CSSURLRewrite", serviceInterface = StreamableResourceSource.class)
198    @Order("after:Minification")
199    public StreamableResourceSource enableCSSURLRewriting(StreamableResourceSource delegate,
200                                                          OperationTracker tracker,
201                                                          AssetSource assetSource,
202                                                          AssetChecksumGenerator checksumGenerator,
203                                                          @Symbol(SymbolConstants.STRICT_CSS_URL_REWRITING) boolean strictCssUrlRewriting)
204    {
205        return new CSSURLRewriter(delegate, tracker, assetSource, checksumGenerator, strictCssUrlRewriting);
206    }
207
208    @Decorate(id = "DisableMinificationForStacks", serviceInterface = StreamableResourceSource.class)
209    @Order("before:Minification")
210    public StreamableResourceSource setupDisableMinificationByJavaScriptStack(StreamableResourceSource delegate,
211                                                                              @Symbol(SymbolConstants.MINIFICATION_ENABLED)
212                                                                              boolean enabled,
213                                                                              JavaScriptStackSource javaScriptStackSource,
214                                                                              Request request)
215    {
216        return enabled
217                ? new JavaScriptStackMinimizeDisabler(delegate, javaScriptStackSource, request)
218                : null;
219    }
220
221    /**
222     * Ensures that all "text/*" assets are given the UTF-8 charset.
223     *
224     * @since 5.4
225     */
226    @Decorate(id = "TextUTF8", serviceInterface = StreamableResourceSource.class)
227    @Order("after:Cache")
228    public StreamableResourceSource setupTextAssetsAsUTF8(StreamableResourceSource delegate)
229    {
230        return new UTF8ForTextAssets(delegate);
231    }
232
233    /**
234     * Adds content types:
235     * <dl>
236     * <dt>css</dt>
237     * <dd>text/css</dd>
238     * <dt>js</dt>
239     * <dd>text/javascript</dd>
240     * <dt>jpg, jpeg</dt>
241     * <dd>image/jpeg</dd>
242     * <dt>gif</dt>
243     * <dd>image/gif</dd>
244     * <dt>png</dt>
245     * <dd>image/png</dd>
246     * <dt>svg</dt>
247     * <dd>image/svg+xml</dd>
248     * <dt>swf</dt>
249     * <dd>application/x-shockwave-flash</dd>
250     * <dt>woff</dt>
251     * <dd>application/font-woff</dd>
252     * <dt>tff</dt> <dd>application/x-font-ttf</dd>
253     * <dt>eot</dt> <dd>application/vnd.ms-fontobject</dd>
254     * </dl>
255     */
256    @Contribute(ContentTypeAnalyzer.class)
257    public void setupDefaultContentTypeMappings(MappedConfiguration<String, String> configuration)
258    {
259        configuration.add("css", "text/css");
260        configuration.add("js", "text/javascript");
261        configuration.add("gif", "image/gif");
262        configuration.add("jpg", "image/jpeg");
263        configuration.add("jpeg", "image/jpeg");
264        configuration.add("png", "image/png");
265        configuration.add("swf", "application/x-shockwave-flash");
266        configuration.add("svg", "image/svg+xml");
267        configuration.add("woff", "application/font-woff");
268        configuration.add("ttf", "application/x-font-ttf");
269        configuration.add("eot", "application/vnd.ms-fontobject");
270    }
271
272    /**
273     * Disables compression for the following content types:
274     * <ul>
275     * <li>image/jpeg</li>
276     * <li>image/gif</li>
277     * <li>image/png</li>
278     * <li>image/svg+xml</li>
279     * <li>application/x-shockwave-flash</li>
280     * <li>application/font-woff</li>
281     * <li>application/x-font-ttf</li>
282     * <li>application/vnd.ms-fontobject</li>
283     * </ul>
284     */
285    @Contribute(CompressionAnalyzer.class)
286    public void disableCompressionForImageTypes(MappedConfiguration<String, Boolean> configuration)
287    {
288        configuration.add("image/*", false);
289        configuration.add("image/svg+xml", true);
290        configuration.add("application/x-shockwave-flash", false);
291        configuration.add("application/font-woff", false);
292        configuration.add("application/x-font-ttf", false);
293        configuration.add("application/vnd.ms-fontobject", false);
294    }
295
296    @Marker(ContextProvider.class)
297    public static AssetFactory buildContextAssetFactory(ApplicationGlobals globals,
298                                                        AssetPathConstructor assetPathConstructor,
299                                                        ResponseCompressionAnalyzer compressionAnalyzer,
300                                                        ResourceChangeTracker resourceChangeTracker,
301                                                        StreamableResourceSource streamableResourceSource)
302    {
303        return new ContextAssetFactory(compressionAnalyzer, resourceChangeTracker, streamableResourceSource, assetPathConstructor, globals.getContext());
304    }
305
306    @Contribute(ClasspathAssetAliasManager.class)
307    public static void addApplicationAndTapestryMappings(MappedConfiguration<String, String> configuration,
308
309                                                         @Symbol(InternalConstants.TAPESTRY_APP_PACKAGE_PARAM)
310                                                         String appPackage)
311    {
312        configuration.add("tapestry", "org/apache/tapestry5");
313
314        configuration.add("app", toPackagePath(appPackage));
315    }
316
317    /**
318     * Contributes an handler for each mapped classpath alias, as well handlers for context assets
319     * and stack assets (combined {@link org.apache.tapestry5.services.javascript.JavaScriptStack} files).
320     */
321    @Contribute(Dispatcher.class)
322    @AssetRequestDispatcher
323    public static void provideBuiltinAssetDispatchers(MappedConfiguration<String, AssetRequestHandler> configuration,
324
325                                                      @ContextProvider
326                                                      AssetFactory contextAssetFactory,
327
328                                                      @Autobuild
329                                                      StackAssetRequestHandler stackAssetRequestHandler,
330
331                                                      ClasspathAssetAliasManager classpathAssetAliasManager,
332                                                      ResourceStreamer streamer,
333                                                      AssetSource assetSource,
334                                                      ClasspathAssetProtectionRule classpathAssetProtectionRule)
335    {
336        Map<String, String> mappings = classpathAssetAliasManager.getMappings();
337
338        for (String folder : mappings.keySet())
339        {
340            String path = mappings.get(folder);
341
342            configuration.add(folder, new ClasspathAssetRequestHandler(streamer, assetSource, path, classpathAssetProtectionRule));
343        }
344
345        configuration.add(RequestConstants.CONTEXT_FOLDER,
346                new ContextAssetRequestHandler(streamer, contextAssetFactory.getRootResource()));
347
348        configuration.add(RequestConstants.STACK_FOLDER, stackAssetRequestHandler);
349
350    }
351
352    @Contribute(ClasspathAssetAliasManager.class)
353    public static void addMappingsForLibraryVirtualFolders(MappedConfiguration<String, String> configuration,
354                                                           ComponentClassResolver resolver)
355    {
356        // Each library gets a mapping or its folder automatically
357
358        Map<String, String> folderToPackageMapping = resolver.getFolderToPackageMapping();
359
360        for (String folder : folderToPackageMapping.keySet())
361        {
362            // This is the 5.3 version, which is still supported:
363            configuration.add(folder, toPackagePath(folderToPackageMapping.get(folder)));
364
365            // This is the 5.4 version; once 5.3 support is dropped, this can be simplified, and the
366            // "meta/" prefix stripped out.
367            String folderSuffix = folder.equals("") ? folder : "/" + folder;
368
369            configuration.add("meta" + folderSuffix, "META-INF/assets" + folderSuffix);
370        }
371    }
372
373    private static String toPackagePath(String packageName)
374    {
375        return packageName.replace('.', '/');
376    }
377
378    /**
379     * Contributes:
380     * <dl>
381     * <dt>ClientLocalization</dt>
382     * <dd>A virtual resource of formatting symbols for decimal numbers</dd>
383     * <dt>Core</dt>
384     * <dd>Built in messages used by Tapestry's default validators and components</dd>
385     * <dt>AppCatalog</dt>
386     * <dd>The Resource defined by {@link SymbolConstants#APPLICATION_CATALOG}</dd>
387     * <dt>
388     *     </dl>
389     *
390     * @since 5.2.0
391     */
392    @Contribute(ComponentMessagesSource.class)
393    public static void setupGlobalMessageCatalog(AssetSource assetSource,
394                                                 @Symbol(SymbolConstants.APPLICATION_CATALOG)
395                                                 Resource applicationCatalog, OrderedConfiguration<Resource> configuration)
396    {
397        configuration.add("ClientLocalization", new ClientLocalizationMessageResource());
398        configuration.add("Core", assetSource.resourceForPath("org/apache/tapestry5/core.properties"));
399        configuration.add("AppCatalog", applicationCatalog);
400    }
401
402    @Contribute(Dispatcher.class)
403    @Primary
404    public static void setupAssetDispatch(OrderedConfiguration<Dispatcher> configuration,
405                                          @AssetRequestDispatcher
406                                          Dispatcher assetDispatcher)
407    {
408
409        // This goes first because an asset to be streamed may have an file
410        // extension, such as
411        // ".html", that will confuse the later dispatchers.
412
413        configuration.add("Asset", assetDispatcher, "before:ComponentEvent");
414    }
415    
416    @Primary
417    public static ClasspathAssetProtectionRule buildClasspathAssetProtectionRule(
418            List<ClasspathAssetProtectionRule> rules, ChainBuilder chainBuilder)
419    {
420        return chainBuilder.build(ClasspathAssetProtectionRule.class, rules);
421    }
422    
423    public static void contributeClasspathAssetProtectionRule(
424            OrderedConfiguration<ClasspathAssetProtectionRule> configuration) 
425    {
426        ClasspathAssetProtectionRule classFileRule = (s) -> s.toLowerCase().endsWith(".class");
427        configuration.add("ClassFile", classFileRule);
428        ClasspathAssetProtectionRule propertiesFileRule = (s) -> s.toLowerCase().endsWith(".properties");
429        configuration.add("PropertiesFile", propertiesFileRule);
430        ClasspathAssetProtectionRule xmlFileRule = (s) -> s.toLowerCase().endsWith(".xml");
431        configuration.add("XMLFile", xmlFileRule);
432        ClasspathAssetProtectionRule folderRule = (s) -> isFolderToBlock(s);
433        configuration.add("Folder", folderRule);
434    }
435    
436    final private static boolean isFolderToBlock(String path) 
437    {
438        path = path.replace('\\', '/');
439        final int lastIndex = path.lastIndexOf('/');
440        if (lastIndex >= 0)
441        {
442            path = path.substring(lastIndex);
443        }
444        return !path.contains(".");
445    }
446    
447}