blob: 8948a1cbcc6a250d3441d8452764a397437f67a9 [file] [log] [blame]
/*
* Copyright (C) 2016 The Android Open Source Project
*
* 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 com.android.apkzlib.zip.compress;
import com.android.apkzlib.zip.CompressionResult;
import com.android.apkzlib.zip.utils.ByteTracker;
import com.android.apkzlib.zip.utils.CloseableByteSource;
import com.google.common.base.Preconditions;
import java.util.concurrent.Executor;
import java.util.zip.Deflater;
import javax.annotation.Nonnull;
/**
* Compressor that tries both the best and default compression algorithms and picks the default
* unless the best is at least a given percentage smaller.
*/
public class BestAndDefaultDeflateExecutorCompressor extends ExecutorCompressor {
/**
* Deflater using the default compression level.
*/
@Nonnull
private final DeflateExecutionCompressor defaultDeflater;
/**
* Deflater using the best compression level.
*/
@Nonnull
private final DeflateExecutionCompressor bestDeflater;
/**
* Minimum best compression size / default compression size ratio needed to pick the default
* compression size.
*/
private final double minRatio;
/**
* Creates a new compressor.
*
* @param executor the executor used to perform compression activities.
* @param tracker the byte tracker to keep track of allocated bytes
* @param minRatio the minimum best compression size / default compression size needed to pick
* the default compression size; if {@code 0.0} then the default compression is always picked,
* if {@code 1.0} then the best compression is always picked unless it produces the exact same
* size as the default compression.
*/
public BestAndDefaultDeflateExecutorCompressor(@Nonnull Executor executor,
@Nonnull ByteTracker tracker, double minRatio) {
super(executor);
Preconditions.checkArgument(minRatio >= 0.0, "minRatio < 0.0");
Preconditions.checkArgument(minRatio <= 1.0, "minRatio > 1.0");
defaultDeflater =
new DeflateExecutionCompressor(executor, tracker, Deflater.DEFAULT_COMPRESSION);
bestDeflater =
new DeflateExecutionCompressor(executor, tracker, Deflater.BEST_COMPRESSION);
this.minRatio = minRatio;
}
@Nonnull
@Override
protected CompressionResult immediateCompress(@Nonnull CloseableByteSource source)
throws Exception {
CompressionResult defaultResult = defaultDeflater.immediateCompress(source);
CompressionResult bestResult = bestDeflater.immediateCompress(source);
double sizeRatio = bestResult.getSize() / (double) defaultResult.getSize();
if (sizeRatio >= minRatio) {
return defaultResult;
} else {
return bestResult;
}
}
}