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

de.schlichtherle.truezip.file.TFile Maven / Gradle / Ivy

Go to download

This module provides the TFile* classes for simple, uniform, transparent, thread-safe, read/write access to archive files as if they were virtual directories in a file system path. This module also provides Swing GUI classes for viewing file trees and choosing entries in archive files.

The newest version!
/*
 * Copyright (C) 2005-2015 Schlichtherle IT Services.
 * All rights reserved. Use is subject to license terms.
 */
package de.schlichtherle.truezip.file;

import de.schlichtherle.truezip.entry.Entry.Access;
import de.schlichtherle.truezip.entry.Entry.Size;
import static de.schlichtherle.truezip.entry.Entry.Type.DIRECTORY;
import static de.schlichtherle.truezip.entry.Entry.Type.FILE;
import static de.schlichtherle.truezip.entry.Entry.UNKNOWN;
import static de.schlichtherle.truezip.entry.EntryName.SEPARATOR_CHAR;
import static de.schlichtherle.truezip.file.ExpertFeature.Level.*;
import static de.schlichtherle.truezip.file.ExpertFeature.Reason.*;
import de.schlichtherle.truezip.fs.*;
import static de.schlichtherle.truezip.fs.FsEntryName.ROOT;
import static de.schlichtherle.truezip.fs.FsOutputOption.EXCLUSIVE;
import static de.schlichtherle.truezip.fs.FsOutputOption.GROW;
import static de.schlichtherle.truezip.fs.FsSyncOption.*;
import static de.schlichtherle.truezip.fs.FsSyncOptions.UMOUNT;
import static de.schlichtherle.truezip.fs.FsUriModifier.CANONICALIZE;
import de.schlichtherle.truezip.io.Paths;
import de.schlichtherle.truezip.io.Paths.Splitter;
import de.schlichtherle.truezip.io.Streams;
import de.schlichtherle.truezip.util.BitField;
import de.schlichtherle.truezip.util.JSE7;
import de.schlichtherle.truezip.util.UriBuilder;
import java.io.*;
import java.net.MalformedURLException;
import java.net.URI;
import java.net.URISyntaxException;
import java.net.URL;
import java.util.*;
import javax.annotation.CheckForNull;
import javax.annotation.Nullable;
import javax.annotation.WillClose;
import javax.annotation.WillNotClose;
import javax.annotation.concurrent.Immutable;
import javax.swing.Icon;
import javax.swing.filechooser.FileSystemView;

/**
 * A replacement for the class {@link File} which provides transparent
 * read/write access to archive files and their entries as if they were
 * (virtual) directories and files.
 * Because this class actually extends the class {@link File} it can get used
 * polymorphically with the class {@link FileSystemView} or any other class
 * which depends on the class {@link File}.
 *
 * 

Bulk I/O Methods

*

* This class provides some convenient methods which use pooled buffers and * pooled threads in order to achieve superior performance as compared to the * naive read-then-write-in-a-loop approach. * These bulk I/O methods fall into the following categories: *

    *
  1. The {@code cp(_p|_r|_rp)?} methods copy files or directory trees. * The method names have been modeled after the Unix command line utility * {@code cp} with its options. *
  2. The {@code mv} methods move files or directory trees. * The method names have been modeled after the Unix command line utility * {@code mv}. *
  3. The {@code rm(_r)?} methods remove files or directory trees. * The method names have been modeled after the Unix command line utility * {@code rm} with its options. *
  4. The {@code input|output} methods copy the given streams to this file or * vice versa. * In contrast to the previous methods they never close their argument * streams, so applications can call them multiple times on the same * streams to concatenate data. *
  5. Finally, the {@link #cat(InputStream, OutputStream)} * method is the core copy engine for all these methods. * It performs the data transfer from an input stream to an output stream. * When used with unbuffered input and output stream * implementations, it delivers the same performance as the transfer * method in the package {@code java.nio}. * Its name is modeled after the Unix command line utility {@code cat}. *
* Important: You must provide the full path name for both * source and destination parameters to any of these methods! * In particular, both the source and destination parameters must either * represent a file or a directory - mixing file and directory parameters will * not work. * This constraint is meant to prevent ambiguous method semantics. * *

Raw Data Copying (RDC)

*

* [Note that this feature has been renamed from Direct Data Copying (DDC) * to Raw Data Copying (RDC) in TrueZIP 7.5] *

* If data is copied from an archive file to another archive file of the * same type, some of the copy methods support a feature called Raw Data * Copying (RDC) to achieve best performance: * With RDC, the raw entry data is copied from the input entry to the * output entry without the need to temporarily reproduce, copy and * process the original entry data again. *

* The benefits of this feature are archive driver specific: * In case of ZIP files with compressed entries, RDC avoids inflating the * entry data when reading the input entry and immediately deflating it * again when writing the output entry. * In case of TAR files, RDC avoids the need to create an additional temporary * file, but shows no impact otherwise because the TAR file format doesn't * support compression. * *

Traversing Directory Trees

*

* When traversing directory trees, e.g. when searching, copying or moving * them, it's important that all file objects use * {@linkplain TArchiveDetector#equals equivalent} {@link TArchiveDetector} * objects for archive file detection in these directory trees. * This is required in order to make sure that the virtual file system state * which is managed by the TrueZIP Kernel does not get bypassed. * Otherwise, file system operations on archive files would yield inconsistent * results and may even cause loss of data! *

* By default, all file objects use {@link TArchiveDetector#ALL} in order to * detect all supported archive types (see {@link TConfig} for other options). * This is fine because it's fail-safe and performs reasonably well when * copying archive files (e.g. ZIP entries won't get recompressed thanks to * RDC). * *

Making Verbatim Copies of Directory Trees

*

* Using the default {@code TArchiveDetector.ALL} results in structural * copies rather than verbatim copies (byte-by-byte copies) of any * archive files within the source directory tree. * This saves you from accidentally bypassing the virtual file system state * which is managed by the TrueZIP Kernel. *

* However, sometimes you may need verbatim copies, e.g. when comparing hash * sums for archive files. * In this case, you need to unmount any archive file in the source and * destination directory trees first, e.g. by calling {@link TVFS#umount()} or * {@link TVFS#umount(TFile)} before you can copy the directory trees. * For the subsequent traversal of the directory trees, you need use * {@code TFile} objects which do not recognize prospective archive * files. * For example, to make a recursive verbatim copy, you could use this: *


 * TFile src = ...; // any
 * TFile dst = ...; // dito
 * ... // any I/O operations
 * TVFS.umount(src); // unmount selectively
 * TVFS.umount(dst); // dito
 * src.toNonArchiveFile().cp_rp(dst.toNonArchiveFile());
 * 
*

* The calls to {@link TVFS#umount(TFile)} selectively unmount any archive * files within the source and destination directory trees and the calls to * {@link #toNonArchiveFile()} ensure that prospective archive file detection * is inhibited when recursively copying the source and destination directory * trees. * *

Detecting Archive Files and False Positives

*

* Whenever an archive file suffix is detected in a path, this class treats * the corresponding file or directory as a prospective archive file. * The word "prospective" suggests that just because a file is named * archive.zip it isn't necessarily a valid ZIP file. * In fact, it could be anything, even a plain directory in the platform file system! *

* Such an invalid archive file is called a false positive archive file. * TrueZIP correctly identifies all types of false positive archive files by * performing a recursive look up operation for the first parent file system * where the respective prospective archive file actually exists and treats it * according to its true state. *

* The following table shows how certain methods in this class behave, * depending upon an archive file's path and its true state in the * first parent file system where it actually exists. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
PathTrue State{@code isArchive()}1{@code isDirectory()}{@code isFile()}{@code exists()}{@code length()}2
archive.zip3Valid ZIP file{@code true}{@code true}{@code false}{@code true}{@code 0}
archive.zipFalse positive: Plain directory{@code true}{@code true}{@code false}{@code true}{@code ?}
archive.zipFalse positive: Plain file{@code true}{@code false}{@code true}{@code true}{@code ?}
archive.zipFalse positive: Regular special file{@code true}{@code false}{@code false}{@code true}{@code ?}
archive.zipFile or directory does not exist{@code true}{@code false}{@code false}{@code false}{@code 0}
 
archive.tzp4Valid RAES encrypted ZIP file with valid key (e.g. password){@code true}{@code true}{@code false}{@code true}{@code 0}
archive.tzpValid RAES encrypted ZIP file with unknown key5{@code true}{@code false}{@code false}{@code true}{@code ?}
archive.tzpFalse positive: Plain directory{@code true}{@code true}{@code false}{@code true}{@code ?}
archive.tzpFalse positive: Plain file{@code true}{@code false}{@code true}{@code true}{@code ?}
archive.tzpFalse positive: Regular special file{@code true}{@code false}{@code false}{@code true}{@code ?}
archive.tzpFile or directory does not exist{@code true}{@code false}{@code false}{@code false}{@code 0}
 
otherPlain directory{@code false}{@code true}{@code false}{@code true}{@code ?}
otherPlain file{@code false}{@code false}{@code true}{@code true}{@code ?}
otherRegular special file{@code false}{@code false}{@code false}{@code true}{@code ?}
otherFile or directory does not exist{@code false}{@code false}{@code false}{@code false}{@code 0}
*
    *
  1. {@link #isArchive} doesn't check the true state of the file - it just * looks at its path: If the path ends with a configured archive file * suffix, {@code isArchive()} always returns {@code true}. *
  2. {@link #length} always returns {@code 0} if the path denotes a * valid archive file. * Otherwise, the return value of {@code length()} depends on the * platform file system, which is indicated by {@code ?}. * For regular directories on Windows/NTFS for example, the return value * would be {@code 0}. *
  3. This example presumes that the JAR of the module * TrueZIP Driver ZIP is present on the run time class path.
  4. *
  5. This example presumes that the JAR of the module * TrueZIP Driver ZIP.RAES (TZP) is present on the run time * class path.
  6. *
  7. The methods behave exactly the same for both archive.zip and * archive.tzp with one exception: If the key for a RAES encrypted * ZIP file remains unknown (e.g. because the user cancelled password * prompting), then these methods behave as if the true state of the path * were a special file, i.e. both {@link #isDirectory} and {@link #isFile} * return {@code false}, while {@link #exists} returns {@code true}.
  8. *
* * @author Christian Schlichtherle */ @Immutable public final class TFile extends File { private static final long serialVersionUID = 3617072259051821745L; /** The prefix of a UNC (a Windows concept). */ private static final String UNC_PREFIX = separator + separator; /** The file system roots. */ private static final Set ROOTS = Collections.unmodifiableSet( new TreeSet(Arrays.asList(listRoots()))); private static final File CURRENT_DIRECTORY = new File("."); /** * The decorated file is used to implement the behaviour of the file system * operations in case this instance represents neither an archive file * nor an entry in an archive file. * If this instance is constructed from another {@code File} * instance, then this field is initialized with that instance. *

* This enables federation of file system implementations and is essential * to enable the broken implementation in * {@link javax.swing.JFileChooser} to browse archive files. */ private transient File file; private transient TArchiveDetector detector; private transient @CheckForNull TFile innerArchive; private transient @CheckForNull TFile enclArchive; private transient @CheckForNull FsEntryName enclEntryName; /** * This refers to the file system controller if and only if this file * refers to a prospective archive file, otherwise it's {@code null}. * This field should be considered to be {@code final}! * * @see #readObject */ @edu.umd.cs.findbugs.annotations.SuppressWarnings("JCIP_FIELD_ISNT_FINAL_IN_IMMUTABLE_CLASS") private transient volatile @CheckForNull FsController controller; /** * Copy constructor. * Equivalent to {@link #TFile(File, TArchiveDetector) * new TFile(template, (TArchiveDetector) null)}. * * @param file the file object to decorate. */ public TFile(File file) { this(file, (TArchiveDetector) null); } /** * Constructs a new {@code TFile} instance which may use the given archive * detector to scan the path name for prospective archive files. * * @param file the file object to decorate. * If this is an instance of this class, its fields are copied and * the {@code detector} parameter is ignored. * @param detector the archive detector to use for scanning the path name * for prospective archive files. * This parameter is ignored if and only if {@code file} is an * instance of this class. * Otherwise, if this parameter is {@code null}, then the * {@linkplain TConfig#getArchiveDetector default archive detector} * is used instead. */ @ExpertFeature(INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public TFile( final File file, final @CheckForNull TArchiveDetector detector) { super(file.getPath()); if (file instanceof TFile) { final TFile tfile = (TFile) file; this.file = tfile.file; this.detector = tfile.detector; this.enclArchive = tfile.enclArchive; this.enclEntryName = tfile.enclEntryName; this.innerArchive = tfile.isArchive() ? this : tfile.innerArchive; this.controller = tfile.controller; } else { this.file = file; this.detector = null != detector ? detector : TConfig.get().getArchiveDetector(); scan(null); } assert invariants(); } /** * Equivalent to {@link #TFile(String, TArchiveDetector) * new TFile(path, (TArchiveDetector) null)}. *

* The {@linkplain TConfig#getArchiveDetector default archive detector} * is used to scan the path name for prospective archive files. * * @param path the path name. */ public TFile(String path) { this(path, (TArchiveDetector) null); } /** * Constructs a new {@code TFile} instance which may use the given * {@link TArchiveDetector} to scan its path name for prospective archive * files. * * @param path the path name. * @param detector the archive detector to use for scanning the path name * for prospective archive files. * If this parameter is {@code null}, then the * {@linkplain TConfig#getArchiveDetector default archive detector} * is used instead. */ @ExpertFeature(INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public TFile( final String path, final @CheckForNull TArchiveDetector detector) { super(path); this.file = new File(path); this.detector = null != detector ? detector : TConfig.get().getArchiveDetector(); scan(null); assert invariants(); } /** * Equivalent to {@link #TFile(String, String, TArchiveDetector) * new TFile(parent, child, null)}. *

* The {@linkplain TConfig#getArchiveDetector default archive detector} * is used to scan the entire path name for prospective archive * files. * * @param parent the parent directory. * @param member the member path name. */ public TFile(@CheckForNull String parent, String member) { this(parent, member, null); } /** * Constructs a new {@code TFile} instance which may use the given archive * detector to scan the entire path name for prospective archive * files. * * @param parent the parent directory. * @param member the member path name. * @param detector the archive detector to use for scanning the path name * for prospective archive files. * If this parameter is {@code null}, then the * {@linkplain TConfig#getArchiveDetector default archive detector} * is used instead. */ @ExpertFeature(INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public TFile( final @CheckForNull String parent, final String member, final @CheckForNull TArchiveDetector detector) { super(parent, member); this.file = new File(parent, member); this.detector = null != detector ? detector : TConfig.get().getArchiveDetector(); scan(null); assert invariants(); } /** * Equivalent to {@link #TFile(File, String, TArchiveDetector) * new TFile(parent, child, null)}. * * @param parent the parent directory. * If this parameter is an instance of this class, its archive * detector is used to scan only the member path name * for prospective archive files. * Otherwise, the * {@linkplain TConfig#getArchiveDetector default archive detector} * is used to to scan the entire path name * for prospective archive files. * @param member the member path name. */ public TFile(@CheckForNull File parent, String member) { this(parent, member, null); } /** * Constructs a new {@code TFile} instance which may use the given archive * detector to scan the path name for prospective archive files. *

* If {@code parent} is an instance of this class, * then {@code detector} is used to scan only the member path name * for prospective archive files. * If {@code detector} is {@code null}, then the * parent's archive detector * is used instead. *

* Otherwise, if {@code parent} is not an instance of this class, * then {@code detector} is used to scan the entire path name * for prospective archive files. * If {@code detector} is {@code null}, then the * {@linkplain TConfig#getArchiveDetector default archive detector} * is used instead. * * @param parent the parent directory. * @param member the member path name. * @param detector the archive detector to use for scanning the path name * for prospective archive files. * If this parameter is {@code null}, then the * {@linkplain TConfig#getArchiveDetector default archive detector} * is used instead. */ @ExpertFeature(INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public TFile( final @CheckForNull File parent, final String member, final @CheckForNull TArchiveDetector detector) { super(parent, member); this.file = new File(parent, member); if (parent instanceof TFile) { final TFile p = (TFile) parent; this.detector = null != detector ? detector : p.detector; scan(p); } else { this.detector = null != detector ? detector : TConfig.get().getArchiveDetector(); scan(null); } assert invariants(); } /** * Constructs a new {@code TFile} instance from the given {@code uri}. * This constructor is equivalent to * new {@link #TFile(FsPath, TArchiveDetector) TFile(FsPath.create(uri, CANONICALIZE), null))}, * * @param uri an absolute URI which has a scheme component which is * known by the * {@linkplain TConfig#getArchiveDetector default archive detector}. * @throws IllegalArgumentException if the given URI does not conform to * the syntax constraints for {@link FsPath}s or * {@link File#File(URI)}. * @see #toURI() * @see #TFile(FsPath) */ public TFile(URI uri) { this(FsPath.create(uri, CANONICALIZE), null); } /** * Constructs a new {@code TFile} instance from the given {@code path}. * This constructor is equivalent to * new {@link #TFile(FsPath, TArchiveDetector) TFile(path, null)} * * @param path a path with an absolute * {@link FsPath#toHierarchicalUri() hierarchical URI} which has a * scheme component which is known by the * {@linkplain TConfig#getArchiveDetector default archive detector}. * @throws IllegalArgumentException if the * {@link FsPath#toHierarchicalUri() hierarchical URI} of the given * path does not conform to the syntax constraints for * {@link File#File(URI)}. * @see #toFsPath() * @see #TFile(URI) */ public TFile(FsPath path) { this(path, null); } /** * Constructs a new {@code TFile} instance for the given {@code path} and * {@code detector}. *

* This constructor is a super set of the super class constructor * {@link File#File(URI)} with the following additional features: * If the given URI is opaque, it must match the pattern * {@code :!/}. * The constructed file object then parses the URI to address an entry in * a federated file system (i.e. prospective archive file) with the name * {@code } in the prospective archive file addressed by * {@code } which is of the type identified by {@code } . * This is recursively applied to access entries within other prospective * archive files until {@code } is a hierarchical URI. * The scheme component of this hierarchical URI must be {@code file}. * * @param path a path with an absolute * {@link FsPath#toHierarchicalUri() hierarchical URI} which has a * scheme component which is known by the given {@code detector}. * @param detector the archive detector to look up archive file system * drivers for the named URI scheme components. * If this parameter is {@code null}, then the * {@linkplain TConfig#getArchiveDetector default archive detector}. * is used instead. * @throws IllegalArgumentException if the * {@link FsPath#toHierarchicalUri() hierarchical URI} of the given * path does not conform to the syntax constraints for * {@link File#File(URI)}. * @see #toFsPath() * @since TrueZIP 7.3.2 */ @ExpertFeature(INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public TFile( final FsPath path, final @CheckForNull TArchiveDetector detector) { super(path.toHierarchicalUri()); parse(path, null != detector ? detector : TConfig.get().getArchiveDetector()); } private void parse( final FsPath path, final TArchiveDetector detector) { this.file = new File(super.getPath()); this.detector = detector; final FsMountPoint mp = path.getMountPoint(); final FsPath mpp = mp.getPath(); final FsEntryName en; if (null == mpp) { assert !path.toUri().isOpaque(); this.enclArchive = null; this.enclEntryName = null; this.innerArchive = null; } else if ((en = path.getEntryName()).isRoot()) { assert path.toUri().isOpaque(); if (mpp.toUri().isOpaque()) { this.enclArchive = new TFile(mpp.getMountPoint(), detector); this.enclEntryName = mpp.getEntryName(); } else { this.enclArchive = null; this.enclEntryName = null; } this.innerArchive = this; // See http://java.net/jira/browse/TRUEZIP-154 . this.controller = getController(mp); } else { assert path.toUri().isOpaque(); this.enclArchive = new TFile(mp, detector); this.enclEntryName = en; this.innerArchive = this.enclArchive; } assert invariants(); } @SuppressWarnings("LeakingThisInConstructor") private TFile( final FsMountPoint mountPoint, final TArchiveDetector detector) { super(mountPoint.toHierarchicalUri()); this.file = new File(super.getPath()); this.detector = detector; final FsPath mpp = mountPoint.getPath(); if (null == mpp) { assert !mountPoint.toUri().isOpaque(); this.enclArchive = null; this.enclEntryName = null; this.innerArchive = null; } else { assert mountPoint.toUri().isOpaque(); if (mpp.toUri().isOpaque()) { this.enclArchive = new TFile(mpp.getMountPoint(), detector); this.enclEntryName = mpp.getEntryName(); } else { this.enclArchive = null; this.enclEntryName = null; } this.innerArchive = this; this.controller = getController(mountPoint); } assert invariants(); } @SuppressWarnings("LeakingThisInConstructor") private TFile( final File file, final @CheckForNull TFile innerArchive, final TArchiveDetector detector) { super(file.getPath()); this.file = file; final String path = file.getPath(); if (null != innerArchive) { final int iapl = innerArchive.getPath().length(); if (path.length() == iapl) { this.detector = innerArchive.detector; this.enclArchive = innerArchive.enclArchive; this.enclEntryName = innerArchive.enclEntryName; this.innerArchive = this; this.controller = innerArchive.controller; } else { this.detector = detector; this.innerArchive = this.enclArchive = innerArchive; try { this.enclEntryName = new FsEntryName( new UriBuilder() .path( path.substring(iapl + 1) // cut off leading separatorChar .replace(separatorChar, SEPARATOR_CHAR)) .getUri(), CANONICALIZE); } catch (URISyntaxException ex) { throw new AssertionError(ex); } } } else { this.detector = detector; } assert invariants(); } /** * Initialize this file object by scanning its path for archive * files, using the given {@code ancestor} file (i.e. a direct or * indirect parent file) if any. * {@code file} and {@code detector} must already be initialized! * Must not be called to re-initialize this object! */ private void scan(final @CheckForNull TFile ancestor) { final String path = super.getPath(); assert ancestor == null || path.startsWith(ancestor.getPath()); assert file.getPath().equals(path); assert null != detector; final StringBuilder enclEntryNameBuf = new StringBuilder(path.length()); scan(ancestor, detector, 0, path, enclEntryNameBuf, new Splitter(separatorChar, false)); try { enclEntryName = 0 >= enclEntryNameBuf.length() ? null : new FsEntryName( new UriBuilder().path(enclEntryNameBuf.toString()).getUri(), CANONICALIZE); } catch (URISyntaxException ex) { throw new AssertionError(ex); } } private void scan( @CheckForNull TFile ancestor, TArchiveDetector detector, int skip, final String path, final StringBuilder enclEntryNameBuf, final Splitter splitter) { if (path == null) { assert null == enclArchive; enclEntryNameBuf.setLength(0); return; } splitter.split(path); final String parent = splitter.getParentPath(); final String member = splitter.getMemberName(); if (0 == member.length() || ".".equals(member)) { // Fall through. } else if ("..".equals(member)) { skip++; } else if (0 < skip) { skip--; } else { if (null != ancestor) { final int pathLen = path.length(); final int ancestorPathLen = ancestor.getPath().length(); if (pathLen == ancestorPathLen) { // Found ancestor: Process it and stop. // Using the following assertion would be wrong: // enclEntryNameBuf may indeed be null if the full path // ends with just a single dot after the last separator, // i.e. the base name is ".", indicating the current // directory. // assert enclEntryNameBuf.getLength() > 0; enclArchive = ancestor.innerArchive; if (!ancestor.isArchive()) { if (ancestor.isEntry()) { assert null != ancestor.enclEntryName; if (0 < enclEntryNameBuf.length()) { enclEntryNameBuf.insert(0, '/'); enclEntryNameBuf.insert(0, ancestor.enclEntryName.getPath()); } else { // TODO: Simplify this! // Example: new TFile(new TFile(new TFile("archive.zip"), "entry"), ".") assert enclArchive == ancestor.enclArchive; enclEntryNameBuf.append(ancestor.enclEntryName.getPath()); } } else { assert null == enclArchive; enclEntryNameBuf.setLength(0); } } else if (0 >= enclEntryNameBuf.length()) { // TODO: Simplify this! // Example: new TFile(new TFile("archive.zip"), ".") assert enclArchive == ancestor; innerArchive = this; enclArchive = ancestor.enclArchive; if (ancestor.enclEntryName != null) enclEntryNameBuf.append(ancestor.enclEntryName.getPath()); } if (this != innerArchive) innerArchive = enclArchive; return; } else if (pathLen < ancestorPathLen) { detector = ancestor.detector; ancestor = ancestor.enclArchive; } } final boolean isArchive = null != detector.getScheme(path); if (0 < enclEntryNameBuf.length()) { if (isArchive) { enclArchive = new TFile(path, detector); // use the same detector for the parent directory if (innerArchive != this) innerArchive = enclArchive; return; } enclEntryNameBuf.insert(0, '/'); enclEntryNameBuf.insert(0, member); } else { if (isArchive) innerArchive = this; enclEntryNameBuf.append(member); } } scan(ancestor, detector, skip, parent, enclEntryNameBuf, splitter); } private Object writeReplace() throws ObjectStreamException { return getAbsoluteFile(); } private void writeObject(ObjectOutputStream out) throws IOException { out.writeObject(toURI()); } private void readObject(ObjectInputStream in) throws IOException, ClassNotFoundException { parse( FsPath.create((URI) in.readObject(), CANONICALIZE), TConfig.get().getArchiveDetector()); } /** * Checks the invariants of this class and throws an AssertionError if * any is violated even if assertion checking is disabled. *

* The constructors call this method like this: *

{@code assert invariants();}
* This calls the method if and only if assertions are enabled in order * to assert that the instance invariants are properly obeyed. * If assertions are disabled, the call to this method is thrown away by * the HotSpot compiler, so there is no performance penalty. * * @throws AssertionError If assertions are enabled and any invariant is * violated. * @return {@code true} */ private boolean invariants() { // Thread-safe caching final File file = this.file; final TFile innerArchive = this.innerArchive; final TFile enclArchive = this.enclArchive; final FsEntryName enclEntryName = this.enclEntryName; assert null != file; assert !(file instanceof TFile); assert file.getPath().equals(super.getPath()); assert null != detector; assert (null != innerArchive) == (getInnerEntryName() != null); assert (null != enclArchive) == (enclEntryName != null); assert this != enclArchive; assert (this == innerArchive) ^ (innerArchive == enclArchive && null == controller); assert null == enclArchive || Paths.contains( enclArchive.getPath(), file.getParentFile().getPath(), separatorChar) && !enclEntryName.toString().isEmpty(); return true; } /** * @deprecated As of TrueZIP 7.5, replaced by {@link TVFS#sync(BitField)}. * @see
#TRUEZIP-236 */ @Deprecated public static void sync(BitField options) throws FsSyncException { TVFS.sync(options); } /** * @deprecated As of TrueZIP 7.5, replaced by {@link TVFS#sync(TFile, BitField)}. * @see #TRUEZIP-236 */ @Deprecated public static void sync(TFile archive, BitField options) throws FsSyncException { if (!archive.isArchive()) throw new IllegalArgumentException(archive + " (not an archive file)"); if (null != archive.getEnclArchive()) throw new IllegalArgumentException(archive + " (not a top level archive file)"); TVFS.sync(archive, options); } /** * @deprecated As of TrueZIP 7.5, replaced by {@link TVFS#umount()}. * @see #TRUEZIP-236 */ @Deprecated public static void umount() throws FsSyncException { TVFS.umount(); } /** * @deprecated As of TrueZIP 7.5, replaced by {@link TVFS#sync(BitField)}. * @see #TRUEZIP-236 */ @Deprecated public static void umount(boolean forceCloseInputAndOutput) throws FsSyncException { TVFS.sync(BitField .of(CLEAR_CACHE) .set(FORCE_CLOSE_INPUT, forceCloseInputAndOutput) .set(FORCE_CLOSE_OUTPUT, forceCloseInputAndOutput)); } /** * @deprecated As of TrueZIP 7.5, replaced by {@link TVFS#sync(BitField)}. * @see #TRUEZIP-236 */ @Deprecated public static void umount( boolean waitCloseInput, boolean forceCloseInput, boolean waitCloseOutput, boolean forceCloseOutput) throws FsSyncException { TVFS.sync(BitField .of(CLEAR_CACHE) .set(WAIT_CLOSE_INPUT, waitCloseInput) .set(FORCE_CLOSE_INPUT, forceCloseInput) .set(WAIT_CLOSE_OUTPUT, waitCloseOutput) .set(FORCE_CLOSE_OUTPUT, forceCloseOutput)); } /** * @deprecated As of TrueZIP 7.5, replaced by {@link TVFS#umount(TFile)}. * @see #TRUEZIP-236 */ @Deprecated public static void umount(TFile archive) throws FsSyncException { sync(archive, UMOUNT); } /** * @deprecated Calling this method is equivalent to * {@link TVFS#sync(TFile, BitField) * TVFS.sync(archive, * BitField.of(FsSyncOption.CLEAR_CACHE) * .set(FsSyncOption.FORCE_CLOSE_INPUT, forceCloseInputAndOutput) * .set(FsSyncOption.FORCE_CLOSE_OUTPUT, forceCloseInputAndOutput)) * }. * @see #TRUEZIP-236 */ @Deprecated public static void umount(TFile archive, boolean forceCloseInputAndOutput) throws FsSyncException { sync( archive, BitField.of(CLEAR_CACHE) .set(FORCE_CLOSE_INPUT, forceCloseInputAndOutput) .set(FORCE_CLOSE_OUTPUT, forceCloseInputAndOutput)); } /** * @deprecated Calling this method is equivalent to * {@link TVFS#sync(TFile, BitField) * TVFS.sync(archive, * BitField.of(FsSyncOption.CLEAR_CACHE) * .set(FsSyncOption.WAIT_CLOSE_INPUT, waitCloseInput) * .set(FsSyncOption.FORCE_CLOSE_INPUT, forceCloseInput) * .set(FsSyncOption.WAIT_CLOSE_OUTPUT, waitCloseOutput) * .set(FsSyncOption.FORCE_CLOSE_OUTPUT, forceCloseOutput)) * }. * @see #TRUEZIP-236 */ @Deprecated public static void umount(TFile archive, boolean waitCloseInput, boolean forceCloseInput, boolean waitCloseOutput, boolean forceCloseOutput) throws FsSyncException { sync( archive, BitField.of(CLEAR_CACHE) .set(WAIT_CLOSE_INPUT, waitCloseInput) .set(FORCE_CLOSE_INPUT, forceCloseInput) .set(WAIT_CLOSE_OUTPUT, waitCloseOutput) .set(FORCE_CLOSE_OUTPUT, forceCloseOutput)); } /** * @deprecated Calling this method is equivalent to * {@link TConfig#isLenient TConfig.get().isLenient()}. * @see #TRUEZIP-238 */ @Deprecated public static boolean isLenient() { return TConfig.get().isLenient(); } /** * @deprecated Calling this method is equivalent to * {@link TConfig#setLenient TConfig.get().setLenient(lenient)}. * @see #TRUEZIP-238 */ @Deprecated public static void setLenient(boolean lenient) { TConfig.get().setLenient(lenient); } /** * @deprecated Calling this method is equivalent to * {@link TConfig#getArchiveDetector TConfig.get().getArchiveDetector()}. * @see #TRUEZIP-238 */ @Deprecated public static TArchiveDetector getDefaultArchiveDetector() { return TConfig.get().getArchiveDetector(); } /** * @deprecated Calling this method is equivalent to * {@link TConfig#setArchiveDetector TConfig.get().setArchiveDetector(detector)}. * @see #TRUEZIP-238 */ @Deprecated public static void setDefaultArchiveDetector(TArchiveDetector detector) { TConfig.get().setArchiveDetector(detector); } /** * Returns a file object for the same path name, but does not detect any * archive file name patterns in the last path name segment. * The parent file object is unaffected by this transformation, so the * path name of this file object may address an entry in an archive file. * * @return A file object for the same path name, but does not detect any * archive file name patterns in the last path name segment. * @see TVFS#umount(TFile) * @since TrueZIP 7.5 */ @ExpertFeature( level=INTERMEDIATE, value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public TFile toNonArchiveFile() { return isArchive() ? new TFile(getParentFile(), getName(), TArchiveDetector.NULL) : this; } /** * Returns the first parent directory (starting from this file) which is * not an archive file or a file located in an archive file. * * @return The first parent directory (starting from this file) which is * not an archive file or a file located in an archive * file. */ public @Nullable TFile getNonArchivedParentFile() { final TFile enclArchive = this.enclArchive; return null != enclArchive ? enclArchive.getNonArchivedParentFile() : getParentFile(); } @Override public @Nullable String getParent() { return file.getParent(); } @Override public @Nullable TFile getParentFile() { final File parent = file.getParentFile(); if (parent == null) return null; final TFile enclArchive = this.enclArchive; if (null != enclArchive && enclArchive.getPath().length() == parent.getPath().length()) { assert enclArchive.getPath().equals(parent.getPath()); return enclArchive; } // This is not only called for performance reasons, but also in order // to prevent the parent path from being rescanned for archive files // with a different detector, which could trigger an update and // reconfiguration of the respective file system controller! return new TFile(parent, enclArchive, detector); } @Override public TFile getAbsoluteFile() { final String p = getAbsolutePath(); return p.equals(getPath()) ? this : new TFile(p, detector); } @Override public String getAbsolutePath() { return file.getAbsolutePath(); } /** * Similar to {@link #getAbsoluteFile()}, but removes any * {@code "."} and {@code ".."} directories from the path name wherever * possible. * The result is similar to {@link #getCanonicalFile()}, but symbolic * links are not resolved. * This may be useful if {@code getCanonicalFile()} throws an * IOException. * * @return The normalized absolute file object denoting the same file or * directory as this instance. * @see #getCanonicalFile() * @see #getNormalizedFile() */ public TFile getNormalizedAbsoluteFile() { final String p = getNormalizedAbsolutePath(); return p.equals(getPath()) ? this : new TFile(p, detector); } /** * Similar to {@link #getAbsolutePath()}, but removes any redundant * {@code "."} and {@code ".."} directories from the path name. * The result is similar to {@link #getCanonicalPath()}, but symbolic * links are not resolved. * This may be useful if {@code getCanonicalPath()} throws an * IOException. * * @return The normalized absolute path string denoting the same file or * directory as this instance. * @see #getCanonicalPath() * @see #getNormalizedPath() */ public String getNormalizedAbsolutePath() { return Paths.normalize(getAbsolutePath(), separatorChar); } /** * Removes any redundant {@code "."} and {@code ".."} directories from the * path name. * * @return The normalized file object denoting the same file or * directory as this instance. */ public TFile getNormalizedFile() { final String p = getNormalizedPath(); return p.equals(getPath()) ? this : new TFile(p, detector); } /** * Removes any redundant {@code "."}, {@code ".."} directories from the * path name. * * @return The normalized path string denoting the same file or * directory as this instance. */ public String getNormalizedPath() { return Paths.normalize(getPath(), separatorChar); } @Override public TFile getCanonicalFile() throws IOException { final String p = getCanonicalPath(); return p.equals(getPath()) ? this : new TFile(p, detector); } @Override public String getCanonicalPath() throws IOException { return file.getCanonicalPath(); } /** * This convenience method simply returns the canonical form of this * abstract path or the normalized absolute form if resolving the * prior fails. * * @return The canonical or absolute path of this file as an * instance of this class. */ public TFile getCanOrAbsFile() { final String p = getCanOrAbsPath(); return p.equals(getPath()) ? this : new TFile(p, detector); } /** * This convenience method simply returns the canonical form of this * abstract path or the normalized absolute form if resolving the * prior fails. * * @return The canonical or absolute path of this file as a * {@code String} instance. */ public String getCanOrAbsPath() { try { return getCanonicalPath(); } catch (IOException ex) { return Paths.normalize(getAbsolutePath(), separatorChar); } } @Override public String getPath() { return file.getPath(); } @Override public String getName() { return file.getName(); } /** * Returns the {@link TArchiveDetector} that was used to detect any archive * files in the path of this file object at construction time. * * @return The {@link TArchiveDetector} that was used to detect any archive * files in the path of this file object at construction time. */ public TArchiveDetector getArchiveDetector() { return detector; } /** * Returns {@code true} if and only if this {@code TFile} addresses an * archive file. * Whether or not this is true solely depends on the * {@link TArchiveDetector} which was used to construct this {@code TFile} * - no file system tests are performed by this method! * * @return {@code true} if and only if this {@code TFile} addresses an * archive file. * @see #isEntry * @see #isDirectory * @see Detecting Archive Paths and False Positives */ public boolean isArchive() { return this == innerArchive; } /** * Returns {@code true} if and only if this {@code TPath} addresses an * entry located within an archive file. * Whether or not this is true solely depends on the * {@link TArchiveDetector} which was used to construct this {@code TPath} * - no file system tests are performed by this method! * * @return {@code true} if and only if this {@code TPath} addresses an * entry located within an archive file. * @see #isArchive * @see #isDirectory * @see Detecting Archive Paths and False Positives */ public boolean isEntry() { return enclEntryName != null; } /** * Returns the innermost archive file object for this file object. * That is, if this object addresses an archive file, then this method * returns {@code this}. * If this object addresses an entry located within an archive file, then * this methods returns the file object representing the enclosing archive * file, or {@code null} otherwise. *

* This method always returns a normalized path, i.e. all occurences of * {@code "."} and {@code ".."} in the path name are removed according to * their meaning wherever possible. *

* In order to support unlimited nesting levels, this method returns * a {@code TFile} instance which may recursively address an entry within * another archive file. * * @return The innermost archive path object for this path object. */ public @CheckForNull TFile getInnerArchive() { return innerArchive; } /** * Returns the entry name relative to the innermost archive file. * That is, if this object addresses an archive file, then this method * returns the empty string {@code ""}. * If this object addresses an entry located within an archive file, * then this method returns the relative path of the entry in the * enclosing archive file separated by the entry separator character * {@code '/'}, or {@code null} otherwise. *

* This method always returns an undotified path, i.e. all redundant * occurences of {@code "."} and {@code ".."} in the path are removed * wherever possible. * * @return The entry name relative to the innermost archive file. */ public @Nullable String getInnerEntryName() { final FsEntryName enclEntryName; return this == innerArchive ? ROOT.getPath() : null == (enclEntryName = this.enclEntryName) ? null : enclEntryName.getPath(); } @Nullable FsEntryName getInnerFsEntryName() { return this == innerArchive ? ROOT : enclEntryName; } /** * Returns the enclosing archive file object for this file object. * That is, if this object addresses an entry located within an archive * file, then this method returns the file object representing the * enclosing archive file, or {@code null} otherwise. *

* This method always returns a normalized path, i.e. all occurences of * {@code "."} and {@code ".."} in the path name are removed according to * their meaning wherever possible. *

* In order to support unlimited nesting levels, this method returns * a {@code TFile} instance which may recursively address an entry within * another archive file. * * @return The enclosing archive file in this path. */ public @CheckForNull TFile getEnclArchive() { return enclArchive; } /** * Returns the entry name relative to the enclosing archive file. * That is, if this object addresses an entry located within an archive * file, then this method returns the relative path of the entry in the * enclosing archive file separated by the entry separator character * {@code '/'}, or {@code null} otherwise. *

* This method always returns an undotified path, i.e. all redundant * occurences of {@code "."} and {@code ".."} in the path are removed * wherever possible. * * @return The entry name relative to the enclosing archive file. */ public @Nullable String getEnclEntryName() { return null == enclEntryName ? null : enclEntryName.getPath(); } @Nullable FsEntryName getEnclFsEntryName() { return enclEntryName; } /** * Returns {@code true} if and only if this file is a * {@linkplain #getTopLevelArchive() top level archive file}. * * @return {@code true} if and only if this file is a * {@linkplain #getTopLevelArchive() top level archive file}. * @since TrueZIP 7.5 */ public boolean isTopLevelArchive() { return getTopLevelArchive() == this; } /** * Returns the top level archive file in the path or {@code null} if this * file object does not name an archive file. * A top level archive is not enclosed in another archive. * If this method returns non-{@code null}, the value denotes the longest * part of the path which may (but does not need to) exist as a plain file * in the platform file system. * * @return The top level archive file in the path or {@code null} if this * file object does not name an archive file. */ public @Nullable TFile getTopLevelArchive() { final TFile enclArchive = this.enclArchive; return null != enclArchive ? enclArchive.getTopLevelArchive() : innerArchive; } /** * Returns the decorated file object. *

* If this file instance has been created from a {@link FileSystemView}, * the decorated file object may be an instance of a sibling class, i.e. * another sub-class of {@link File}. * * @return An instance of the {@link File File} class or any of its * sub-classes, but never an instance of this class and never * {@code null}. * @deprecated Using the resulting {@link File} object for file system * access would compete with the TrueZIP Kernel for I/O and may * easily corrupt the state of the (virtual) file system space, * including loss of data! */ @Deprecated public File getFile() { return file; } /** * Returns a file system controller if and only if the path denotes an * archive file, or {@code null} otherwise. *

* TODO: Consider making this public in order to enable applications to * get access to archive entry properties. * * @return A file system controller if and only if the path denotes an * archive file, or {@code null} otherwise. */ @Nullable FsController getController() { final FsController controller = this.controller; if (this != innerArchive || null != controller) return controller; final File file = this.file; final String path = Paths.normalize(file.getPath(), separatorChar); final FsScheme scheme = detector.getScheme(path); // See http://java.net/jira/browse/TRUEZIP-154 . if (null == scheme) throw new ServiceConfigurationError( "Unknown file system scheme for path \"" + path + "\"! Check run-time class path configuration."); final FsMountPoint mountPoint; try { final TFile enclArchive = this.enclArchive; final FsEntryName enclEntryName = this.enclEntryName; assert (null != enclArchive) == (null != enclEntryName); mountPoint = new FsMountPoint(scheme, null == enclArchive ? new FsPath( file) : new FsPath( enclArchive .getController() .getModel() .getMountPoint(), enclEntryName)); } catch (URISyntaxException ex) { throw new AssertionError(ex); } return this.controller = getController(mountPoint); } @SuppressWarnings("deprecation") private FsController getController(FsMountPoint mountPoint) { return TConfig.get().getFsManager().getController(mountPoint, detector); } /** * Returns {@code true} if and only if the path represented * by this instance is a direct or indirect parent of the path * represented by the given {@code file}. *

* Note: *

    *
  • This method uses the absolute path name of the given files. *
  • This method does not access the file system. * It just tests the paths. *
* * @param file The file object for the path to test for being a direct or * indirect child of the path of this instance. * @return {@code true} if and only if the path represented * by this instance is a direct or indirect parent of the path * represented by the given {@code file}. */ public boolean isParentOf(final File file) { final String a = this.getAbsolutePath(); final String b = file.getAbsoluteFile().getParent(); return b != null ? Paths.contains(a, b, separatorChar) : false; } @Override public boolean isAbsolute() { return file.isAbsolute(); } @Override public boolean isHidden() { return file.isHidden(); } /** * Returns {@code true} if and only if the path represented * by this instance contains the path represented by the given {@code file}, * where a path is said to contain another path if and only * if it's equal or an ancestor of the other path. *

* Note: *

    *
  • This method uses the absolute path name of the given files. *
  • This method does not access the file system. * It just tests the paths. *
* * @param file The file object for the path to test for being contained by * the path of this instance. * @return {@code true} if and only if the path represented * by this instance contains the path represented by the given * {@code file} * @throws NullPointerException If the parameter is {@code null}. */ public boolean contains(File file) { return contains(this, file); } /** * Returns {@code true} if and only if the path represented * by {@code a} contains the path represented by {@code b}, * where a path is said to contain another path if and only * if it's equal or an ancestor of the other path. *

* Note: *

    *
  • This method uses the absolute path name of the given files. *
  • This method does not access the file system. * It just tests the paths. *
* * @param a the file to test for containing {@code b}. * @param b the file to test for being contained by {@code a}. * @return {@code true} if and only if the path represented * by {@code a} contains the path represented by {@code b}. * @throws NullPointerException If any parameter is {@code null}. */ public static boolean contains(File a, File b) { return Paths.contains( a.getAbsolutePath(), b.getAbsolutePath(), separatorChar); } /** * Returns {@code true} if and only if this file denotes a file system * root or a UNC (if running on the Windows platform). * * @return {@code true} if and only if this file denotes a file system * root or a UNC (if running on the Windows platform). */ public boolean isFileSystemRoot() { final TFile canOrAbsFile = getCanOrAbsFile(); return ROOTS.contains(canOrAbsFile) || isUNC(canOrAbsFile.getPath()); } /** * Returns {@code true} if and only if this file denotes a UNC. * Note that this should be relevant on the Windows platform only. * * @return {@code true} if and only if this file denotes a UNC. */ public boolean isUNC() { return isUNC(getCanOrAbsPath()); } /** * Returns {@code true} if and only if this file denotes a UNC. * Note that this may be only relevant on the Windows platform. * * @param path a file path. * @return {@code true} if and only if {@code path} denotes a UNC. */ private static boolean isUNC(String path) { return path.startsWith(UNC_PREFIX) && path.indexOf(separatorChar, 2) > 2; } /** * {@inheritDoc} *

* The implementation in the class {@link TFile} delegates the call to its * {@link #getFile() decorated file}. * * @see #equals(Object) */ @Override public int hashCode() { return file.hashCode(); } /** * {@inheritDoc} *

* The implementation in the class {@link TFile} delegates the call to its * {@link #getFile() decorated file}. * This implies that only the hierarchicalized file system path * of this file instance is considered in the comparison. * E.g. {@code new TFile(FsPath.create("zip:file:/archive!/entry"))} and * {@code new TFile(FsPath.create("tar:file:/archive!/entry"))} would * compare equal because their hierarchicalized file system path is * {@code "file:/archive/entry"}. *

* More formally, let {@code a} and {@code b} be two TFile objects. * Then if the expression * {@code a.toFsPath().toHierarchicalUri().equals(b.toFsPath().toHierarchicalUri())} * is true, the expression {@code a.equals(b)} is also true. *

* Note that this does not work vice versa: * E.g. on Windows, the expression * {@code new TFile("file").equals(new TFile("FILE"))} is true, but * {@code new TFile("file").toFsPath().toHierarchicalUri().equals(new TFile("FILE").toFsPath().toHierarchicalUri())} * is false because {@link FsPath#equals(Object)} is case sensitive. * * @param that the object to get compared with this object * @see #hashCode() * @see #compareTo(File) */ @Override @SuppressWarnings({"EqualsWhichDoesntCheckParameterClass"}) public boolean equals(Object that) { return file.equals(that); } /** * {@inheritDoc } *

* The implementation in the class {@link TFile} delegates the call to its * {@link #getFile() decorated file} object. * This implies that only the hierarchicalized file system path * of this file instance is considered in the comparison. * E.g. {@code new TFile(FsPath.create("zip:file:/archive!/entry"))} and * {@code new TFile(FsPath.create("tar:file:/archive!/entry"))} would * compare equal because their hierarchicalized file system path is * {@code "file:/archive/entry"}. *

* More formally, let {@code a} and {@code b} be two TFile objects. * Now if the expression * {@code a.toFsPath().toHierarchicalUri().compareTo(b.toFsPath().toHierarchicalUri()) == 0} * is true, then the expression {@code a.compareTo(b) == 0} is also true. *

* Note that this does not work vice versa: * E.g. on Windows, the expression * {@code new TFile("file").compareTo(new TFile("FILE")) == 0} is true, but * {@code new TFile("file").toFsPath().toHierarchicalUri().compareTo(new TFile("FILE").toFsPath().toHierarchicalUri()) == 0} * is false because {@link FsPath#equals(Object)} is case sensitive. * * @param that the file object to get compared with this object * @see #equals(Object) */ @Override public int compareTo(File that) { return file.compareTo(that); } @Override public String toString() { return file.toString(); } @Deprecated @Override @SuppressWarnings("deprecation") public URL toURL() throws MalformedURLException { return null != innerArchive ? toURI().toURL() : file.toURL(); } /** * In case no prospective archive file has been detected in the path name * at construction time, this method behaves like its super class * implementation. *

* Otherwise, an opaque URI of the form {@code :!/} is * returned, where {@code } is the URI scheme component identifying * a file system driver, {@code } is the URI returned by this method * for the innermost archive file and {@code } is the relative path * name of the the entry addressed by this file object in the innermost * archive file. * If this file object addresses an archive file, then {@code } is the * URI which would have been returned by this method if the file name of * the archive file had not been detected as a prospective archive file * and {@code entry} is an empty string. * *

Example URIs

*

*

    *
  • {@code file:/foo} addresses a regular file
  • *
  • {@code war:file:/foo.war!/} addresses the root entry of the WAR file * addressed by {@code file:/foo.war}.
  • *
  • {@code war:file:/foo.war!/META-INF/MANIFEST.MF} addresses the entry * {@code META-INF/MANIFEST.MF} in the WAR file addressed by * {@code file:/foo.war}.
  • *
  • {@code jar:war:file:/foo.war!/WEB-INF/lib/bar.jar!/META-INF/MANIFEST.MF} * addresses the entry {@code META-INF/MANIFEST.MF} in the JAR file * addressed by {@code war:file:/foo.war!/WEB-INF/lib/bar.jar}, which * recursively addresses the entry {@code WEB-INF/lib/bar.jar} in the * WAR file addressed by {@code file:/foo.war}.
  • *
* * @return A URI for this file object. * @see #TFile(URI) * @see #toFsPath() */ @Override public URI toURI() { try { if (this == innerArchive) { final FsScheme scheme = getScheme(); if (null != enclArchive) { assert null != enclEntryName; return new FsMountPoint( scheme, new FsPath( new FsMountPoint(enclArchive.toURI(), CANONICALIZE), enclEntryName)).toUri(); } else { return new FsMountPoint(scheme, new FsPath(file)).toUri(); } } else if (null != enclArchive) { assert null != enclEntryName; return new FsPath( new FsMountPoint(enclArchive.toURI(), CANONICALIZE), enclEntryName).toUri(); } else { return file.toURI(); } } catch (URISyntaxException ex) { throw new AssertionError(ex); } } /** * Returns a file system path which is consistent with {@link #toURI()}. * * @return A file system path which is consistent with {@link #toURI()}. * @see #TFile(FsPath) * @see #toURI() */ public FsPath toFsPath() { try { if (this == innerArchive) { final FsScheme scheme = getScheme(); if (null != enclArchive) { assert null != enclEntryName; return new FsPath( new FsMountPoint( scheme, new FsPath( new FsMountPoint(enclArchive.toURI(), CANONICALIZE), enclEntryName)), ROOT); } else { return new FsPath( new FsMountPoint(scheme, new FsPath(file)), ROOT); } } else if (null != enclArchive) { assert null != enclEntryName; return new FsPath( new FsMountPoint(enclArchive.toURI(), CANONICALIZE), enclEntryName); } else { return new FsPath(file); } } catch (URISyntaxException ex) { throw new AssertionError(ex); } } private @Nullable FsScheme getScheme() { if (this != innerArchive) return null; final FsController controller = this.controller; if (null != controller) return controller.getModel().getMountPoint().getScheme(); return detector.getScheme(file.getPath()); } /** * The reasons for always throwing an {@link UnsupportedOperationException} * are: *

*

    *
  1. Circular dependency issues: * This method introduces a circular dependency between the * packages {@code java.io} and {@code java.nio.file}. * Circular dependencies are the root of all evil in API design: * They limit reusability and extensibility because you cannot reuse * or exchange a package individually - it's either both or none at * all.
  2. *
  3. Extensibility issues: * Whatever {@link java.nio.file.Path} object would be returned here, * you could not exchange its implementation. * This is because the NIO.2 API lacks a feature to create a * {@code Path} object by looking up an appropriate file * system provider from a plain path string: *
      *
    • The super class implementation of this method always uses the * {@link java.nio.file.FileSystems#getDefault() default file system provider}.
    • *
    • {@link java.nio.file.Paths#get(String, String[])} always uses the * default file system provider, too.
    • *
    * Using {@link URI}s is no alternative, too, because the various URI * schemes provided by the method {@link #toURI()} cannot be made * compatible with the singular URI scheme provided by the method * {@link java.nio.file.spi.FileSystemProvider#getScheme()} of any * NIO.2 {@link java.nio.file.spi.FileSystemProvider} implementation. *
  4. *
  5. Behavior: A typical {@code Path} implementation is greedy, * i.e. when creating another {@code Path} object from an instance of * the implementation class (e.g. by calling * {@link java.nio.file.Path#resolve(String)}), then the returned * object is typically another instance of this implementation class * rather than some other {@code Path} implementation class which may * be required to do I/O on the resulting path. *
*

* As an alternative, you can always create a {@code Path} * object from a {@link File} object {@code file} as follows: *

*

    *
  • Associated with the default file system provider: * {@link java.nio.file.Paths#get(String, String[]) Paths.get(file.getPath())}.
  • *
  • Associated with a TrueZIP file system provider: * {@code new de.schlichtherle.truezip.nio.file.TPath(file)}. * This requires the TrueZIP Path module to be present on the compile * time class path.
  • *
* * @deprecated Throws {@link UnsupportedOperationException}. * @throws UnsupportedOperationException always. */ @Deprecated @Override public java.nio.file.Path toPath() { throw new UnsupportedOperationException("Use a Path constructor or method instead!"); } /** * {@inheritDoc} * * @see
Detecting Archive Paths and False Positives */ @Override public boolean exists() { // DONT test existance of getEnclFsEntryName() in enclArchive because // it doesn't need to exist - see // http://java.net/jira/browse/TRUEZIP-136 . if (null != innerArchive) { try { FsEntry entry = innerArchive.getController() .getEntry(getInnerFsEntryName()); return null != entry; } catch (IOException ex) { return false; } } return file.exists(); } /** * Similar to its super class implementation, but returns * {@code false} for a valid archive file, too. *

* For archive file validation its virtual file system gets mounted. * In case a RAES encrypted ZIP file gets mounted, the user gets prompted * for its password unless the default configuration for key management * hasn't been overridden. * * @see Detecting Archive Paths and False Positives */ @Override public boolean isFile() { if (null != innerArchive) { try { FsEntry entry = innerArchive.getController() .getEntry(getInnerFsEntryName()); return null != entry && entry.isType(FILE); } catch (IOException ex) { return false; } } return file.isFile(); } /** * Similar to its super class implementation, but returns * {@code true} for a valid archive file, too. *

* For archive file validation its virtual file system gets mounted. * In case a RAES encrypted ZIP file gets mounted, the user gets prompted * for its password unless the default configuration for key management * hasn't been overridden. * * @see Detecting Archive Paths and False Positives * @see #isArchive * @see #isEntry */ @Override public boolean isDirectory() { if (null != innerArchive) { try { FsEntry entry = innerArchive.getController() .getEntry(getInnerFsEntryName()); return null != entry && entry.isType(DIRECTORY); } catch (IOException ex) { return false; } } return file.isDirectory(); } /** * If this file refers to an archive file, its archive driver is asked to * return an icon to be displayed for this file. * Otherwise, null is returned. * * @return An icon or {@code null}. * @deprecated GUI features will be removed from this class in TrueZIP 8. */ @Deprecated public @CheckForNull Icon getOpenIcon() { return isArchive() ? ((FsArchiveDriver) getDriver(getScheme())) .getOpenIcon(getController().getModel()) : null; } /** * If this file refers to an archive file, its archive driver is asked to * return an icon to be displayed for this file. * Otherwise, null is returned. * * @return An icon or {@code null}. * @deprecated GUI features will be removed from this class in TrueZIP 8. */ @Deprecated public @CheckForNull Icon getClosedIcon() { return isArchive() ? ((FsArchiveDriver) getDriver(getScheme())) .getClosedIcon(getController().getModel()) : null; } private @Nullable FsDriver getDriver(FsScheme scheme) { return detector.get().get(scheme); } @Override public boolean canRead() { if (null != innerArchive) { try { return innerArchive.getController() .isReadable(getInnerFsEntryName()); } catch (IOException ex) { return false; } } return file.canRead(); } @Override public boolean canWrite() { if (null != innerArchive) { try { return innerArchive.getController() .isWritable(getInnerFsEntryName()); } catch (IOException ex) { return false; } } return file.canWrite(); } @Override public boolean canExecute() { if (null != innerArchive) { try { return innerArchive.getController() .isExecutable(getInnerFsEntryName()); } catch (IOException ex) { return false; } } return file.canExecute(); } /** * Like the super class implementation, but is aware of archive * files in its path. * For entries in a archive file, this is effectively a no-op: * The method will only return {@code true} if the entry isExisting and the * archive file was mounted read only. */ @Override public boolean setReadOnly() { if (null != innerArchive) { try { innerArchive.getController().setReadOnly(getInnerFsEntryName()); return true; } catch (IOException ex) { return false; } } return file.setReadOnly(); } /** * Returns the (uncompressed) length of the file. * The length returned of a valid archive file is {@code 0} in order * to properly emulate virtual directories across all platforms. *

* For archive file validation its virtual file system gets mounted. * In case a RAES encrypted ZIP file gets mounted, the user gets prompted * for its password unless the default configuration for key management * hasn't been overridden. * * @see Detecting Archive Paths and False Positives */ @Override public long length() { if (null != innerArchive) { final FsEntry entry; try { entry = innerArchive.getController().getEntry(getInnerFsEntryName()); } catch (IOException ex) { return 0; } if (null == entry) return 0; final long size = entry.getSize(Size.DATA); return UNKNOWN != size ? size : 0; } return file.length(); } /** * Returns a {@code long} value representing the time this file was * last modified, measured in milliseconds since the epoch (00:00:00 GMT, * January 1, 1970), or {@code 0L} if the file does not exist or if an * I/O error occurs or if this is a ghost directory in an archive file. * * @see Package description for more information * about ghost directories */ @Override public long lastModified() { if (null != innerArchive) { final FsEntry entry; try { entry = innerArchive.getController().getEntry(getInnerFsEntryName()); } catch (IOException ex) { return 0; } if (null == entry) return 0; final long time = entry.getTime(Access.WRITE); return UNKNOWN != time ? time : 0; } return file.lastModified(); } /** * Sets the last modification of this file or (virtual) directory. * If this is a ghost directory within an archive file, it's reincarnated * as a plain directory within the archive file. *

* Note that calling this method may incur a severe performance penalty * if the file is an entry in an archive file which has just been written * (such as after a normal copy operation). * If you want to copy a file's contents as well as its last modification * time, use {@link #cp_p(File, File)} instead. * * @see #cp_p(File, File) * @see Package description for more information * about ghost directories */ @Override public boolean setLastModified(final long time) { if (null != innerArchive) { try { innerArchive.getController().setTime( getInnerFsEntryName(), BitField.of(Access.WRITE), time, TConfig.get().getOutputPreferences()); return true; } catch (IOException ex) { return false; } } return file.setLastModified(time); } /** * Returns the names of the members in this (virtual) directory in a newly * created array. * The returned array is not sorted. * This is the most efficient list method. *

* Note: Archive entries with absolute paths are ignored by * this method and are never returned. * * @return A possibly empty array with the members of this (virtual) * directory or {@code null} if this instance does not refer to a * (virtual) directory or if the virtual directory is inaccessible * due to an I/O failure. */ @Override public @Nullable String[] list() { if (null != innerArchive) { final FsEntry entry; try { entry = innerArchive.getController().getEntry(getInnerFsEntryName()); } catch (IOException ex) { return null; } if (null == entry) return null; final Set members = entry.getMembers(); return null == members ? null : members.toArray(new String[members.size()]); } return file.list(); } /** * Returns the names of the members in this directory which are * accepted by {@code filenameFilter} in a newly created array. * The returned array is not sorted. *

* Note: Archive entries with absolute paths are ignored by * this method and are never returned. * * @return A possibly empty array with the members of this (virtual) * directory or {@code null} if this instance does not refer to a * (virtual) directory or if the virtual directory is inaccessible * due to an I/O failure. */ @Override public @Nullable String[] list(final @CheckForNull FilenameFilter filter) { if (null != innerArchive) { final FsEntry entry; try { entry = innerArchive.getController().getEntry(getInnerFsEntryName()); } catch (IOException ex) { return null; } final Set members = members(entry); if (null == members) return null; if (null == filter) return members.toArray(new String[members.size()]); final Collection accepted = new ArrayList(members.size()); for (final String member : members) if (filter.accept(this, member)) accepted.add(member); return accepted.toArray(new String[accepted.size()]); } return file.list(filter); } /** * Equivalent to {@link #listFiles(FilenameFilter, TArchiveDetector) * listFiles((FilenameFilter) null, getArchiveDetector())}. */ @Override public @Nullable TFile[] listFiles() { return listFiles((FilenameFilter) null, detector); } /** * Returns {@code TFile} objects for the members in this directory * in a newly created array. * The returned array is not sorted. *

* Note that archive entries with absolute paths are ignored by this * method and are never returned. * * @param detector The archive detector to detect any archives files in * the member file names. * @return A possibly empty array with the members of this (virtual) * directory or {@code null} if this instance does not refer to a * (virtual) directory or if the virtual directory is inaccessible * due to an I/O failure. */ @ExpertFeature(value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public @Nullable TFile[] listFiles(TArchiveDetector detector) { return listFiles((FilenameFilter) null, detector); } /** * Equivalent to {@link #listFiles(FilenameFilter, TArchiveDetector) * listFiles(filenameFilter, getArchiveDetector())}. */ @Override public @Nullable TFile[] listFiles(@CheckForNull FilenameFilter filter) { return listFiles(filter, detector); } /** * Returns {@code TFile} objects for the members in this directory * which are accepted by {@code filenameFilter} in a newly created * array. * The returned array is not sorted. *

* Note that archive entries with absolute paths are ignored by this * method and are never returned. * * @param filter the file filter. * @param detector the archive detector to detect any archives files in * the member file names. * @return A possibly empty array with the members of this (virtual) * directory or {@code null} if this instance does not refer to a * (virtual) directory or if the virtual directory is inaccessible * due to an I/O failure. */ @ExpertFeature(value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public @Nullable TFile[] listFiles( final @CheckForNull FilenameFilter filter, final TArchiveDetector detector) { if (null != innerArchive) { final FsEntry entry; try { entry = innerArchive.getController().getEntry(getInnerFsEntryName()); } catch (IOException ex) { return null; } return filter(members(entry), filter, detector); } else { return filter(list(file.list(filter)), (FilenameFilter) null, detector); } } private static @CheckForNull Set members(@CheckForNull FsEntry entry) { return null == entry ? null : entry.getMembers(); } private static @CheckForNull List list(@CheckForNull String[] list) { return null == list ? null : Arrays.asList(list); } private @Nullable TFile[] filter( final @CheckForNull Collection members, final @CheckForNull FilenameFilter filter, final TArchiveDetector detector) { if (null == members) return null; if (null != filter) { final Collection accepted = new ArrayList(members.size()); for (final String member : members) if (filter.accept(this, member)) accepted.add(new TFile(this, member, detector)); return accepted.toArray(new TFile[accepted.size()]); } else { final TFile[] accepted = new TFile[members.size()]; int i = 0; for (final String member : members) accepted[i++] = new TFile(this, member, detector); return accepted; } } /** * Equivalent to {@link #listFiles(FileFilter, TArchiveDetector) * listFiles(fileFilter, getArchiveDetector())}. */ @Override public @Nullable TFile[] listFiles(@CheckForNull FileFilter filter) { return listFiles(filter, detector); } /** * Returns {@code TFile} objects for the members in this directory * which are accepted by {@code fileFilter} in a newly created array. * The returned array is not sorted. *

* Note that archive entries with absolute paths are ignored by this * method and are never returned. * * @param filter the file filter. * @param detector The archive detector to detect any archives files in * the member file names. * @return A possibly empty array with the members of this (virtual) * directory or {@code null} if this instance does not refer to a * (virtual) directory or if the virtual directory is inaccessible * due to an I/O failure. */ @ExpertFeature(value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public @Nullable TFile[] listFiles( final @CheckForNull FileFilter filter, final TArchiveDetector detector) { if (null != innerArchive) { final FsEntry entry; try { entry = innerArchive.getController().getEntry(getInnerFsEntryName()); } catch (IOException ex) { return null; } return filter(members(entry), filter, detector); } else { return filter(list(file.list()), filter, detector); } } private @Nullable TFile[] filter( final @CheckForNull Collection members, final @CheckForNull FileFilter filter, final TArchiveDetector detector) { if (null == members) return null; if (null != filter) { final Collection accepted = new ArrayList(members.size()); for (final String member : members) { final TFile file = new TFile(this, member, detector); if (filter.accept(file)) accepted.add(file); } return accepted.toArray(new TFile[accepted.size()]); } else { final TFile[] accepted = new TFile[members.size()]; int i = 0; for (final String member : members) accepted[i++] = new TFile(this, member, detector); return accepted; } } /** * Creates a new, empty file similar to its superclass implementation. * Note that this method doesn't create archive files because archive * files are virtual directories, not files! * * @see #mkdir */ @Override public boolean createNewFile() throws IOException { if (null != innerArchive) { final FsController controller = innerArchive.getController(); final FsEntryName entryName = getInnerFsEntryName(); // FIXME: This is not properly isolated. if (null != controller.getEntry(entryName)) return false; controller.mknod( entryName, FILE, TConfig.get().getOutputPreferences().set(EXCLUSIVE), null); return true; } return file.createNewFile(); } @Override public boolean mkdirs() { if (null == innerArchive) return file.mkdirs(); final TFile parent = getParentFile(); if (null != parent && !parent.exists()) parent.mkdirs(); // TODO: Profile: return parent.isDirectory() && mkdir(); // May perform better in certain situations where (probably false // positive) archive files are involved. return mkdir(); } /** * Creates a new, empty (virtual) directory similar to its superclass * implementation. * This method creates an archive file if {@link #isArchive} returns * {@code true}. * Example: * {@code new TFile("archive.zip").mkdir();} *

* Alternatively, archive files get created automatically by simply * creating their entries. * Example: * {@code new TFileOutputStream("archive.zip/README");} * This assumes the default configuration where * {@link TConfig#isLenient TConfig.get().isLenient()} is true. */ @Override public boolean mkdir() { if (null != innerArchive) { try { innerArchive.getController().mknod( getInnerFsEntryName(), DIRECTORY, TConfig.get().getOutputPreferences(), null); return true; } catch (IOException ex) { return false; } } return file.mkdir(); } /** * Ensures that a (virtual) directory with {@link #getPath() this path name} * exists in the (federated) file system. * * @param recursive whether or not any missing ancestor directories shall * get created if required. * @return {@code this} * @throws IOException if any I/O error occurs. */ public TFile mkdir(final boolean recursive) throws IOException { final TFile innerArchive = this.innerArchive; if (null != innerArchive) { if (recursive) { final TFile parent = getParentFile(); if (null != parent && !parent.exists()) parent.mkdir(recursive); } final FsController controller = innerArchive.getController(); final FsEntryName innerEntryName = getInnerFsEntryName(); try { controller.mknod( innerEntryName, DIRECTORY, TConfig.get().getOutputPreferences(), null); } catch (IOException ex) { final FsEntry entry = controller.getEntry(innerEntryName); if (null == entry || !entry.isType(DIRECTORY)) throw ex; } } else { final File dir = file; if (!(recursive ? dir.mkdirs() : dir.mkdir()) && !dir.isDirectory()) throw new IOException(dir + " (cannot create directory)"); } return this; } /** * {@inheritDoc} * * @deprecated This method just returns a boolean value to indicate failure, * which is hard to analyze. * @see #rm() */ @Deprecated @Override public boolean delete() { try { rm(this); return true; } catch (IOException ex) { return false; } } /** * Equivalent to {@link #rm(File) rm(this)}. * * @return {@code this} * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods */ public TFile rm() throws IOException { rm(this); return this; } /** * Deletes the given file or directory. * If the file is a directory, it must be empty. * * @param node the file or directory. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods */ public static void rm(File node) throws IOException { if (node instanceof TFile) { TFile file = (TFile) node; if (null != file.innerArchive) { file.innerArchive.getController().unlink( file.getInnerFsEntryName(), TConfig.get().getOutputPreferences()); return; } node = file.file; } if (!node.delete()) throw new IOException(node + " (cannot delete)"); } /** * Equivalent to {@link #rm_r(File) rm_r(this)}. * * @return {@code this} * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods * @see Traversing Directory Trees */ public TFile rm_r() throws IOException { TBIO.rm_r(this, detector); return this; } /** * Recursively deletes the given file or directory tree. *

* If {@code node} is an instance of this * class, its {@link #getArchiveDetector() archive detector} * is used to detect prospective archive files in the directory tree. * Otherwise, * {@link TArchiveDetector#NULL} * is used to detect prospective archive files in the directory tree. *

* This file system operation is not atomic. * * @param node the file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods * @see Traversing Directory Trees */ public static void rm_r(File node) throws IOException { TBIO.rm_r(node, node instanceof TFile ? ((TFile) node).detector : TArchiveDetector.NULL); } @Override public void deleteOnExit() { if (innerArchive != null) { // Support for this operation for archive files and entries has been // removed in TrueZIP 7 because using a shutdown hook uncautiously // introduces a memory leak in dynamic class loader environments. throw new UnsupportedOperationException(); } file.deleteOnExit(); } /** * {@inheritDoc} * * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @return {@code true} on success or {@code false} otherwise. * @deprecated This method just returns a boolean value to indicate failure, * which is hard to analyze. * @see #mv(File) */ @Deprecated @Override public boolean renameTo(final File dst) { try { mv(this, dst, detector); return true; } catch (IOException ex) { return false; } } /** * Equivalent to {@link #mv(File, File, TArchiveDetector) mv(this, dst, getArchiveDetector())}. * * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @return {@code this} * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods * @see Traversing Directory Trees */ public TFile mv(File dst) throws IOException { mv(this, dst, detector); return this; } /** * Moves the given source file or directory to the given destination file * or directory. *

* In certain cases, this method might perform a recursive copy-then-delete * operation rather than an atomic move operation. * In these cases, an attempt is made to copy all attributes of each * source file to the destination file, too. * Which attributes are actually copied is specific to the destination * file system driver implementation, but the minimum guarantee is to * copy the last modification time. * * @param src the source file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param detector the archive detector to use for detecting any archive * files in the source directory tree. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods * @see Traversing Directory Trees */ @ExpertFeature( level=INTERMEDIATE, value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public static void mv( final File src, final File dst, final TArchiveDetector detector) throws IOException { final boolean srcArchived; final File srcDelegate; if (src instanceof TFile) { final TFile srcFile = (TFile) src; srcArchived = null != srcFile.getInnerArchive(); srcDelegate = srcFile.getFile(); } else { srcArchived = false; srcDelegate = src; } final boolean dstArchived; final File dstDelegate; if (dst instanceof TFile) { final TFile dstFile = (TFile) dst; dstArchived = null != dstFile.getInnerArchive(); dstDelegate = dstFile.getFile(); } else { dstArchived = false; dstDelegate = dst; } if (!srcArchived && !dstArchived) { if (srcDelegate.renameTo(dstDelegate)) return; throw new IOException(src + " (cannot rename to " + dst + ")"); } TBIO.mv(src, dst, detector); } /** * Copies the data from the input stream {@code in} to the output stream * {@code out} and closes both streams - even if an exception occurs. *

* This is a high performance implementation which uses a pooled background * thread to fill a FIFO of pooled buffers which is concurrently flushed by * the current thread. * It performs best when used with unbuffered streams. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesn/a
Copies directories recursivelyn/a
Reads and overwrites special filesn/a
Closes parameter stream(s)Always
Raw Data Copying (RDC)n/a
Deletes partial written files on failuren/a
Deletes partial written directories on failuren/a
AtomicNo
* * @param in the input stream. * @param out the output stream. * @throws IOException if any I/O error occurs. * @see #cat(InputStream, OutputStream) * @see Bulk I/O Methods */ public static void cp( final @WillClose InputStream in, final @WillClose OutputStream out) throws IOException { Streams.copy(in, out); } /** * Copies the input stream {@code in} to the file {@code dst} and * closes the stream - even if an exception occurs. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesNone
Copies directories recursivelyNo
Reads and overwrites special filesYes
Closes parameter stream(s)Always
Raw Data Copying (RDC)No
Deletes partial written files on failureYes
Deletes partial written directories on failuren/a
AtomicNo
* * @param in the input stream. * @param dst the destination file. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods */ @edu.umd.cs.findbugs.annotations.SuppressWarnings("OBL_UNSATISFIED_OBLIGATION") public static void cp(final @WillClose InputStream in, final File dst) throws IOException { if (null == in) throw new NullPointerException(); @WillClose TFileOutputStream out = null; try { out = new TFileOutputStream(dst); } finally { if (null == out) // exception? in.close(); } try { cp(in, out); } catch (final IOException ex) { rm(dst); throw ex; } } /** * Copies the file {@code src} to the output stream {@code out} and * closes the stream - even if an exception occurs. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesNone
Copies directories recursivelyNo
Reads and overwrites special filesYes
Closes parameter stream(s)Always
Raw Data Copying (RDC)No
Deletes partial written files on failuren/a
Deletes partial written directories on failuren/a
AtomicNo
* * @param src the source file. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param out the output stream. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods */ public static void cp(final File src, final @WillClose OutputStream out) throws IOException { if (null == out) throw new NullPointerException(); @WillClose TFileInputStream in = null; try { in = new TFileInputStream(src); } finally { if (null == in) // exception? out.close(); } cp(in, out); } /** * Equivalent to {@link #cp(File, File) cp(this, dst)}. * * @param dst the destination file. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @return {@code this} * @throws IOException if any I/O error occurs. */ public TFile cp(File dst) throws IOException { TBIO.cp(false, this, dst); return this; } /** * Copies the file {@code src} to the file {@code dst}. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesNone
Copies directories recursivelyNo
Reads and overwrites special filesYes
Closes parameter stream(s)n/a
Raw Data Copying (RDC)Yes
Deletes partial written files on failureYes
Deletes partial written directories on failuren/a
AtomicNo
* * @param src the source file. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param dst the destination file. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods */ public static void cp(File src, File dst) throws IOException { TBIO.cp(false, src, dst); } /** * Equivalent to {@link #cp_p(File, File) cp_p(this, dst)}. * * @param dst the destination file. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @return {@code this} * @throws IOException if any I/O error occurs. */ public TFile cp_p(File dst) throws IOException { TBIO.cp(true, this, dst); return this; } /** * Copies the file {@code src} to the file {@code dst} and attempts to * copy all attributes of the source file to the destination file, too. * Which attributes are actually copied is specific to the source and * destination file system driver implementations, but the minimum * guarantee is to copy the last modification time. * For example, starting with TrueZIP 7.2, the last modification, last * access and creation times are copied if all of the following are true: *

*

    *
  1. Both parameters refer to the platform file system * (even if any one is only a {@code java.io.File}), and *
  2. the JVM complies to JSE 7, and *
  3. the TrueZIP Driver FILE module is used. *
*

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesBest effort
Copies directories recursivelyNo
Reads and overwrites special filesYes
Closes parameter stream(s)n/a
Raw Data Copying (RDC)Yes
Deletes partial written files on failureYes
Deletes partial written directories on failuren/a
AtomicNo
* * @param src the source file. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param dst the destination file. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods */ public static void cp_p(File src, File dst) throws IOException { TBIO.cp(true, src, dst); } /** * Equivalent to {@link #cp_r(File, File, TArchiveDetector, TArchiveDetector) cp_r(this, dst, getArchiveDetector(), getArchiveDetector())}. * * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @return {@code this} * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods * @see Traversing Directory Trees */ public TFile cp_r(File dst) throws IOException { TBIO.cp_r(false, this, dst, detector, detector); return this; } /** * Equivalent to {@link #cp_r(File, File, TArchiveDetector, TArchiveDetector) cp_r(this, dst, detector, detector)}. * * @param src the source file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param detector the archive detector to use for detecting any * archive files within the source and destination * directory tree. * @throws IOException if any I/O error occurs. * @since TrueZIP 7.2 * @see Bulk I/O Methods * @see Traversing Directory Trees */ @ExpertFeature( level=INTERMEDIATE, value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public static void cp_r(File src, File dst, TArchiveDetector detector) throws IOException { TBIO.cp_r(false, src, dst, detector, detector); } /** * Recursively copies the file or directory {@code src} * to the file or directory {@code dst}. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesNone
Copies directories recursivelyYes
Reads and overwrites special filesNo
Closes parameter stream(s)n/a
Raw Data Copying (RDC)Yes
Deletes partial written files on failureYes
Deletes partial written directories on failureNo
AtomicNo
* * @param src the source file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param srcDetector the archive detector to use for detecting any * archive files within the source directory tree. * @param dstDetector the archive detector to use for detecting any * archive files within the destination directory tree. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods * @see Traversing Directory Trees */ @ExpertFeature( level=INTERMEDIATE, value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public static void cp_r(File src, File dst, TArchiveDetector srcDetector, TArchiveDetector dstDetector) throws IOException { TBIO.cp_r(false, src, dst, srcDetector, dstDetector); } /** * Equivalent to {@link #cp_rp(File, File, TArchiveDetector, TArchiveDetector) cp_rp(this, dst, getArchiveDetector(), getArchiveDetector())}. * * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @return {@code this} * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods * @see Traversing Directory Trees */ public TFile cp_rp(File dst) throws IOException { TBIO.cp_r(true, this, dst, detector, detector); return this; } /** * Equivalent to {@link #cp_rp(File, File, TArchiveDetector, TArchiveDetector) cp_rp(this, dst, detector, detector)}. * * @param src the source file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param detector the archive detector to use for detecting any * archive files within the source and destination * directory tree. * @throws IOException if any I/O error occurs. * @since TrueZIP 7.2 * @see Bulk I/O Methods * @see Traversing Directory Trees */ @ExpertFeature( level=INTERMEDIATE, value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public static void cp_rp(File src, File dst, TArchiveDetector detector) throws IOException { TBIO.cp_r(true, src, dst, detector, detector); } /** * Recursively copies the file or directory {@code src} to the file or * directory {@code dst} and attempts to copy all attributes of each * source file to the destination file, too. * Which attributes are actually copied is specific to the source and * destination file system driver implementations, but the minimum * guarantee is to copy the last modification time - except for archive * files. * For example, starting with TrueZIP 7.2, the last modification, last * access and creation times are copied if all of the following are true: *

*

    *
  1. Both parameters refer to the platform file system * (even if any one is only a {@code java.io.File}), and *
  2. the JVM complies to JSE 7, and *
  3. the TrueZIP Driver FILE module is used. *
*

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesBest effort
Copies directories recursivelyYes
Reads and overwrites special filesNo
Closes parameter stream(s)n/a
Raw Data Copying (RDC)Yes
Deletes partial written files on failureYes
Deletes partial written directories on failureNo
AtomicNo
* * @param src the source file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param dst the destination file or directory tree. * Note that although this just needs to be a plain {@code File}, * archive files and entries are only supported for instances of * this class. * @param srcDetector the archive detector to use for detecting any * archive files within the source directory tree. * @param dstDetector the archive detector to use for detecting any * archive files within the destination directory tree. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods * @see Traversing Directory Trees */ @ExpertFeature( level=INTERMEDIATE, value=INJECTING_A_DIFFERENT_DETECTOR_FOR_THE_SAME_PATH_MAY_CORRUPT_DATA) public static void cp_rp(File src, File dst, TArchiveDetector srcDetector, TArchiveDetector dstDetector) throws IOException { TBIO.cp_r(true, src, dst, srcDetector, dstDetector); } /** * Copies the input stream {@code in} to this file or entry in an archive * file * without closing the stream. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesn/a
Copies directories recursivelyn/a
Reads and overwrites special filesYes
Closes parameter stream(s)Never
Raw Data Copying (RDC)n/a
Deletes partial written files on failureYes
Deletes partial written directories on failuren/a
AtomicNo
* * @param in the input stream. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods */ public void input(final @WillNotClose InputStream in) throws IOException { if (null == in) throw new NullPointerException(); try { final @WillClose TFileOutputStream out = new TFileOutputStream(this); try { Streams.cat(in, out); } finally { out.close(); } } catch (final IOException ex) { rm(this); throw ex; } } /** * Copies this file or entry in an archive file to the output stream * {@code out} * without closing the stream. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesn/a
Copies directories recursivelyn/a
Reads and overwrites special filesYes
Closes parameter stream(s)Never
Raw Data Copying (RDC)n/a
Deletes partial written files on failuren/a
Deletes partial written directories on failuren/a
AtomicNo
* * @param out the output stream. * @throws IOException if any I/O error occurs. * @see Bulk I/O Methods */ public void output(final @WillNotClose OutputStream out) throws IOException { if (null == out) throw new NullPointerException(); final @WillClose TFileInputStream in = new TFileInputStream(this); try { Streams.cat(in, out); } finally { in.close(); } } /** * Copies the data from the given input stream to the given output stream * without closing them. * The name of this method is inspired by the Unix command line utility * {@code cat} because you could use it to concatenate the * contents of multiple streams. *

* This is a high performance implementation which uses a pooled background * thread to fill a FIFO of data buffers which is concurrently flushed by * the current thread. * It performs best when used with unbuffered streams. *

*

* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * *
FeatureSupported
Preserves file attributesn/a
Copies directories recursivelyn/a
Reads and overwrites special filesn/a
Closes parameter stream(s)Never
Raw Data Copying (RDC)n/a
Deletes partial written files on failuren/a
Deletes partial written directories on failuren/a
AtomicNo
* * @param in the input stream. * @param out the output stream. * @throws IOException if any I/O error occurs. * @see #cp(InputStream, OutputStream) * @see Bulk I/O Methods */ public static void cat( final @WillNotClose InputStream in, final @WillNotClose OutputStream out) throws IOException { Streams.cat(in, out); } /** * Compacts this archive file by removing any redundant archive entry * contents and meta data, including central directories. * If this file isn't a * {@linkplain #isTopLevelArchive() top level archive file}, * then this operation does nothing and returns immediately. *

* This operation is intended to compact archive files which have been * frequently updated with {@link FsOutputOption#GROW} or similar means. * If this output option preference is set and an archive file is updated * frequently, then over time a lot of redundant artifacts such as archive * entry contents and meta data, including central directories may be * physically present in the archive file, even if all its entries have * been deleted. * This operation could then get used to remove any redundant artifacts * again. *

* Mind that this operation has no means to detect if there is actually any * redundant data present in this archive file. * Any invocation will perform exactly the same steps, so if this archive * file is already compact, then this will just waste time and temporary * space in the platform file system. *

* Note that this operation is not thread-safe and hence not atomic, so you * should not concurrently access this archive file or any of its entries! *

* This operation performs in the order of O(s), where s is * the total getSize of the archive file either before (worst case) or after * (best case) compacting it. * If this archive file has already been mounted, then s is the * total getSize of the archive file after compacting it (best case). * Otherwise, the definition of s is specific to the archive file * system driver. * Usually, if the archive file contains a central directory, you could * expect the best case, otherwise the worst case, but this information * is given without warranty. *

* If this archive file has been successfully compacted, then it's left * unmounted, so any subsequent operation will mount it again, which * requires additional time. * * @return this * @throws IOException On any I/O error. * @see FsOutputOption#GROW * @since TrueZIP 7.3 */ public TFile compact() throws IOException { // See http://java.net/jira/browse/TRUEZIP-205 . if (isTopLevelArchive()) compact(this); return this; } private static void compact(TFile grown) throws IOException { assert grown.isArchive(); grown = grown.getNormalizedFile(); assert grown.isArchive(); final File dir = getParent(grown); final String suffix = getSuffix(grown); final TConfig config = TConfig.push(); try { // Switch off FsOutputOption.GROW. config.setOutputPreferences( config.getOutputPreferences().clear(GROW)); // Create temp file. final TFile compact = new TFile(createTempFile("tzp", suffix, dir)); compact.rm(); try { // Make a structural copy of the grown archive file, thereby // compacting it. grown.cp_rp(compact); // Unmount both archive files so we can delete and move them // safely and fast like regular files. TVFS.umount(grown); TVFS.umount(compact); // Move the compacted archive file over to the grown archive // file like a regular file. if (!move(compact.toNonArchiveFile(), grown.toNonArchiveFile())) throw new IOException(compact + " (cannot move to " + grown + ")"); } catch (final IOException ex) { try { compact.rm(); } catch (final IOException ex2) { if (JSE7.AVAILABLE) ex.addSuppressed(ex2); } throw ex; } } finally { config.close(); } } private static File getParent(final File file) { final File parent = file.getParentFile(); return null != parent ? parent : CURRENT_DIRECTORY; } private static @Nullable String getSuffix(final TFile file) { final FsScheme scheme = file.getScheme(); return null != scheme ? "." + scheme : null; } private static boolean move(File src, File dst) { return src.exists() && (!dst.exists() || dst.delete()) && src.renameTo(dst); } }





© 2015 - 2025 Weber Informatics LLC | Privacy Policy