• 类型转换概念

1、从html表单页面到一个Action对象,类型转化是从字符串到一个非字符串:html并没有“类型”的概念,每个表单输入的信息都只可能是一个字符串或者一个字符串数组,但是在服务器端,必须把String字符串转化为一种特定的数据类型;

2、在Struts2中,把请求参数映射到Action的属性的工作由ParametersInterceptor拦截器负责,它默认是defaultStack拦截器栈中的一员。Parameters拦截器可以自动完成字符串和基本类型之间转换。

ParameterInterceptor:

  1 /*
  2  * Copyright 2002-2007,2009 The Apache Software Foundation.
  3  *
  4  * Licensed under the Apache License, Version 2.0 (the "License");
  5  * you may not use this file except in compliance with the License.
  6  * You may obtain a copy of the License at
  7  *
  8  *      http://www.apache.org/licenses/LICENSE-2.0
  9  *
 10  * Unless required by applicable law or agreed to in writing, software
 11  * distributed under the License is distributed on an "AS IS" BASIS,
 12  * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 13  * See the License for the specific language governing permissions and
 14  * limitations under the License.
 15  */
 16 package com.opensymphony.xwork2.interceptor;
 17 
 18 import com.opensymphony.xwork2.ActionContext;
 19 import com.opensymphony.xwork2.ActionInvocation;
 20 import com.opensymphony.xwork2.security.AcceptedPatternsChecker;
 21 import com.opensymphony.xwork2.security.ExcludedPatternsChecker;
 22 import com.opensymphony.xwork2.ValidationAware;
 23 import com.opensymphony.xwork2.XWorkConstants;
 24 import com.opensymphony.xwork2.conversion.impl.InstantiatingNullHandler;
 25 import com.opensymphony.xwork2.conversion.impl.XWorkConverter;
 26 import com.opensymphony.xwork2.inject.Inject;
 27 import com.opensymphony.xwork2.util.ClearableValueStack;
 28 import com.opensymphony.xwork2.util.LocalizedTextUtil;
 29 import com.opensymphony.xwork2.util.MemberAccessValueStack;
 30 import com.opensymphony.xwork2.util.ValueStack;
 31 import com.opensymphony.xwork2.util.ValueStackFactory;
 32 import com.opensymphony.xwork2.util.logging.Logger;
 33 import com.opensymphony.xwork2.util.logging.LoggerFactory;
 34 import com.opensymphony.xwork2.util.reflection.ReflectionContextState;
 35 
 36 import java.util.Collection;
 37 import java.util.Comparator;
 38 import java.util.Map;
 39 import java.util.TreeMap;
 40 
 41 
 42 /**
 43  * <!-- START SNIPPET: description -->
 44  * This interceptor sets all parameters on the value stack.
 45  *
 46  * This interceptor gets all parameters from {@link ActionContext#getParameters()} and sets them on the value stack by
 47  * calling {@link ValueStack#setValue(String, Object)}, typically resulting in the values submitted in a form
 48  * request being applied to an action in the value stack. Note that the parameter map must contain a String key and
 49  * often containers a String[] for the value.
 50  *
 51  * The interceptor takes one parameter named 'ordered'. When set to true action properties are guaranteed to be
 52  * set top-down which means that top action's properties are set first. Then it's subcomponents properties are set.
 53  * The reason for this order is to enable a 'factory' pattern. For example, let's assume that one has an action
 54  * that contains a property named 'modelClass' that allows to choose what is the underlying implementation of model.
 55  * By assuring that modelClass property is set before any model properties are set, it's possible to choose model
 56  * implementation during action.setModelClass() call. Similiarily it's possible to use action.setPrimaryKey()
 57  * property set call to actually load the model class from persistent storage. Without any assumption on parameter
 58  * order you have to use patterns like 'Preparable'.
 59  *
 60  * Because parameter names are effectively OGNL statements, it is important that security be taken in to account.
 61  * This interceptor will not apply any values in the parameters map if the expression contains an assignment (=),
 62  * multiple expressions (,), or references any objects in the context (#). This is all done in the {@link
 63  * #acceptableName(String)} method. In addition to this method, if the action being invoked implements the {@link
 64  * ParameterNameAware} interface, the action will be consulted to determine if the parameter should be set.
 65  *
 66  * In addition to these restrictions, a flag ({@link ReflectionContextState#DENY_METHOD_EXECUTION}) is set such that
 67  * no methods are allowed to be invoked. That means that any expression such as <i>person.doSomething()</i> or
 68  * <i>person.getName()</i> will be explicitely forbidden. This is needed to make sure that your application is not
 69  * exposed to attacks by malicious users.
 70  *
 71  * While this interceptor is being invoked, a flag ({@link ReflectionContextState#CREATE_NULL_OBJECTS}) is turned
 72  * on to ensure that any null reference is automatically created - if possible. See the type conversion documentation
 73  * and the {@link InstantiatingNullHandler} javadocs for more information.
 74  *
 75  * Finally, a third flag ({@link XWorkConverter#REPORT_CONVERSION_ERRORS}) is set that indicates any errors when
 76  * converting the the values to their final data type (String[] -&gt; int) an unrecoverable error occured. With this
 77  * flag set, the type conversion errors will be reported in the action context. See the type conversion documentation
 78  * and the {@link XWorkConverter} javadocs for more information.
 79  *
 80  * If you are looking for detailed logging information about your parameters, turn on DEBUG level logging for this
 81  * interceptor. A detailed log of all the parameter keys and values will be reported.
 82  *
 83  * <b>Note:</b> Since XWork 2.0.2, this interceptor extends {@link MethodFilterInterceptor}, therefore being
 84  * able to deal with excludeMethods / includeMethods parameters. See [Workflow Interceptor]
 85  * (class {@link DefaultWorkflowInterceptor}) for documentation and examples on how to use this feature.
 86  * <!-- END SNIPPET: description -->
 87  *
 88  * <u>Interceptor parameters:</u>
 89  *
 90  * <!-- START SNIPPET: parameters -->
 91  *
 92  * <ul>
 93  * <li>ordered - set to true if you want the top-down property setter behaviour</li>
 94  * <li>acceptParamNames - a comma delimited list of regular expressions to describe a whitelist of accepted parameter names.
 95  * Don't change the default unless you know what you are doing in terms of security implications</li>
 96  * <li>excludeParams - a comma delimited list of regular expressions to describe a blacklist of not allowed parameter names</li>
 97  * <li>paramNameMaxLength - the maximum length of parameter names; parameters with longer names will be ignored; the default is 100 characters</li>
 98  * </ul>
 99  *
100  * <!-- END SNIPPET: parameters -->
101  *
102  *  <u>Extending the interceptor:</u>
103  *
104  * <!-- START SNIPPET: extending -->
105  *
106  *  The best way to add behavior to this interceptor is to utilize the {@link ParameterNameAware} interface in your
107  * actions. However, if you wish to apply a global rule that isn't implemented in your action, then you could extend
108  * this interceptor and override the {@link #acceptableName(String)} method.
109  *
110  * <!-- END SNIPPET: extending -->
111  *
112  *
113  * <!-- START SNIPPET: extending-warning -->
114  * Using {@link ParameterNameAware} could be dangerous as {@link ParameterNameAware#acceptableParameterName(String)} takes precedence
115  * over ParametersInterceptor which means if ParametersInterceptor excluded given parameter name you can accept it with
116  * {@link ParameterNameAware#acceptableParameterName(String)}.
117  *
118  * The best idea is to define very tight restrictions with ParametersInterceptor and relax them per action with
119  * {@link ParameterNameAware#acceptableParameterName(String)}
120  * <!-- END SNIPPET: extending-warning -->
121  *
122  *
123  * <u>Example code:</u>
124  *
125  * <pre>
126  * <!-- START SNIPPET: example -->
127  * &lt;action name="someAction" class="com.examples.SomeAction"&gt;
128  *     &lt;interceptor-ref name="params"/&gt;
129  *     &lt;result name="success"&gt;good_result.ftl&lt;/result&gt;
130  * &lt;/action&gt;
131  * <!-- END SNIPPET: example -->
132  * </pre>
133  *
134  * @author Patrick Lightbody
135  */
136 public class ParametersInterceptor extends MethodFilterInterceptor {
137 
138     private static final Logger LOG = LoggerFactory.getLogger(ParametersInterceptor.class);
139 
140     protected static final int PARAM_NAME_MAX_LENGTH = 100;
141 
142     private int paramNameMaxLength = PARAM_NAME_MAX_LENGTH;
143     private boolean devMode = false;
144 
145     protected boolean ordered = false;
146 
147     private ValueStackFactory valueStackFactory;
148     private ExcludedPatternsChecker excludedPatterns;
149     private AcceptedPatternsChecker acceptedPatterns;
150 
151     @Inject
152     public void setValueStackFactory(ValueStackFactory valueStackFactory) {
153         this.valueStackFactory = valueStackFactory;
154     }
155 
156     @Inject(XWorkConstants.DEV_MODE)
157     public void setDevMode(String mode) {
158         devMode = "true".equalsIgnoreCase(mode);
159     }
160 
161     @Inject
162     public void setExcludedPatterns(ExcludedPatternsChecker excludedPatterns) {
163         this.excludedPatterns = excludedPatterns;
164     }
165 
166     @Inject
167     public void setAcceptedPatterns(AcceptedPatternsChecker acceptedPatterns) {
168         this.acceptedPatterns = acceptedPatterns;
169     }
170 
171     /**
172      * If the param name exceeds the configured maximum length it will not be
173      * accepted.
174      *
175      * @param paramNameMaxLength Maximum length of param names
176      */
177     public void setParamNameMaxLength(int paramNameMaxLength) {
178         this.paramNameMaxLength = paramNameMaxLength;
179     }
180 
181     static private int countOGNLCharacters(String s) {
182         int count = 0;
183         for (int i = s.length() - 1; i >= 0; i--) {
184             char c = s.charAt(i);
185             if (c == '.' || c == '[') count++;
186         }
187         return count;
188     }
189 
190     /**
191      * Compares based on number of '.' and '[' characters (fewer is higher)
192      */
193     static final Comparator<String> rbCollator = new Comparator<String>() {
194         public int compare(String s1, String s2) {
195             int l1 = countOGNLCharacters(s1),
196                 l2 = countOGNLCharacters(s2);
197             return l1 < l2 ? -1 : (l2 < l1 ? 1 : s1.compareTo(s2));
198         }
199 
200     };
201 
202     @Override
203     public String doIntercept(ActionInvocation invocation) throws Exception {
204         Object action = invocation.getAction();
205         if (!(action instanceof NoParameters)) {
206             ActionContext ac = invocation.getInvocationContext();
207             final Map<String, Object> parameters = retrieveParameters(ac);
208 
209             if (LOG.isDebugEnabled()) {
210                 LOG.debug("Setting params " + getParameterLogMap(parameters));
211             }
212 
213             if (parameters != null) {
214                 Map<String, Object> contextMap = ac.getContextMap();
215                 try {
216                     ReflectionContextState.setCreatingNullObjects(contextMap, true);
217                     ReflectionContextState.setDenyMethodExecution(contextMap, true);
218                     ReflectionContextState.setReportingConversionErrors(contextMap, true);
219 
220                     ValueStack stack = ac.getValueStack();
221                     setParameters(action, stack, parameters);
222                 } finally {
223                     ReflectionContextState.setCreatingNullObjects(contextMap, false);
224                     ReflectionContextState.setDenyMethodExecution(contextMap, false);
225                     ReflectionContextState.setReportingConversionErrors(contextMap, false);
226                 }
227             }
228         }
229         return invocation.invoke();
230     }
231 
232     /**
233      * Gets the parameter map to apply from wherever appropriate
234      *
235      * @param ac The action context
236      * @return The parameter map to apply
237      */
238     protected Map<String, Object> retrieveParameters(ActionContext ac) {
239         return ac.getParameters();
240     }
241 
242 
243     /**
244      * Adds the parameters into context's ParameterMap
245      *
246      * @param ac        The action context
247      * @param newParams The parameter map to apply
248      *                  <p/>
249      *                  In this class this is a no-op, since the parameters were fetched from the same location.
250      *                  In subclasses both retrieveParameters() and addParametersToContext() should be overridden.
251      */
252     protected void addParametersToContext(ActionContext ac, Map<String, Object> newParams) {
253     }
254 
255     protected void setParameters(final Object action, ValueStack stack, final Map<String, Object> parameters) {
256         Map<String, Object> params;
257         Map<String, Object> acceptableParameters;
258         if (ordered) {
259             params = new TreeMap<String, Object>(getOrderedComparator());
260             acceptableParameters = new TreeMap<String, Object>(getOrderedComparator());
261             params.putAll(parameters);
262         } else {
263             params = new TreeMap<String, Object>(parameters);
264             acceptableParameters = new TreeMap<String, Object>();
265         }
266 
267         for (Map.Entry<String, Object> entry : params.entrySet()) {
268             String name = entry.getKey();
269             Object value = entry.getValue();
270             if (isAcceptableParameter(name, action)) {
271                 acceptableParameters.put(name, entry.getValue());
272             }
273         }
274 
275         ValueStack newStack = valueStackFactory.createValueStack(stack);
276         boolean clearableStack = newStack instanceof ClearableValueStack;
277         if (clearableStack) {
278             //if the stack's context can be cleared, do that to prevent OGNL
279             //from having access to objects in the stack, see XW-641
280             ((ClearableValueStack)newStack).clearContextValues();
281             Map<String, Object> context = newStack.getContext();
282             ReflectionContextState.setCreatingNullObjects(context, true);
283             ReflectionContextState.setDenyMethodExecution(context, true);
284             ReflectionContextState.setReportingConversionErrors(context, true);
285 
286             //keep locale from original context
287             context.put(ActionContext.LOCALE, stack.getContext().get(ActionContext.LOCALE));
288         }
289 
290         boolean memberAccessStack = newStack instanceof MemberAccessValueStack;
291         if (memberAccessStack) {
292             //block or allow access to properties
293             //see WW-2761 for more details
294             MemberAccessValueStack accessValueStack = (MemberAccessValueStack) newStack;
295             accessValueStack.setAcceptProperties(acceptedPatterns.getAcceptedPatterns());
296             accessValueStack.setExcludeProperties(excludedPatterns.getExcludedPatterns());
297         }
298 
299         for (Map.Entry<String, Object> entry : acceptableParameters.entrySet()) {
300             String name = entry.getKey();
301             Object value = entry.getValue();
302             try {
303                 newStack.setParameter(name, value);
304             } catch (RuntimeException e) {
305                 if (devMode) {
306                     notifyDeveloperParameterException(action, name, e.getMessage());
307                 }
308             }
309         }
310 
311         if (clearableStack && (stack.getContext() != null) && (newStack.getContext() != null))
312             stack.getContext().put(ActionContext.CONVERSION_ERRORS, newStack.getContext().get(ActionContext.CONVERSION_ERRORS));
313 
314         addParametersToContext(ActionContext.getContext(), acceptableParameters);
315     }
316 
317     protected void notifyDeveloperParameterException(Object action, String property, String message) {
318         String developerNotification = LocalizedTextUtil.findText(ParametersInterceptor.class, "devmode.notification",
319                 ActionContext.getContext().getLocale(), "Developer Notification:\n{0}",
320                 new Object[]{
321                         "Unexpected Exception caught setting '" + property + "' on '" + action.getClass() + ": " + message
322                 }
323         );
324         LOG.error(developerNotification);
325         // see https://issues.apache.org/jira/browse/WW-4066
326         if (action instanceof ValidationAware) {
327             Collection<String> messages = ((ValidationAware) action).getActionMessages();
328             messages.add(message);
329             ((ValidationAware) action).setActionMessages(messages);
330         }
331     }
332 
333     /**
334      * Checks if name of parameter can be accepted or thrown away
335      *
336      * @param name parameter name
337      * @param action current action
338      * @return true if parameter is accepted
339      */
340     protected boolean isAcceptableParameter(String name, Object action) {
341         ParameterNameAware parameterNameAware = (action instanceof ParameterNameAware) ? (ParameterNameAware) action : null;
342         return acceptableName(name) && (parameterNameAware == null || parameterNameAware.acceptableParameterName(name));
343     }
344 
345     /**
346      * Gets an instance of the comparator to use for the ordered sorting.  Override this
347      * method to customize the ordering of the parameters as they are set to the
348      * action.
349      *
350      * @return A comparator to sort the parameters
351      */
352     protected Comparator<String> getOrderedComparator() {
353         return rbCollator;
354     }
355 
356     protected String getParameterLogMap(Map<String, Object> parameters) {
357         if (parameters == null) {
358             return "NONE";
359         }
360 
361         StringBuilder logEntry = new StringBuilder();
362         for (Map.Entry entry : parameters.entrySet()) {
363             logEntry.append(String.valueOf(entry.getKey()));
364             logEntry.append(" => ");
365             if (entry.getValue() instanceof Object[]) {
366                 Object[] valueArray = (Object[]) entry.getValue();
367                 logEntry.append("[ ");
368                 if (valueArray.length > 0 ) {
369                     for (int indexA = 0; indexA < (valueArray.length - 1); indexA++) {
370                         Object valueAtIndex = valueArray[indexA];
371                         logEntry.append(String.valueOf(valueAtIndex));
372                         logEntry.append(", ");
373                     }
374                     logEntry.append(String.valueOf(valueArray[valueArray.length - 1]));
375                 }
376                 logEntry.append(" ] ");
377             } else {
378                 logEntry.append(String.valueOf(entry.getValue()));
379             }
380         }
381 
382         return logEntry.toString();
383     }
384 
385     protected boolean acceptableName(String name) {
386         boolean accepted = isWithinLengthLimit(name) && !isExcluded(name) && isAccepted(name);
387         if (devMode && accepted) { // notify only when in devMode
388             LOG.debug("Parameter [#0] was accepted and will be appended to action!", name);
389         }
390         return accepted;
391     }
392 
393     protected boolean isWithinLengthLimit( String name ) {
394         boolean matchLength = name.length() <= paramNameMaxLength;
395         if (!matchLength) {
396             notifyDeveloper("Parameter [#0] is too long, allowed length is [#1]", name, String.valueOf(paramNameMaxLength));
397         }
398         return matchLength;
399     }
400 
401     protected boolean isAccepted(String paramName) {
402         AcceptedPatternsChecker.IsAccepted result = acceptedPatterns.isAccepted(paramName);
403         if (result.isAccepted()) {
404             return true;
405         }
406         notifyDeveloper("Parameter [#0] didn't match accepted pattern [#1]!", paramName, result.getAcceptedPattern());
407         return false;
408     }
409 
410     protected boolean isExcluded(String paramName) {
411         ExcludedPatternsChecker.IsExcluded result = excludedPatterns.isExcluded(paramName);
412         if (result.isExcluded()) {
413             notifyDeveloper("Parameter [#0] matches excluded pattern [#1]!", paramName, result.getExcludedPattern());
414             return true;
415         }
416         return false;
417     }
418 
419     private void notifyDeveloper(String message, String... parameters) {
420         if (devMode) {
421             LOG.warn(message, parameters);
422         } else {
423             if (LOG.isDebugEnabled()) {
424                 LOG.debug(message, parameters);
425             }
426         }
427     }
428 
429     /**
430      * Whether to order the parameters or not
431      *
432      * @return True to order
433      */
434     public boolean isOrdered() {
435         return ordered;
436     }
437 
438     /**
439      * Set whether to order the parameters by object depth or not
440      *
441      * @param ordered True to order them
442      */
443     public void setOrdered(boolean ordered) {
444         this.ordered = ordered;
445     }
446 
447     /**
448      * Sets a comma-delimited list of regular expressions to match
449      * parameters that are allowed in the parameter map (aka whitelist).
450      * <p/>
451      * Don't change the default unless you know what you are doing in terms
452      * of security implications.
453      *
454      * @param commaDelim A comma-delimited list of regular expressions
455      */
456     public void setAcceptParamNames(String commaDelim) {
457         acceptedPatterns.setAcceptedPatterns(commaDelim);
458     }
459 
460     /**
461      * Sets a comma-delimited list of regular expressions to match
462      * parameters that should be removed from the parameter map.
463      *
464      * @param commaDelim A comma-delimited list of regular expressions
465      */
466     public void setExcludeParams(String commaDelim) {
467         excludedPatterns.setExcludedPatterns(commaDelim);
468     }
469 
470 }
View Code

相关文章: