All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.tapestry5.internal.webresources.AbstractMinimizer Maven / Gradle / Ivy

// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.

package org.apache.tapestry5.internal.webresources;

import org.apache.tapestry5.internal.TapestryInternalUtils;
import org.apache.tapestry5.internal.services.assets.BytestreamCache;
import org.apache.tapestry5.internal.services.assets.StreamableResourceImpl;
import org.apache.tapestry5.ioc.IOOperation;
import org.apache.tapestry5.ioc.OperationTracker;
import org.apache.tapestry5.services.assets.AssetChecksumGenerator;
import org.apache.tapestry5.services.assets.CompressionStatus;
import org.apache.tapestry5.services.assets.ResourceMinimizer;
import org.apache.tapestry5.services.assets.StreamableResource;
import org.slf4j.Logger;

import java.io.ByteArrayOutputStream;
import java.io.IOException;
import java.io.InputStream;

/**
 * Base class for resource minimizers.
 *
 * @since 5.3
 */
public abstract class AbstractMinimizer implements ResourceMinimizer
{
    private static final double NANOS_TO_MILLIS = 1.0d / 1000000.0d;

    protected final Logger logger;

    protected final OperationTracker tracker;

    private final AssetChecksumGenerator checksumGenerator;

    private final String resourceType;

    public AbstractMinimizer(Logger logger, OperationTracker tracker, AssetChecksumGenerator checksumGenerator, String resourceType)
    {
        this.logger = logger;
        this.tracker = tracker;
        this.resourceType = resourceType;
        this.checksumGenerator = checksumGenerator;
    }

    @Override
    public StreamableResource minimize(final StreamableResource input) throws IOException
    {
        if (!isEnabled(input))
        {
            return input;
        }

        long startNanos = System.nanoTime();

        final ByteArrayOutputStream bos = new ByteArrayOutputStream(1000);

        tracker.perform("Minimizing " + input, new IOOperation()
        {
            @Override
            public Void perform() throws IOException
            {
                InputStream in = doMinimize(input);

                TapestryInternalUtils.copy(in, bos);

                in.close();

                return null;
            }
        });

        // The content is minimized, but can still be (GZip) compressed.

        StreamableResource output = new StreamableResourceImpl("minimized " + input.getDescription(),
                input.getContentType(), CompressionStatus.COMPRESSABLE,
                input.getLastModified(), new BytestreamCache(bos), checksumGenerator, input.getResponseCustomizer());

        if (logger.isInfoEnabled())
        {
            long elapsedNanos = System.nanoTime() - startNanos;

            int inputSize = input.getSize();
            int outputSize = output.getSize();

            double elapsedMillis = ((double) elapsedNanos) * NANOS_TO_MILLIS;
            // e.g., reducing 100 bytes to 25 would be a (100-25)/100 reduction, or 75%
            double reduction = 100d * ((double) (inputSize - outputSize)) / ((double) inputSize);

            logger.info(String.format("Minimized %s (%,d input bytes of %s to %,d output bytes in %.2f ms, %.2f%% reduction)",
                    input.getDescription(), inputSize, resourceType, outputSize, elapsedMillis, reduction));
        }

        return output;
    }

    /**
     * Implemented in subclasses to do the actual work.
     *
     * @param resource
     *         content to minimize
     * @return stream of minimized content
     */
    protected abstract InputStream doMinimize(StreamableResource resource) throws IOException;

    /**
     * Determines if the resource can be minimized.
     *
     * @return true, subclasses may override
     */
    protected boolean isEnabled(StreamableResource resource)
    {
        return true;
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy