/* * Copyright 2004-2012 the Seasar Foundation and the Others. * * 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 org.seasar.util.io; import java.io.File; import java.util.jar.JarEntry; import java.util.jar.JarFile; import java.util.zip.ZipEntry; import java.util.zip.ZipInputStream; import org.seasar.util.lang.ClassUtil; import org.seasar.util.zip.ZipInputStreamUtil; import static org.seasar.util.collection.EnumerationIterator.*; import static org.seasar.util.misc.AssertionUtil.*; /** * クラスを横断して処理するためのハンドラです。 * * @author koichik * @see ClassHandler * @see TraversalUtil */ public abstract class ClassTraversalUtil { /** クラスファイルの拡張子 */ protected static final String CLASS_SUFFIX = ".class"; /** WARファイルの拡張子 */ protected static final String WAR_FILE_EXTENSION = ".war"; /** WARファイル内のクラスファイルのエントリプレフィックス */ protected static final String WEB_INF_CLASSES_PATH = "WEB-INF/classes/"; /** * ルートディレクトリ配下を処理します。 * * @param rootDir * ルートディレクトリ。{@literal null}であってはいけません * @param handler * クラスを処理するハンドラ。{@literal null}であってはいけません */ public static void forEach(final File rootDir, final ClassHandler handler) { assertArgumentNotNull("rootDir", rootDir); assertArgumentNotNull("handler", handler); forEach(rootDir, null, handler); } /** * ファイルシステムに含まれるクラスをトラバースします。 * * @param rootDir * ルートディレクトリ。{@literal null}であってはいけません * @param rootPackage * ルートパッケージ * @param handler * クラスを処理するハンドラ。{@literal null}であってはいけません */ public static void forEach(final File rootDir, final String rootPackage, final ClassHandler handler) { assertArgumentNotNull("rootDir", rootDir); assertArgumentNotNull("handler", handler); final File packageDir = getPackageDir(rootDir, rootPackage); if (packageDir.exists()) { traverseFileSystem(packageDir, rootPackage, handler); } } /** * Jarファイルに含まれるクラスをトラバースします。 * <p> * 指定されたJarファイルが拡張子<code>.war</code>を持つ場合は、 Jarファイル内のエントリのうち、 接頭辞 * <code>WEB-INF/classes</code>で始まるパスを持つエントリがトラバースの対象となります。 * クラスを処理するハンドラには、接頭辞を除くエントリ名が渡されます。 例えばJarファイル内に * <code>/WEB-INF/classes/ccc/ddd/Eee.class</code>というクラスファイルが存在すると、 ハンドラには * パッケージ名<code>ccc.ddd</code>およびクラス名<code>Eee</code>が渡されます。 * </p> * * @param jarFile * Jarファイル。{@literal null}であってはいけません * @param handler * クラスを処理するハンドラ。{@literal null}であってはいけません */ public static void forEach(final JarFile jarFile, final ClassHandler handler) { assertArgumentNotNull("jarFile", jarFile); assertArgumentNotNull("handler", handler); if (jarFile.getName().toLowerCase().endsWith(WAR_FILE_EXTENSION)) { forEach(jarFile, WEB_INF_CLASSES_PATH, handler); } else { forEach(jarFile, "", handler); } } /** * Jarファイルに含まれるクラスをトラバースします。 * <p> * Jarファイル内のエントリのうち、接頭辞で始まるパスを持つエントリがトラバースの対象となります。 * クラスを処理するハンドラには、接頭辞を除くエントリ名が渡されます。 例えば接頭辞が <code>/aaa/bbb/</code> * で、Jarファイル内に <code>/aaa/bbb/ccc/ddd/Eee.class</code>というクラスファイルが存在すると、 * ハンドラには パッケージ名<code>ccc.ddd</code>およびクラス名<code>Eee</code>が渡されます。 * </p> * * @param jarFile * Jarファイル。{@literal null}であってはいけません * @param prefix * トラバースするリソースの名前が含む接頭辞。{@literal null}であってはいけません。 * 空文字列でない場合はスラッシュ('/')で終了していなければなりません * @param handler * クラスを処理するハンドラ。{@literal null}であってはいけません */ public static void forEach(final JarFile jarFile, final String prefix, final ClassHandler handler) { assertArgumentNotNull("jarFile", jarFile); assertArgumentNotNull("prefix", prefix); assertArgumentNotNull("handler", handler); final int startPos = prefix.length(); for (final JarEntry entry : iterable(jarFile.entries())) { final String entryName = entry.getName().replace('\\', '/'); if (entryName.startsWith(prefix) && entryName.endsWith(CLASS_SUFFIX)) { final String className = entryName.substring( startPos, entryName.length() - CLASS_SUFFIX.length()).replace( '/', '.'); final int pos = className.lastIndexOf('.'); final String packageName = (pos == -1) ? null : className.substring(0, pos); final String shortClassName = (pos == -1) ? className : className.substring(pos + 1); handler.processClass(packageName, shortClassName); } } } /** * ZIPファイル形式の入力ストリームに含まれるクラスをトラバースします。 * * @param zipInputStream * ZIPファイル形式の入力ストリーム。{@literal null}であってはいけません * @param handler * クラスを処理するハンドラ。{@literal null}であってはいけません */ public static void forEach(final ZipInputStream zipInputStream, final ClassHandler handler) { assertArgumentNotNull("zipInputStream", zipInputStream); assertArgumentNotNull("handler", handler); forEach(zipInputStream, "", handler); } /** * ZIPファイル形式の入力ストリームに含まれるクラスをトラバースします。 * <p> * 入力ストリーム内のエントリのうち、接頭辞で始まるパスを持つエントリがトラバースの対象となります。 * クラスを処理するハンドラには、接頭辞を除くエントリ名が渡されます。 例えば接頭辞が <code>/aaa/bbb/</code> * で、入力ストリーム内に <code>/aaa/bbb/ccc/ddd/Eee.class</code>というクラスファイルが存在すると、 * ハンドラには パッケージ名<code>ccc.ddd</code>およびクラス名<code>Eee</code>が渡されます。 * </p> * * @param zipInputStream * ZIPファイル形式の入力ストリーム。{@literal null}であってはいけません * @param prefix * トラバースするリソースの名前が含む接頭辞。{@literal null}であってはいけません。 * 空文字列でない場合はスラッシュ('/')で終了していなければなりません * * @param handler * クラスを処理するハンドラ。{@literal null}であってはいけません */ public static void forEach(final ZipInputStream zipInputStream, final String prefix, final ClassHandler handler) { assertArgumentNotNull("zipInputStream", zipInputStream); assertArgumentNotNull("prefix", prefix); assertArgumentNotNull("handler", handler); final int startPos = prefix.length(); ZipEntry entry = null; while ((entry = ZipInputStreamUtil.getNextEntry(zipInputStream)) != null) { try { final String entryName = entry.getName().replace('\\', '/'); if (entryName.startsWith(prefix) && entryName.endsWith(CLASS_SUFFIX)) { final String className = entryName .substring( startPos, entryName.length() - CLASS_SUFFIX.length()) .replace('/', '.'); final int pos = className.lastIndexOf('.'); final String packageName = (pos == -1) ? null : className.substring(0, pos); final String shortClassName = (pos == -1) ? className : className.substring(pos + 1); handler.processClass(packageName, shortClassName); } } finally { ZipInputStreamUtil.closeEntry(zipInputStream); } } } /** * ファイルシステムに含まれるクラスをトラバースします。 * * @param dir * 基点となるディレクトリ * @param packageName * トラバースするパッケージ名 * @param handler * クラスを処理するハンドラ */ protected static void traverseFileSystem(final File dir, final String packageName, final ClassHandler handler) { for (final File file : dir.listFiles()) { final String fileName = file.getName(); if (file.isDirectory()) { traverseFileSystem( file, ClassUtil.concatName(packageName, fileName), handler); } else if (fileName.endsWith(".class")) { final String shortClassName = fileName.substring( 0, fileName.length() - CLASS_SUFFIX.length()); handler.processClass(packageName, shortClassName); } } } /** * ルートパッケージに対応するディレクトリを表す{@link File}を返します。 * * @param rootDir * ルートディレクトリ * @param rootPackage * ルートパッケージ * @return パッケージに対応するディレクトリを表す{@link File} */ protected static File getPackageDir(final File rootDir, final String rootPackage) { File packageDir = rootDir; if (rootPackage != null) { for (final String name : rootPackage.split("\\.")) { packageDir = new File(packageDir, name); } } return packageDir; } }