⭐ 欢迎来到虫虫下载站! | 📦 资源下载 📁 资源专辑 ℹ️ 关于我们
⭐ 虫虫下载站

📄 sourcefinder.java

📁 A static analysis tool to find bugs in Java programs
💻 JAVA
字号:
/* * Bytecode Analysis Framework * Copyright (C) 2003,2004 University of Maryland *  * This library is free software; you can redistribute it and/or * modify it under the terms of the GNU Lesser General Public * License as published by the Free Software Foundation; either * version 2.1 of the License, or (at your option) any later version. *  * This library is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the GNU * Lesser General Public License for more details. *  * You should have received a copy of the GNU Lesser General Public * License along with this library; if not, write to the Free Software * Foundation, Inc., 59 Temple Place, Suite 330, Boston, MA  02111-1307  USA */package edu.umd.cs.findbugs.ba;import java.io.File;import java.io.FileNotFoundException;import java.io.IOException;import java.io.InputStream;import java.net.JarURLConnection;import java.net.MalformedURLException;import java.net.URL;import java.util.Enumeration;import java.util.LinkedHashMap;import java.util.LinkedList;import java.util.List;import java.util.Map;import java.util.zip.ZipFile;import java.util.zip.ZipEntry;import edu.umd.cs.findbugs.SourceLineAnnotation;import edu.umd.cs.findbugs.SystemProperties;/** * Class to open input streams on source files. * It maintains a "source path", which is like a classpath, * but for finding source files instead of class files. */public class SourceFinder {	private static final boolean DEBUG = SystemProperties.getBoolean("srcfinder.debug");	private static final int CACHE_SIZE = 50;	/* ----------------------------------------------------------------------	 * Helper classes	 * ---------------------------------------------------------------------- */	/**	 * Cache of SourceFiles.	 * We use this to avoid repeatedly having to read	 * frequently accessed source files.	 */	private static class Cache extends LinkedHashMap<String, SourceFile> {		/**		 * 		 */		private static final long serialVersionUID = 1L;		@Override				 protected boolean removeEldestEntry(Map.Entry<String, SourceFile> eldest) {			return size() >= CACHE_SIZE;		}	}	/**	 * A repository of source files.	 */	private interface SourceRepository {		public boolean contains(String fileName);		public boolean isPlatformDependent();		public SourceFileDataSource getDataSource(String fileName);	}	/**	 * A directory containing source files.	 */	private static class DirectorySourceRepository implements SourceRepository {		private String baseDir;		public DirectorySourceRepository(String baseDir) {			this.baseDir = baseDir;		}		@Override		public String toString() {			return "DirectorySourceRepository:" + baseDir;		}		public boolean contains(String fileName) {			File file = new File(getFullFileName(fileName));			boolean exists = file.exists();			if (DEBUG) System.out.println("Exists " + exists + " for " + file);			return exists;		}		public boolean isPlatformDependent() {			return true;		}		public SourceFileDataSource getDataSource(String fileName) {			return new FileSourceFileDataSource(getFullFileName(fileName));		}		private String getFullFileName(String fileName) {			return baseDir + File.separator + fileName;		}	}	 static class JarURLConnectionSourceRepository extends ZipSourceRepository {		public JarURLConnectionSourceRepository(String url) throws MalformedURLException, IOException {			super(((JarURLConnection) new URL("jar:" + url +"!/").openConnection()).getJarFile());			if (DEBUG) {				System.out.println("JarURLConnectionSourceRepository entries");			for(Enumeration<? extends ZipEntry> e = zipFile.entries(); e.hasMoreElements(); ) {				ZipEntry ze = e.nextElement();				System.out.println(ze.getName());			}			}		}	}	/**	 * A zip or jar archive containing source files.	 */	 static class ZipSourceRepository implements SourceRepository {		 ZipFile zipFile;		public ZipSourceRepository(ZipFile zipFile) {			this.zipFile = zipFile;		}		public boolean contains(String fileName) {			return zipFile.getEntry(fileName) != null;		}		public boolean isPlatformDependent() {			return false;		}		public SourceFileDataSource getDataSource(String fileName) {			return new ZipSourceFileDataSource(zipFile, fileName);		}	}	/* ----------------------------------------------------------------------	 * Fields	 * ---------------------------------------------------------------------- */	private List<SourceRepository> repositoryList;	private Cache cache;	/* ----------------------------------------------------------------------	 * Public methods	 * ---------------------------------------------------------------------- */	/**	 * Constructor.	 */	public SourceFinder() {		if (DEBUG) System.out.println("Debugging SourceFinder");		repositoryList = new LinkedList<SourceRepository>();		cache = new Cache();	}	/**	 * Set the list of source directories.	 */	public void setSourceBaseList(List<String> sourceBaseList) {		for (String repos : sourceBaseList) {			if (repos.endsWith(".zip") || repos.endsWith(".jar")) {				// Zip or jar archive				try {					if (repos.startsWith("http:") || repos.startsWith("https:") || repos.startsWith("file:")) 						repositoryList.add(new JarURLConnectionSourceRepository(repos));					else 						repositoryList.add(new ZipSourceRepository(new ZipFile(repos)));				} catch (IOException e) {					// Ignored - we won't use this archive				}			} else {				// Directory				repositoryList.add(new DirectorySourceRepository(repos));			}		}	}	/**	 * Open an input stream on a source file in given package.	 *	 * @param packageName the name of the package containing the class whose source file is given	 * @param fileName    the unqualified name of the source file	 * @return an InputStream on the source file	 * @throws IOException if a matching source file cannot be found	 */	public InputStream openSource(String packageName, String fileName) throws IOException {		SourceFile sourceFile = findSourceFile(packageName, fileName);		return sourceFile.getInputStream();	}	public InputStream openSource(SourceLineAnnotation source) throws IOException {		SourceFile sourceFile = findSourceFile(source);		return sourceFile.getInputStream();	}	public SourceFile findSourceFile(SourceLineAnnotation source) throws IOException {		if (source.isSourceFileKnown())			return findSourceFile(source.getPackageName(), source.getSourceFile());		String packageName = source.getPackageName();		String baseClassName = source.getClassName();		int i = baseClassName.lastIndexOf('.');		baseClassName = baseClassName.substring(i+1);		int j = baseClassName.indexOf("$");		if (j >= 0)			baseClassName = baseClassName.substring(0,j);		return findSourceFile(packageName, baseClassName + ".java");	}	/**	 * Open a source file in given package.	 *	 * @param packageName the name of the package containing the class whose source file is given	 * @param fileName    the unqualified name of the source file	 * @return the source file	 * @throws IOException if a matching source file cannot be found	 */	public SourceFile findSourceFile(String packageName, String fileName) throws IOException {		// On windows the fileName specification is different between a file in a directory tree, and a 		// file in a zip file. In a directory tree the separator used is '\', while in a zip it's '/'		// Therefore for each repository figure out what kind it is and use the appropriate separator.		// In all practicality, this code could just use the hardcoded '/' char, as windows can open 		// files with this separator, but to allow for the mythical 'other' platform that uses an		// alternate separator, make a distinction		// Create a fully qualified source filename using the package name for both directories and zips		String platformName = packageName.replace('.', File.separatorChar) + 								(packageName.length() > 0 ? File.separator : "") + fileName;		String canonicalName = packageName.replace('.', '/') + 								(packageName.length() > 0 ? "/" : "") + fileName;		// Is the file in the cache already? Always cache it with the canonical name		SourceFile sourceFile = cache.get(canonicalName);		if (sourceFile != null)			return sourceFile;		// Find this source file, add its data to the cache		if (DEBUG) System.out.println("Trying " + fileName +  " in package " + packageName + "...");		// Query each element of the source path to find the requested source file		for (SourceRepository repos : repositoryList) {			fileName = repos.isPlatformDependent() ? platformName : canonicalName;			if (DEBUG) System.out.println("Looking in " + repos  + " for " + fileName);			if (repos.contains(fileName)) {				// Found it				sourceFile = new SourceFile(repos.getDataSource(fileName));				cache.put(canonicalName, sourceFile); // always cache with canonicalName				return sourceFile;			}		}		throw new FileNotFoundException("Can't find source file " + fileName);	}}// vim:ts=4

⌨️ 快捷键说明

复制代码 Ctrl + C
搜索代码 Ctrl + F
全屏模式 F11
切换主题 Ctrl + Shift + D
显示快捷键 ?
增大字号 Ctrl + =
减小字号 Ctrl + -