net.java.truevfs.access.TApplication Maven / Gradle / Ivy
Show all versions of truevfs-access Show documentation
/*
* Copyright (C) 2005-2015 Schlichtherle IT Services.
* All rights reserved. Use is subject to license terms.
*/
package net.java.truevfs.access;
import net.java.truevfs.kernel.spec.FsSyncException;
import net.java.truevfs.kernel.spec.FsSyncWarningException;
/**
* A template class which aids in establishing the typical
* setup-work-sync life cycle of a TrueVFS application.
* The established life cycle consists of the {@link #setup} phase and one or
* more iterations of the {@link #work} and finally {@link #sync} phases.
*
* Subclass or copy-and-edit the source code of this template class to meet
* your requirements.
*
*
* @param the {@link Exception} class to throw by {@link #work} and
* {@link #run}.
* @author Christian Schlichtherle
*/
public abstract class TApplication {
/**
* Runs the setup-work-sync life cycle.
* At first, this method calls {@link #setup}.
* Then, a loop is entered which calls {@link #work} and (in a finally
* block) {@link #sync}.
* If {@link #work} returns a negative integer, then the loop is
* repeated.
* Otherwise, the return value is used as the
* {@link System#exit(int) exit status} of the VM.
*
* Note that calling {@link #sync} in a finally-block ensures that all
* unsynchronized changes to the contents of all federated file systems
* (i.e. archive files) get committed to their respective parent file
* system, even if {@link #work} throws an exception.
*
* @param args an array of arguments for this application.
* @return The exit status of this application.
* @throws E At the discretion of the {@link #work} method.
* @throws FsSyncException At the discretion of the {@link #sync} method.
*/
protected final int run(String[] args) throws E, FsSyncException {
setup();
int status;
do {
try {
status = work(args);
} finally {
sync();
}
} while (0 > status);
return status;
}
/**
* Runs the setup phase.
*
* This method is called by {@link #run} at the start of the life cycle.
* Its task is to configure the default behavior of the TrueVFS Access File* API
* in order to answer the following questions:
*
* - Which file extensions shall get detected to indicate archive
* files and hence as virtual directories?
*
- Which kind of temporary buffers shall get used?
*
- Shall missing archive files and directory entries get automatically
* created whenever required?
*
*
* Note that the method body of the implementation in the class
* {@link TApplication} is empty!
* This means that the initial setup is not changed.
* The initial setup is determined by loading service classes from the
* class path and applying reasonable defaults if nothing is found.
*
* As an alternative to overriding this method, the setup may get changed
* in a sub class constructor.
* This would ensure it's changed exactly only once and not on every call
* to {@link #run}.
*
*
Examples
*
* As the most simple use case, a client application might want to filter
* the initial setup for the archive file extensions it wants to detect.
* This is done as follows:
*
{@code
* TConfig.get().setArchiveDetector(new TArchiveDetector("ear|jar|war"));
* }
*
* This will filter all file system drivers found on the class path in the
* initial setup so that only files with the pattern {@code *.ear},
* {@code *.jar} or {@code *.war} (case ignoring) are detected
* as prospective archive files.
* If no file system driver is present for a named extension in the initial
* setup, an {@link IllegalArgumentException} is thrown.
*
* The beauty of this simple example is that it does not require to change
* the compile time class path.
* For more advanced examples however, the file {@code pom.xml} needs to
* get edited so that the respective modules get added to the compile time
* class path.
*
* The constructors of the {@link TFile} class use the
* {@link TArchiveDetector} class to scan a path name for extensions of
* archive files which shall be treated like virtual directories.
* You can either explicitly inject this ArchiveDetector dependency into a
* TFile constructor or you can rely on the value of the class property
* {@code defaultArchiveDetector}.
*
* By default, all new {@link TFile} objects will use a default
* {@link TArchiveDetector} which recognizes the canonical archive file
* extensions registered by all archive driver modules which are present on
* the class path at run time - these can be configured by editing the
* file {@code pom.xml}.
*
* However, if you use the following statement, all {@link TFile} objects
* will use the given default {@link TArchiveDetector} which recognizes
* only the given canonical file extensions for TAR, TAR.GZ, TAR.BZ2 and
* ZIP files as archive files and hence as virtual directories.
* This requires the JARs for the archive driver modules
* {@code truevfs-driver-tar} and {@code truevfs-driver-zip} to be present
* on the class path at compile time:
*
{@code
* TConfig.get().setArchiveDetector(
* new TArchiveDetector(
* TArchiveDetector.NULL,
* new Object[][] {
* { "tar", new TarDriver(IOPoolLocator.SINGLETON) },
* { "tgz|tar.gz", new TarGZipDriver(IOPoolLocator.SINGLETON) },
* { "tbz|tb2|tar.bz2", new TarBZip2Driver(IOPoolLocator.SINGLETON) },
* { "zip", new ZipDriver(IOPoolLocator.SINGLETON)},
* }));
* }
*
* Another typical use case is to recognize only Java artifacts.
*
{@code
* TConfig.get().setArchiveDetector(
* new TArchiveDetector(
* "ear|jar|war",
* new JarDriver(IOPoolLocator.SINGLETON)));
* }
*
* ... or an application file format.
*
{@code
* TConfig.get().setArchiveDetector(
* new TArchiveDetector(
* "foo",
* new JarDriver(IOPoolLocator.SINGLETON)));
* }
*
* ... or an encrypted application file format.
* This driver authenticates input archive files up to 512 KB using the
* Message Authentication Code (MAC) specified by the RAES file format.
* For larger input archive files, it just checks the CRC-32 value whenever
* an archive entry input stream is closed.
* CRC-32 has frequent collisions when compared to a MAC.
* However, it should not be feasible to make an undetectable modification.
* The driver also uses unencrypted temporary files for archive entries
* whenever required.
*
{@code
* TConfig.get().setArchiveDetector(
* new TArchiveDetector(
* "bar",
* new SafeZipRaesDriver(IOPoolLocator.SINGLETON)));
* }
*
* If you're a bit paranoid, then you could use the following driver
* instead:
* This driver authenticates every input archive file using the Message
* Authentication Code (MAC) specified by the RAES file format, which
* makes it comparably slow.
* The driver also uses unencrypted temporary files for archive entries
* whenever required.
*
{@code
* TConfig.get().setArchiveDetector(
* new TArchiveDetector(
* "bar",
* new ParanoidZipRaesDriver(IOPoolLocator.SINGLETON)));
* }
*
* And finally, if you're quite paranoid, then this driver is for you:
* This driver authenticates every input archive file using the Message
* Authentication Code (MAC) specified by the RAES file format, which
* makes it comparably slow.
* The driver also uses unencrypted byte arrays for temporary storage of
* archive entries whenever required.
* If you were completely paranoid, you would even want to use encrypted
* byte arrays or wipe them with nulls after use.
* However, then you would have to write this yourself! ;-)
*
{@code
* TConfig.get().setArchiveDetector(
* new TArchiveDetector(
* "bar",
* new ParanoidZipRaesDriver(new ByteArrayIOPoolService(2048))));
* }
*
* Last but not least, the following class property controls whether
* archive files and their member directories get automatically created
* whenever required.
* The default value of this class property is {@code true}!
*
{@code
* TConfig.get().setLenient(false);
* }
*/
@SuppressWarnings("NoopMethodInAbstractClass")
protected void setup() { }
/**
* Runs the work phase.
*
* This method is called by {@link #run} at least once and repeatedly
* called until it returns a non-negative integer for use as the
* {@link System#exit(int) exist status} of the VM.
* After this method, the {@link #sync} method is called in a finally-block.
*
* Avoid repeating this method and updating the same archive file upon
* each call!
* This would degrade the overall performance from O(n) to O(m*n),
* where m is the number of new or modified entries and n is the number
* of all entries in the archive file!
*
* @param args an array of arguments for this application.
* @return A negative integer in order to continue calling this method
* in a loop.
* Otherwise, the return value is used as the
* {@link System#exit(int) exit status} of the VM.
* @throws E At the discretion of the implementation.
*/
protected abstract int work(String[] args) throws E;
/**
* Runs the sync phase.
*
* This method is called by {@link #run} in a finally-block after each call
* to {@link #work}.
* Its task is to commit any unsynchronized changes to the contents of
* archive files and purge all cached data so that third parties can safely
* interact with these archive files until the next access by this TrueVFS
* application.
*
* The implementation in the class {@link TApplication} simply calls
* {@link TVFS#umount() TVFS.umount()}.
*
* @throws FsSyncWarningException if only warning conditions
* occur.
* This implies that the respective parent file system has been
* updated with constraints, such as a failure to set the last
* modification time of the entry for the federated file system
* (i.e. archive file) in its parent file system.
* @throws FsSyncException if any error conditions occur.
* This implies loss of data!
*/
protected void sync() throws FsSyncException {
TVFS.umount();
}
}