com.alibaba.antx.util.ZipScanner Maven / Gradle / Ivy
/*
* Copyright (c) 2002-2012 Alibaba Group Holding Limited.
* All rights reserved.
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software
* distributed under the License is distributed on an "AS IS" BASIS,
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
* See the License for the specific language governing permissions and
* limitations under the License.
*/
package com.alibaba.antx.util;
import java.io.BufferedInputStream;
import java.io.File;
import java.io.IOException;
import java.io.InputStream;
import java.net.MalformedURLException;
import java.net.URL;
import java.util.ArrayList;
import java.util.List;
import java.util.zip.ZipEntry;
import java.util.zip.ZipInputStream;
public class ZipScanner {
/** The base directory to be scanned. */
protected URL zipURL;
/** The patterns for the files to be included. */
protected String[] includes;
/** The patterns for the files to be excluded. */
protected String[] excludes;
/**
* The files which matched at least one include and no excludes and were
* selected.
*/
protected List filesIncluded;
/** The files which did not match any includes or selectors. */
protected List filesNotIncluded;
/** The files which matched at least one include and at least one exclude. */
protected List filesExcluded;
/**
* The directories which matched at least one include and no excludes and
* were selected.
*/
protected List dirsIncluded;
/** The directories which were found and did not match any includes. */
protected List dirsNotIncluded;
/**
* The directories which matched at least one include and at least one
* exclude.
*/
protected List dirsExcluded;
/** Whether or not the file system should be treated as a case sensitive one. */
protected boolean isCaseSensitive = true;
/** Whether or not everything tested so far has been included. */
protected boolean everythingIncluded = true;
/** Sole constructor. */
public ZipScanner() {
}
/**
* Tests whether or not a given path matches the start of a given pattern up
* to the first "".
*
* This is not a general purpose test and should only be used if you can
* live with false positives. For example, pattern=\a
and
* str=b
will yield true
.
*
*
* @param pattern The pattern to match against. Must not be
* null
.
* @param str The path to match, as a String. Must not be null
.
* @return whether or not a given path matches the start of a given pattern
* up to the first "".
*/
protected static boolean matchPatternStart(String pattern, String str) {
return SelectorUtil.matchPatternStart(pattern, str);
}
/**
* Tests whether or not a given path matches the start of a given pattern up
* to the first "".
*
* This is not a general purpose test and should only be used if you can
* live with false positives. For example, pattern=\a
and
* str=b
will yield true
.
*
*
* @param pattern The pattern to match against. Must not be
* null
.
* @param str The path to match, as a String. Must not be null
.
* @param isCaseSensitive Whether or not matching should be performed case
* sensitively.
* @return whether or not a given path matches the start of a given pattern
* up to the first "".
*/
protected static boolean matchPatternStart(String pattern, String str, boolean isCaseSensitive) {
return SelectorUtil.matchPatternStart(pattern, str, isCaseSensitive);
}
/**
* Tests whether or not a given path matches a given pattern.
*
* @param pattern The pattern to match against. Must not be
* null
.
* @param str The path to match, as a String. Must not be null
.
* @return true
if the pattern matches against the string, or
* false
otherwise.
*/
protected static boolean matchPath(String pattern, String str) {
return SelectorUtil.matchPath(pattern, str);
}
/**
* Tests whether or not a given path matches a given pattern.
*
* @param pattern The pattern to match against. Must not be
* null
.
* @param str The path to match, as a String. Must not be null
.
* @param isCaseSensitive Whether or not matching should be performed case
* sensitively.
* @return true
if the pattern matches against the string, or
* false
otherwise.
*/
protected static boolean matchPath(String pattern, String str, boolean isCaseSensitive) {
return SelectorUtil.matchPath(pattern, str, isCaseSensitive);
}
/**
* Tests whether or not a string matches against a pattern. The pattern may
* contain two special characters:
* '' means zero or more characters
* '?' means one and only one character
*
* @param pattern The pattern to match against. Must not be
* null
.
* @param str The string which must be matched against the pattern. Must not
* be null
.
* @return true
if the string matches against the pattern, or
* false
otherwise.
*/
public static boolean match(String pattern, String str) {
return SelectorUtil.match(pattern, str);
}
/**
* Tests whether or not a string matches against a pattern. The pattern may
* contain two special characters:
* '' means zero or more characters
* '?' means one and only one character
*
* @param pattern The pattern to match against. Must not be
* null
.
* @param str The string which must be matched against the pattern. Must not
* be null
.
* @param isCaseSensitive Whether or not matching should be performed case
* sensitively.
* @return true
if the string matches against the pattern, or
* false
otherwise.
*/
protected static boolean match(String pattern, String str, boolean isCaseSensitive) {
return SelectorUtil.match(pattern, str, isCaseSensitive);
}
/**
* Sets the base directory to be scanned. This is the directory which is
* scanned recursively. All '/' and '\' characters are replaced by
* File.separatorChar
, so the separator used need not match
* File.separatorChar
.
*
* @param basedir The base directory to scan. Must not be null
.
*/
public void setZipURL(String zipURL) throws MalformedURLException {
setSrc(new URL(zipURL));
}
/**
* Sets the base directory to be scanned. This is the directory which is
* scanned recursively.
*
* @param basedir The base directory for scanning. Should not be
* null
.
*/
public void setSrc(URL zipURL) {
this.zipURL = zipURL;
}
/**
* Returns the base directory to be scanned. This is the directory which is
* scanned recursively.
*
* @return the base directory to be scanned
*/
public URL getZipURL() {
return zipURL;
}
/**
* Sets whether or not the file system should be regarded as case sensitive.
*
* @param isCaseSensitive whether or not the file system should be regarded
* as a case sensitive one
*/
public void setCaseSensitive(boolean isCaseSensitive) {
this.isCaseSensitive = isCaseSensitive;
}
/**
* Sets the list of include patterns to use. All '/' and '\' characters are
* replaced by File.separatorChar
, so the separator used need
* not match File.separatorChar
.
*
* When a pattern ends with a '/' or '\', "" is appended.
*
*
* @param includes A list of include patterns. May be null
,
* indicating that all files should be included. If a non-
* null
list is given, all elements must be non-
* null
.
*/
public void setIncludes(String[] includes) {
if (includes == null) {
this.includes = null;
} else {
this.includes = new String[includes.length];
for (int i = 0; i < includes.length; i++) {
String pattern;
pattern = includes[i].replace('\\', '/');
if (pattern.endsWith(File.separator)) {
pattern += "**";
}
this.includes[i] = pattern;
}
}
}
/**
* Sets the list of exclude patterns to use. All '/' and '\' characters are
* replaced by File.separatorChar
, so the separator used need
* not match File.separatorChar
.
*
* When a pattern ends with a '/' or '\', "" is appended.
*
*
* @param excludes A list of exclude patterns. May be null
,
* indicating that no files should be excluded. If a non-
* null
list is given, all elements must be non-
* null
.
*/
public void setExcludes(String[] excludes) {
if (excludes == null) {
this.excludes = null;
} else {
this.excludes = new String[excludes.length];
for (int i = 0; i < excludes.length; i++) {
String pattern;
pattern = excludes[i].replace('\\', '/');
if (pattern.endsWith(File.separator)) {
pattern += "**";
}
this.excludes[i] = pattern;
}
}
}
/**
* Returns whether or not the scanner has included all the files or
* directories it has come across so far.
*
* @return true
if all files and directories which have been
* found so far have been included.
*/
public boolean isEverythingIncluded() {
return everythingIncluded;
}
/**
* Scans the base directory for files which match at least one include
* pattern and don't match any exclude patterns. If there are selectors then
* the files must pass muster there, as well.
*
* @throws IllegalStateException if the base directory was set
* incorrectly (i.e. if it is null
, doesn't
* exist, or isn't a directory).
*/
public void scan() throws IllegalStateException, IOException {
if (zipURL == null) {
throw new IllegalStateException("No zipURL set");
}
if (includes == null) {
// No includes supplied, so set it to 'matches all'
includes = new String[1];
includes[0] = "**";
}
if (excludes == null) {
excludes = new String[0];
}
filesIncluded = new ArrayList();
filesNotIncluded = new ArrayList();
filesExcluded = new ArrayList();
dirsIncluded = new ArrayList();
dirsNotIncluded = new ArrayList();
dirsExcluded = new ArrayList();
if (isIncluded("")) {
if (!isExcluded("")) {
dirsIncluded.add("");
} else {
dirsExcluded.add("");
}
} else {
dirsNotIncluded.add("");
}
scanZipFile(zipURL);
}
/**
* Scans the given directory for files and directories. Found files and
* directories are placed in their respective collections, based on the
* matching of includes, excludes, and the selectors. When a directory is
* found, it is scanned recursively.
*
* @param dir The directory to scan. Must not be null
.
* @param vpath The path relative to the base directory (needed to prevent
* problems with an absolute path when using dir). Must not be
* null
.
* @param fast Whether or not this call is part of a fast scan.
* @see #filesIncluded
* @see #filesNotIncluded
* @see #filesExcluded
* @see #dirsIncluded
* @see #dirsNotIncluded
* @see #dirsExcluded
* @see #slowScan
*/
protected void scanZipFile(URL zipURL) throws IOException {
InputStream istream = null;
ZipInputStream zipStream = null;
try {
istream = zipURL.openStream();
if (!(istream instanceof BufferedInputStream)) {
istream = new BufferedInputStream(istream, 8192);
}
zipStream = new ZipInputStream(istream);
ZipEntry entry;
while ((entry = zipStream.getNextEntry()) != null) {
String name = entry.getName().replace('\\', '/');
if (name.endsWith(File.separator)) {
name = name.substring(0, name.length() - 1);
}
if (entry.isDirectory()) {
if (isIncluded(name)) {
if (!isExcluded(name)) {
dirsIncluded.add(name);
} else {
everythingIncluded = false;
dirsExcluded.add(name);
}
} else {
everythingIncluded = false;
dirsNotIncluded.add(name);
}
} else {
if (isIncluded(name)) {
if (!isExcluded(name)) {
filesIncluded.add(name);
} else {
everythingIncluded = false;
filesExcluded.add(name);
}
} else {
everythingIncluded = false;
filesNotIncluded.add(name);
}
}
}
} finally {
if (zipStream != null) {
try {
zipStream.close();
} catch (IOException e) {
}
}
if (istream != null) {
try {
istream.close();
} catch (IOException e) {
}
}
}
}
/**
* Tests whether or not a name matches against at least one include pattern.
*
* @param name The name to match. Must not be null
.
* @return true
when the name matches against at least one
* include pattern, or false
otherwise.
*/
protected boolean isIncluded(String name) {
for (String include : includes) {
if (matchPath(include, name, isCaseSensitive)) {
return true;
}
}
return false;
}
/**
* Tests whether or not a name matches against at least one exclude pattern.
*
* @param name The name to match. Must not be null
.
* @return true
when the name matches against at least one
* exclude pattern, or false
otherwise.
*/
protected boolean isExcluded(String name) {
for (String exclude : excludes) {
if (matchPath(exclude, name, isCaseSensitive)) {
return true;
}
}
return false;
}
/**
* Returns the names of the files which matched at least one of the include
* patterns and none of the exclude patterns. The names are relative to the
* base directory.
*
* @return the names of the files which matched at least one of the include
* patterns and none of the exclude patterns.
*/
public String[] getIncludedFiles() {
return (String[]) filesIncluded.toArray(new String[filesIncluded.size()]);
}
/**
* Returns the names of the files which matched none of the include
* patterns. The names are relative to the base directory. This involves
* performing a slow scan if one has not already been completed.
*
* @return the names of the files which matched none of the include
* patterns.
* @see #slowScan
*/
public String[] getNotIncludedFiles() {
return (String[]) filesNotIncluded.toArray(new String[filesNotIncluded.size()]);
}
/**
* Returns the names of the files which matched at least one of the include
* patterns and at least one of the exclude patterns. The names are relative
* to the base directory. This involves performing a slow scan if one has
* not already been completed.
*
* @return the names of the files which matched at least one of the include
* patterns and at at least one of the exclude patterns.
* @see #slowScan
*/
public String[] getExcludedFiles() {
return (String[]) filesExcluded.toArray(new String[filesExcluded.size()]);
}
/**
* Returns the names of the directories which matched at least one of the
* include patterns and none of the exclude patterns. The names are relative
* to the base directory.
*
* @return the names of the directories which matched at least one of the
* include patterns and none of the exclude patterns.
*/
public String[] getIncludedDirectories() {
return (String[]) dirsIncluded.toArray(new String[dirsIncluded.size()]);
}
/**
* Returns the names of the directories which matched none of the include
* patterns. The names are relative to the base directory. This involves
* performing a slow scan if one has not already been completed.
*
* @return the names of the directories which matched none of the include
* patterns.
* @see #slowScan
*/
public String[] getNotIncludedDirectories() {
return (String[]) dirsNotIncluded.toArray(new String[dirsNotIncluded.size()]);
}
/**
* Returns the names of the directories which matched at least one of the
* include patterns and at least one of the exclude patterns. The names are
* relative to the base directory. This involves performing a slow scan if
* one has not already been completed.
*
* @return the names of the directories which matched at least one of the
* include patterns and at least one of the exclude patterns.
* @see #slowScan
*/
public String[] getExcludedDirectories() {
return (String[]) dirsExcluded.toArray(new String[dirsExcluded.size()]);
}
/** Adds default exclusions to the current exclusions set. */
public void addDefaultExcludes() {
int excludesLength = excludes == null ? 0 : excludes.length;
String[] newExcludes;
newExcludes = new String[excludesLength + FileUtil.DEFAULT_EXCLUDES.length];
if (excludesLength > 0) {
System.arraycopy(excludes, 0, newExcludes, 0, excludesLength);
}
for (int i = 0; i < FileUtil.DEFAULT_EXCLUDES.length; i++) {
newExcludes[i + excludesLength] = FileUtil.DEFAULT_EXCLUDES[i].replace('\\', '/');
}
excludes = newExcludes;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy