📄 jspcompiler.java
字号:
/* * The Apache Software License, Version 1.1 * * Copyright (c) 1999 The Apache Software Foundation. All rights * reserved. * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions * are met: * * 1. Redistributions of source code must retain the above copyright * notice, this list of conditions and the following disclaimer. * * 2. Redistributions in binary form must reproduce the above copyright * notice, this list of conditions and the following disclaimer in * the documentation and/or other materials provided with the * distribution. * * 3. The end-user documentation included with the redistribution, if * any, must include the following acknowlegement: * "This product includes software developed by the * Apache Software Foundation (http://www.apache.org/)." * Alternately, this acknowlegement may appear in the software itself, * if and wherever such third-party acknowlegements normally appear. * * 4. The names "The Jakarta Project", "Tomcat", and "Apache Software * Foundation" must not be used to endorse or promote products derived * from this software without prior written permission. For written * permission, please contact apache@apache.org. * * 5. Products derived from this software may not be called "Apache" * nor may "Apache" appear in their names without prior written * permission of the Apache Group. * * THIS SOFTWARE IS PROVIDED ``AS IS'' AND ANY EXPRESSED OR IMPLIED * WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES * OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE * DISCLAIMED. IN NO EVENT SHALL THE APACHE SOFTWARE FOUNDATION OR * ITS CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, * SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT * LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF * USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND * ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT * OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF * SUCH DAMAGE. * ==================================================================== * * This software consists of voluntary contributions made by many * individuals on behalf of the Apache Software Foundation. For more * information on the Apache Software Foundation, please see * <http://www.apache.org/>. * */ package org.apache.jasper.compiler;import java.io.File;import java.io.FileNotFoundException;import org.apache.jasper.JspCompilationContext;import org.apache.jasper.Constants;import org.apache.jasper.JasperException;/** * JspCompiler is an implementation of Compiler with a funky code * mangling and code generation scheme! * * The reason that it is both a sub-class of compiler and an implementation * of mangler is because the isOutDated method that is overridden and the * name mangulation both depend on the actual existance of other class and * java files. I.e. the value of a mangled name is a function of both the * name to be mangled and also of the state of the scratchdir. * * @author Anil K. Vijendran */public class JspCompiler extends Compiler implements Mangler { String pkgName, javaFileName, classFileName; String realClassName; File jsp; String outputDir; // ClassFileData cfd; boolean outDated; static final int JSP_TOKEN_LEN= Constants.JSP_TOKEN.length(); public JspCompiler(JspCompilationContext ctxt) throws JasperException { super(ctxt); this.jsp = new File(ctxt.getJspFile()); this.outputDir = ctxt.getOutputDir(); this.outDated = false; setMangler(this); // If the .class file exists and is outdated, compute a new // class name if( isOutDated() ) { generateNewClassName(); } } private void generateNewClassName() { File classFile = new File(getClassFileName()); if (! classFile.exists()) { String prefix = getPrefix(jsp.getPath()); realClassName= prefix + getBaseClassName() + Constants.JSP_TOKEN + "0"; return; } String cn=getRealClassName(); String baseClassName = cn. substring(0, cn.lastIndexOf(Constants.JSP_TOKEN)); int jspTokenIdx=cn.lastIndexOf(Constants.JSP_TOKEN); String versionS=cn.substring(jspTokenIdx + JSP_TOKEN_LEN, cn.length()); int number= Integer.valueOf(versionS).intValue(); number++; realClassName = baseClassName + Constants.JSP_TOKEN + number; } /** Return the real class name for the JSP, including package and * version. * * This method is called when the server is started and a .class file * is found from a previous compile or when the .class file is older, * to find next version. */ public final String getRealClassName() { if( realClassName!=null ) return realClassName; // System.out.println("JspCompiler: extract class name and version "); try { realClassName = ClassName.getClassName( getClassFileName() ); } catch( JasperException ex) { // ops, getClassName should throw something ex.printStackTrace(); return null; } return realClassName; } public final String getClassName() { // CFD gives you the whole class name // This method returns just the class name sans the package String cn=getRealClassName(); int lastDot = cn.lastIndexOf('.'); String className=null; if (lastDot != -1) className = cn.substring(lastDot+1, cn.length()); else // no package name case className = cn; return className; } public final String getJavaFileName() { if( javaFileName!=null ) return javaFileName; javaFileName = getClassName() + ".java"; if (outputDir != null && !outputDir.equals("")) javaFileName = outputDir + File.separatorChar + javaFileName; return javaFileName; } public final String getClassFileName() { if( classFileName!=null) return classFileName; // computeClassFileName(); String prefix = getPrefix(jsp.getPath()); classFileName = prefix + getBaseClassName() + ".class"; if (outputDir != null && !outputDir.equals("")) classFileName = outputDir + File.separatorChar + classFileName; return classFileName; } public static String [] keywords = { "abstract", "boolean", "break", "byte", "case", "catch", "char", "class", "const", "continue", "default", "do", "double", "else", "extends", "final", "finally", "float", "for", "goto", "if", "implements", "import", "instanceof", "int", "interface", "long", "native", "new", "package", "private", "protected", "public", "return", "short", "static", "super", "switch", "synchronized", "this", "throw", "throws", "transient", "try", "void", "volatile", "while" }; public final String getPackageName() { if( pkgName!=null) return pkgName; // compute package name String pathName = jsp.getPath(); StringBuffer modifiedpkgName = new StringBuffer (); int indexOfSepChar = pathName.lastIndexOf(File.separatorChar); if (indexOfSepChar == -1 || indexOfSepChar == 0) pkgName = null; else { for (int i = 0; i < keywords.length; i++) { char fs = File.separatorChar; int index1 = pathName.indexOf(fs + keywords[i]); int index2 = pathName.indexOf(keywords[i]); if (index1 == -1 && index2 == -1) continue; int index = (index2 == -1) ? index1 : index2; while (index != -1) { String tmpathName = pathName.substring (0,index+1) + '%'; pathName = tmpathName + pathName.substring (index+2); index = pathName.indexOf(fs + keywords[i]); } } // XXX fix for paths containing '.'. // Need to be more elegant here. pathName = pathName.replace('.','_'); pkgName = pathName.substring(0, pathName.lastIndexOf( File.separatorChar)).replace(File.separatorChar, '.'); for (int i=0; i<pkgName.length(); i++) if (Character.isLetter(pkgName.charAt(i)) == true || pkgName.charAt(i) == '.') { modifiedpkgName.append(pkgName.substring(i,i+1)); } else modifiedpkgName.append(mangleChar(pkgName.charAt(i))); if (modifiedpkgName.charAt(0) == '.') { String modifiedpkgNameString = modifiedpkgName.toString(); pkgName = modifiedpkgNameString. substring(1, modifiedpkgName.length ()); } else pkgName = modifiedpkgName.toString(); } return pkgName; } private final String getBaseClassName() { String className; if (jsp.getName().endsWith(".jsp")) className = jsp.getName().substring(0, jsp.getName().length() - 4); else className = jsp.getName(); // Fix for invalid characters. If you think of more add to the list. StringBuffer modifiedClassName = new StringBuffer(); for (int i = 0; i < className.length(); i++) { if (Character.isLetterOrDigit(className.charAt(i)) == true) modifiedClassName.append(className.substring(i,i+1)); else modifiedClassName.append(mangleChar(className.charAt(i))); } return modifiedClassName.toString(); } private final String getPrefix(String pathName) { if (pathName != null) { StringBuffer modifiedName = new StringBuffer(); for (int i = 0; i < pathName.length(); i++) { if (Character.isLetter(pathName.charAt(i)) == true) modifiedName.append(pathName.substring(i,i+1)); else modifiedName.append(mangleChar(pathName.charAt(i))); } return modifiedName.toString(); } else return ""; } private static final String mangleChar(char ch) { if(ch == File.separatorChar) { ch = '/'; } String s = Integer.toHexString(ch); int nzeros = 5 - s.length(); char[] result = new char[6]; result[0] = '_'; for (int i = 1; i <= nzeros; i++) result[i] = '0'; for (int i = nzeros+1, j = 0; i < 6; i++, j++) result[i] = s.charAt(j); return new String(result); } /** * Determines whether the current JSP class is older than the JSP file * from whence it came */ public boolean isOutDated() { File jspReal = null; String realPath = ctxt.getRealPath(jsp.getPath()); if (realPath == null) return true; jspReal = new File(realPath); File classFile = new File(getClassFileName()); if (classFile.exists()) { outDated = classFile.lastModified() < jspReal.lastModified(); } else { outDated = true; } return outDated; }}
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -