| /* |
| * Licensed to the Apache Software Foundation (ASF) under one or more |
| * contributor license agreements. See the NOTICE file distributed with |
| * this work for additional information regarding copyright ownership. |
| * The ASF licenses this file to You 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.apache.jasper.compiler; |
| |
| import jakarta.servlet.ServletContext; |
| |
| /** |
| * Provides {@link StringInterpreter} instances for JSP compilation. |
| * |
| * The search order is as follows: |
| * <ol> |
| * <li>StringInterpreter instance or implementation class name provided as a |
| * ServletContext attribute</li> |
| * <li>Implementation class named in a ServletContext initialisation parameter |
| * </li> |
| * <li>Default implementation</li> |
| * </ol> |
| */ |
| public class StringInterpreterFactory { |
| |
| public static final String STRING_INTERPRETER_CLASS_NAME = StringInterpreter.class.getName(); |
| |
| private static final StringInterpreter DEFAULT_INSTANCE = new DefaultStringInterpreter(); |
| |
| |
| /** |
| * Obtain the correct String Interpreter for the given web application. |
| * @param context The Servlet context |
| * @return the String interpreter |
| * @throws Exception If an error occurs creating the interpreter |
| */ |
| public static StringInterpreter getStringInterpreter(ServletContext context) |
| throws Exception { |
| |
| StringInterpreter result = null; |
| |
| // Search for an implementation |
| // 1. ServletContext attribute (set by application or cached by a |
| // previous call to this method). |
| Object attribute = context.getAttribute(STRING_INTERPRETER_CLASS_NAME); |
| if (attribute instanceof StringInterpreter) { |
| return (StringInterpreter) attribute; |
| } else if (attribute instanceof String) { |
| result = createInstance(context, (String) attribute); |
| } |
| |
| // 2. ServletContext init parameter |
| if (result == null) { |
| String className = context.getInitParameter(STRING_INTERPRETER_CLASS_NAME); |
| if (className != null) { |
| result = createInstance(context, className); |
| } |
| } |
| |
| // 3. Default |
| if (result == null) { |
| result = DEFAULT_INSTANCE; |
| } |
| |
| // Cache the result for next time |
| context.setAttribute(STRING_INTERPRETER_CLASS_NAME, result); |
| return result; |
| } |
| |
| |
| private static StringInterpreter createInstance(ServletContext context, |
| String className) throws Exception { |
| return (StringInterpreter) context.getClassLoader().loadClass( |
| className).getConstructor().newInstance(); |
| } |
| |
| |
| private StringInterpreterFactory() { |
| // Utility class. Hide default constructor. |
| } |
| |
| |
| public static class DefaultStringInterpreter implements StringInterpreter { |
| |
| @Override |
| public String convertString(Class<?> c, String s, String attrName, |
| Class<?> propEditorClass, boolean isNamedAttribute) { |
| |
| String quoted = s; |
| if (!isNamedAttribute) { |
| quoted = Generator.quote(s); |
| } |
| |
| if (propEditorClass != null) { |
| String className = c.getCanonicalName(); |
| return "(" |
| + className |
| + ")org.apache.jasper.runtime.JspRuntimeLibrary.getValueFromBeanInfoPropertyEditor(" |
| + className + ".class, \"" + attrName + "\", " + quoted |
| + ", " + propEditorClass.getCanonicalName() + ".class)"; |
| } else if (c == String.class) { |
| return quoted; |
| } else if (c == boolean.class) { |
| return JspUtil.coerceToPrimitiveBoolean(s, isNamedAttribute); |
| } else if (c == Boolean.class) { |
| return JspUtil.coerceToBoolean(s, isNamedAttribute); |
| } else if (c == byte.class) { |
| return JspUtil.coerceToPrimitiveByte(s, isNamedAttribute); |
| } else if (c == Byte.class) { |
| return JspUtil.coerceToByte(s, isNamedAttribute); |
| } else if (c == char.class) { |
| return JspUtil.coerceToChar(s, isNamedAttribute); |
| } else if (c == Character.class) { |
| return JspUtil.coerceToCharacter(s, isNamedAttribute); |
| } else if (c == double.class) { |
| return JspUtil.coerceToPrimitiveDouble(s, isNamedAttribute); |
| } else if (c == Double.class) { |
| return JspUtil.coerceToDouble(s, isNamedAttribute); |
| } else if (c == float.class) { |
| return JspUtil.coerceToPrimitiveFloat(s, isNamedAttribute); |
| } else if (c == Float.class) { |
| return JspUtil.coerceToFloat(s, isNamedAttribute); |
| } else if (c == int.class) { |
| return JspUtil.coerceToInt(s, isNamedAttribute); |
| } else if (c == Integer.class) { |
| return JspUtil.coerceToInteger(s, isNamedAttribute); |
| } else if (c == short.class) { |
| return JspUtil.coerceToPrimitiveShort(s, isNamedAttribute); |
| } else if (c == Short.class) { |
| return JspUtil.coerceToShort(s, isNamedAttribute); |
| } else if (c == long.class) { |
| return JspUtil.coerceToPrimitiveLong(s, isNamedAttribute); |
| } else if (c == Long.class) { |
| return JspUtil.coerceToLong(s, isNamedAttribute); |
| } else if (c == Object.class) { |
| return quoted; |
| } |
| |
| String result = coerceToOtherType(c, s, isNamedAttribute); |
| |
| if (result != null) { |
| return result; |
| } |
| |
| String className = c.getCanonicalName(); |
| return "(" |
| + className |
| + ")org.apache.jasper.runtime.JspRuntimeLibrary.getValueFromPropertyEditorManager(" |
| + className + ".class, \"" + attrName + "\", " + quoted |
| + ")"; |
| } |
| |
| |
| /** |
| * Intended to be used by sub-classes that don't need/want to |
| * re-implement the logic in |
| * {@link #convertString(Class, String, String, Class, boolean)}. |
| * |
| * @param c unused |
| * @param s unused |
| * @param isNamedAttribute unused |
| * |
| * @return Always {@code null} |
| */ |
| protected String coerceToOtherType(Class<?> c, String s, boolean isNamedAttribute) { |
| return null; |
| } |
| } |
| } |