org.hsqldb.lib.tar.DbBackup Maven / Gradle / Ivy
/* Copyright (c) 2001-2011, 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.tar;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.util.Properties;
import org.hsqldb.lib.InputStreamInterface;
import org.hsqldb.lib.InputStreamWrapper;
/**
* Works with tar archives containing HSQLDB database instance backups.
* Viz, creating, examining, or extracting these archives.
*
* This class provides OO Tar backup-creation control.
* The extraction and listing features are implemented only in static fashion
* in the Main method, which provides a consistent interface for all three
* features from the command-line.
*
* For tar creation, the default behavior is to fail if the target archive
* exists, and to abort if any database change is detected.
* Use the JavaBean setters to changes this behavior.
* See the main(String[]) method for details about command-line usage.
*
*
* @see
* The database backup section of the HyperSQL User Guide
* @see #main(String[])
* @see #setOverWrite(boolean)
* @see #setAbortUponModify(boolean)
* @author Blaine Simpson (blaine dot simpson at admc dot com)
* @author Fred Toussi (fredt@users dot sourceforge.net)
* @version 2.3.0
* @since 2.0.0
*/
public class DbBackup {
/**
* Command line invocation to create, examine, or extract HSQLDB database
* backup tar archives.
*
* This class stores tar entries as relative files without specifying
* parent directories, in what is commonly referred to as tar bomb
* format.
* The set of files is small, with known extensions, and the potential
* inconvenience of messing up the user's current directory is more than
* compensated by making it easier for the user to restore to a new
* database URL location at a peer level to the original.
*
* Automatically calculates buffer sizes based on the largest component
* file (for "save" mode) or tar file size (for other modes).
*
* Run
* java -cp path/to/hsqldb.jar org.hsqldb.lib.tar.DbBackup
*
for syntax help.
*
*/
public static void main(String[] sa)
throws IOException, TarMalformatException {
try {
if (sa.length < 1) {
System.out.println(
RB.DbBackup_syntax.getString(DbBackup.class.getName()));
System.out.println();
System.out.println(RB.listing_format.getString());
System.exit(0);
}
if (sa[0].equals("--save")) {
boolean overWrite = sa.length > 1
&& sa[1].equals("--overwrite");
if (sa.length != (overWrite ? 4
: 3)) {
throw new IllegalArgumentException();
}
DbBackup backup = new DbBackup(new File(sa[sa.length - 2]),
sa[sa.length - 1]);
backup.setOverWrite(overWrite);
backup.write();
} else if (sa[0].equals("--list")) {
if (sa.length < 2) {
throw new IllegalArgumentException();
}
String[] patternStrings = null;
if (sa.length > 2) {
patternStrings = new String[sa.length - 2];
for (int i = 2; i < sa.length; i++) {
patternStrings[i - 2] = sa[i];
}
}
new TarReader(new File(sa[1]), TarReader
.LIST_MODE, patternStrings, new Integer(DbBackup
.generateBufferBlockValue(new File(sa[1]))), null)
.read();
} else if (sa[0].equals("--extract")) {
boolean overWrite = sa.length > 1
&& sa[1].equals("--overwrite");
int firstPatInd = overWrite ? 4
: 3;
if (sa.length < firstPatInd) {
throw new IllegalArgumentException();
}
String[] patternStrings = null;
if (sa.length > firstPatInd) {
patternStrings = new String[sa.length - firstPatInd];
for (int i = firstPatInd; i < sa.length; i++) {
patternStrings[i - firstPatInd] = sa[i];
}
}
File tarFile = new File(sa[overWrite ? 2
: 1]);
int tarReaderMode = overWrite ? TarReader.OVERWRITE_MODE
: TarReader.EXTRACT_MODE;
new TarReader(
tarFile, tarReaderMode, patternStrings,
new Integer(DbBackup.generateBufferBlockValue(tarFile)),
new File(sa[firstPatInd - 1])).read();
} else {
throw new IllegalArgumentException();
}
} catch (IllegalArgumentException iae) {
System.out.println(
RB.DbBackup_syntaxerr.getString(DbBackup.class.getName()));
System.exit(2);
}
}
protected File dbDir;
protected File archiveFile;
protected String instanceName;
protected boolean overWrite = false; // Defaults no NO OVERWRITE
protected boolean abortUponModify = true; // Defaults to ABORT-UPON-MODIFY
File[] componentFiles;
InputStreamInterface[] componentStreams;
boolean[] existList;
boolean[] ignoreList;
/**
* Instantiate a DbBackup instance for creating a Database Instance backup.
*
* Much validation is deferred until the write() method, to prevent
* problems with files changing between the constructor and the write call.
*/
public DbBackup(File archiveFile, String dbPath) {
this.archiveFile = archiveFile;
File dbPathFile = new File(dbPath);
dbDir = dbPathFile.getAbsoluteFile().getParentFile();
instanceName = dbPathFile.getName();
componentFiles = new File[] {
new File(dbDir, instanceName + ".properties"),
new File(dbDir, instanceName + ".script"),
new File(dbDir, instanceName + ".data"),
new File(dbDir, instanceName + ".backup"),
new File(dbDir, instanceName + ".log"),
new File(dbDir, instanceName + ".lobs")
};
componentStreams = new InputStreamInterface[componentFiles.length];
existList = new boolean[componentFiles.length];
ignoreList = new boolean[componentFiles.length];
}
/**
* Used for SCRIPT backup
*/
public DbBackup(File archiveFile, String dbPath, boolean script) {
this.archiveFile = archiveFile;
File dbPathFile = new File(dbPath);
dbDir = dbPathFile.getAbsoluteFile().getParentFile();
instanceName = dbPathFile.getName();
componentFiles = new File[]{
new File(dbDir, instanceName + ".script"), };
componentStreams = new InputStreamInterface[componentFiles.length];
existList = new boolean[componentFiles.length];
ignoreList = new boolean[componentFiles.length];
abortUponModify = false;
}
/**
* Overrides file with stream.
*/
public void setStream(String fileExtension, InputStreamInterface is) {
for (int i = 0; i < componentFiles.length; i++) {
if (componentFiles[i].getName().endsWith(fileExtension)) {
componentStreams[i] = is;
break;
}
}
}
public void setFileIgnore(String fileExtension) {
for (int i = 0; i < componentFiles.length; i++) {
if (componentFiles[i].getName().endsWith(fileExtension)) {
ignoreList[i] = true;
break;
}
}
}
/**
* Defaults to false.
*
* If false, then attempts to write a tar file that already exist will
* abort.
*/
public void setOverWrite(boolean overWrite) {
this.overWrite = overWrite;
}
/**
* Defaults to true.
*
* If true, then the write() method will validate that the database is
* closed, and it will verify that no DB file changes between when we
* start writing the tar, and when we finish.
*/
public void setAbortUponModify(boolean abortUponModify) {
this.abortUponModify = abortUponModify;
}
public boolean getOverWrite() {
return overWrite;
}
public boolean getAbortUponModify() {
return abortUponModify;
}
/**
* This method always backs up the .properties and .script files.
* It will back up all of .backup, .data, and .log which exist.
*
* If abortUponModify is set, no tar file will be created, and this
* method will throw.
*
* @throws IOException for any of many possible I/O problems
* @throws IllegalStateException only if abortUponModify is set, and
* database is open or is modified.
*/
public void write() throws IOException, TarMalformatException {
long startTime = new java.util.Date().getTime();
checkEssentialFiles();
TarGenerator generator = new TarGenerator(archiveFile, overWrite,
new Integer(DbBackup.generateBufferBlockValue(componentFiles)));
for (int i = 0; i < componentFiles.length; i++) {
boolean exists = componentStreams[i] != null
|| componentFiles[i].exists();
if (!exists) {
continue;
// We've already verified that required files exist, therefore
// there is no error condition here.
}
if (ignoreList[i]) {
continue;
}
if (componentStreams[i] == null) {
generator.queueEntry(componentFiles[i].getName(),
componentFiles[i]);
existList[i] = true;
} else {
generator.queueEntry(componentFiles[i].getName(),
componentStreams[i]);
}
}
generator.write();
checkFilesNotChanged(startTime);
}
public void writeAsFiles() throws IOException {
int bufferSize = 512
* DbBackup.generateBufferBlockValue(componentFiles);
byte[] writeBuffer = new byte[bufferSize];
checkEssentialFiles();
for (int i = 0; i < componentFiles.length; i++) {
if (ignoreList[i]) {
continue;
}
if (!componentFiles[i].exists()) {
continue;
}
File outFile = new File(archiveFile, componentFiles[i].getName());
FileOutputStream fileOut = new FileOutputStream(outFile);
if (componentStreams[i] == null) {
componentStreams[i] = new InputStreamWrapper(
new FileInputStream(componentFiles[i]));
}
InputStreamInterface instream = componentStreams[i];
while (true) {
int count = instream.read(writeBuffer, 0, writeBuffer.length);
if (count <= 0) {
break;
}
fileOut.write(writeBuffer, 0, count);
}
instream.close();
fileOut.flush();
fileOut.getFD().sync();
fileOut.close();
}
}
void checkEssentialFiles()
throws FileNotFoundException, IllegalStateException {
if (!componentFiles[0].getName().endsWith(".properties")) {
return;
}
for (int i = 0; i < 2; i++) {
boolean exists = componentStreams[i] != null
|| componentFiles[i].exists();
if (!exists) {
// First 2 files are REQUIRED
throw new FileNotFoundException(
RB.file_missing.getString(
componentFiles[i].getAbsolutePath()));
}
}
if (!abortUponModify) {
return;
}
Properties p = new Properties();
FileInputStream fis = null;
try {
File propertiesFile = componentFiles[0];
fis = new FileInputStream(propertiesFile);
p.load(fis);
} catch (IOException io) {}
finally {
try {
if (fis != null) {
fis.close();
}
} catch (IOException io) {}
finally {
fis = null; // Encourage buffer GC
}
}
String modifiedString = p.getProperty("modified");
if (modifiedString != null
&& (modifiedString.equalsIgnoreCase("yes")
|| modifiedString.equalsIgnoreCase("true"))) {
throw new IllegalStateException(
RB.modified_property.getString(modifiedString));
}
}
void checkFilesNotChanged(long startTime) throws FileNotFoundException {
// abortUponModify is used with offline invocation only
if (!abortUponModify) {
return;
}
try {
for (int i = 0; i < componentFiles.length; i++) {
if (componentFiles[i].exists()) {
if (!existList[i]) {
throw new FileNotFoundException(
RB.file_disappeared.getString(
componentFiles[i].getAbsolutePath()));
}
if (componentFiles[i].lastModified() > startTime) {
throw new FileNotFoundException(
RB.file_changed.getString(
componentFiles[i].getAbsolutePath()));
}
} else if (existList[i]) {
throw new FileNotFoundException(
RB.file_appeared.getString(
componentFiles[i].getAbsolutePath()));
}
}
} catch (IllegalStateException ise) {
if (!archiveFile.delete()) {
System.out.println(
RB.cleanup_rmfail.getString(
archiveFile.getAbsolutePath()));
// Be-it-known. This method can write to stderr if
// abortUponModify is true.
}
throw ise;
}
}
/**
* @todo - Supply a version of my MemTest program which people can run
* one time when the server can be starved of RAM, and save the available
* RAM quantity to a text file. We can then really crank up the buffer
* size to make transfers really efficient.
*/
/**
* Return a 512-block buffer size suggestion, based on the size of what
* needs to be read or written, and default and typical JVM constraints.
*
* Algorithm details:
*
* Minimum system I want support is a J2SE system with 256M physical
* RAM. This system can hold a 61 MB byte array (real 1024^2 M).
* (61MB with Java 1.6, 62MB with Java 1.4).
* This decreases to just 60 MB with (pre-production, non-optimized)
* HSQLDB v. 1.9 on Java 1.6.
* Allow the user 40 MB of for data (this only corresponds to a much
* smaller quantity of real data due to the huge overhead of Java and
* database structures).
* This allows 20 MB for us to use. User can easily use more than this
* by raising JVM settings and/or getting more PRAM or VRAM.
* Therefore, ceiling = 20MB = 20 MB / .5 Kb = 40 k blocks
*
* We make the conservative simplification that each data file contains
* just one huge data entry component. This is a good estimation, since in
* most cases, the contents of the single largest file will be many orders
* of magnitude larger than the other files and the single block entry
* headers.
*
* We aim for reading or writing these biggest file with 10 reads/writes.
* In the case of READING Gzip files, there will actually be many more
* reads than this, but that's the price you pay for smaller file size.
*
*
* @param files Null array elements are permitted. They will just be
* skipped by the algorithm.
*/
static protected int generateBufferBlockValue(File[] files) {
long maxFileSize = 0;
for (int i = 0; i < files.length; i++) {
if (files[i] == null) {
continue;
}
if (files[i].length() > maxFileSize) {
maxFileSize = files[i].length();
}
}
int idealBlocks = (int) (maxFileSize / (10L * 512L));
// I.e., 1/10 of the file, in units of 512 byte blocks.
// It's fine that operations will truncate down instead of round.
if (idealBlocks < 1) {
return 1;
}
if (idealBlocks > 40 * 1024) {
return 40 * 1024;
}
return idealBlocks;
}
/**
* Convenience wrapper for generateBufferBlockValue(File[]).
*
* @see #generateBufferBlockValue(File[])
*/
static protected int generateBufferBlockValue(File file) {
return generateBufferBlockValue(new File[]{ file });
}
}