/* Copyright (c) 2001-2019, The HSQL Development Group
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without
 * modification, are permitted provided that the following conditions are met:
 *
 * Redistributions of source code must retain the above copyright notice, this
 * list of conditions and the following disclaimer.
 *
 * Redistributions in binary form must reproduce the above copyright notice,
 * this list of conditions and the following disclaimer in the documentation
 * and/or other materials provided with the distribution.
 *
 * Neither the name of the HSQL Development Group nor the names of its
 * contributors may be used to endorse or promote products derived from this
 * software without specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
 * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
 * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE
 * ARE DISCLAIMED. IN NO EVENT SHALL HSQL DEVELOPMENT GROUP, HSQLDB.ORG,
 * OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL,
 * EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO,
 * PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
 * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */


package org.hsqldb.lib;

import java.io.FileOutputStream;
import java.io.IOException;
import java.io.InputStream;
import java.io.OutputStream;
import java.util.zip.Deflater;
import java.util.zip.DeflaterOutputStream;
import java.util.zip.GZIPInputStream;
import java.util.zip.GZIPOutputStream;
import java.util.zip.Inflater;
import java.util.zip.InflaterInputStream;

import org.hsqldb.lib.java.JavaSystem;

Creates a direct, compressed or decompressed copy of a file.
Author:Blaine Simpson (blaine dot simpson at admc dot com)
Version:1.9.0
Since:1.9.0
/** * Creates a direct, compressed or decompressed copy of a file. * * @author Blaine Simpson (blaine dot simpson at admc dot com) * @version 1.9.0 * @since 1.9.0 */
public class FileArchiver { public static final int COMPRESSION_NONE = 0; public static final int COMPRESSION_ZIP = 1; public static final int COMPRESSION_GZIP = 2; private static final int COPY_BLOCK_SIZE = 1 << 16; /* public static void compressFile(String infilename, String outfilename, FileAccess storage) throws IOException { FileArchiver.archive(infilename, outfilename, storage, COMPRESSION_ZIP); } public static void decompressFile(String infilename, String outfilename, FileAccess storage) throws IOException { FileArchiver.unarchive( infilename, outfilename, storage, COMPRESSION_ZIP); } public static void restoreFile(String infilename, String outfilename, FileAccess storage) throws IOException { FileArchiver.unarchive( infilename, outfilename, storage, COMPRESSION_NONE); } */ public static void copyFile(String infilename, String outfilename, FileAccess storage) throws IOException { FileArchiver.archive(infilename, outfilename, storage, COMPRESSION_NONE); } public static void archive(String infilename, String outfilename, FileAccess storage, int compressionType) throws IOException { InputStream in = null; OutputStream f = null; OutputStream fOut = null; DeflaterOutputStream deflater = null; boolean completed = false; // if there is no file if (!storage.isStreamElement(infilename)) { return; } try { byte[] b = new byte[COPY_BLOCK_SIZE]; in = storage.openInputStreamElement(infilename); f = storage.openOutputStreamElement(outfilename, true); fOut = f; switch (compressionType) { case COMPRESSION_ZIP : f = deflater = new DeflaterOutputStream(f, new Deflater(Deflater.BEST_SPEED), b.length); break; case COMPRESSION_GZIP : f = deflater = new GZIPOutputStream(f, b.length); break; case COMPRESSION_NONE : break; default : throw new RuntimeException("FileArchiver" + compressionType); } while (true) { int l = in.read(b, 0, b.length); if (l == -1) { break; } f.write(b, 0, l); } completed = true; } catch (Throwable e) { throw JavaSystem.toIOException(e); } finally { try { if (in != null) { in.close(); } if (f != null) { if (deflater != null) { deflater.finish(); } if (fOut instanceof FileOutputStream) { storage.getFileSync(fOut).sync(); } f.close(); } if (!completed && storage.isStreamElement(outfilename)) { storage.removeElement(outfilename); } } catch (Throwable e) { throw JavaSystem.toIOException(e); } } } public static void unarchive(String infilename, String outfilename, FileAccess storage, int compressionType) throws IOException { InputStream f = null; OutputStream outstream = null; boolean completed = false; try { if (!storage.isStreamElement(infilename)) { return; } storage.removeElement(outfilename); byte[] b = new byte[COPY_BLOCK_SIZE]; f = storage.openInputStreamElement(infilename); switch (compressionType) { case COMPRESSION_ZIP : f = new InflaterInputStream(f, new Inflater()); break; case COMPRESSION_GZIP : f = new GZIPInputStream(f, b.length); break; case COMPRESSION_NONE : break; default : throw new RuntimeException("FileArchiver: " + compressionType); } outstream = storage.openOutputStreamElement(outfilename, false); while (true) { int l = f.read(b, 0, b.length); if (l == -1) { break; } outstream.write(b, 0, l); } completed = true; } catch (Throwable e) { throw JavaSystem.toIOException(e); } finally { try { if (f != null) { f.close(); } if (outstream != null) { outstream.flush(); if (outstream instanceof FileOutputStream) { storage.getFileSync(outstream).sync(); } outstream.close(); } if (!completed && storage.isStreamElement(outfilename)) { storage.removeElement(outfilename); } } catch (Throwable e) { throw JavaSystem.toIOException(e); } } } }