• Struts(二十四):短路验证&重写实现转换验证失败时短路&非字段验证


    • 短路验证:

     若对一个字段使用多个验证器,默认情况下会执行所有的验证。若希望前面的验证器没有通过,后面的验证器就不再执行,可以使用短路验证。

    1、如下拦截器,如果输入字符串,提交表单后,默认是会出现三个错误:字段转换失败、conversion验证失败、取值范围失败。

    <!DOCTYPE validators PUBLIC
            "-//Apache Struts//XWork Validator 1.0.2//EN"
            "http://struts.apache.org/dtds/xwork-validator-1.0.2.dtd">
    
    <validators>
        <field name="age">
            <field-validator type="conversion">
                <message>Conversion error!</message>
            </field-validator>
            <field-validator type="int">
                <param name="max">180</param>
                <param name="min">1</param>
                <message key="ageErrorMsg"></message>
            </field-validator>
        </field>
    </validators>

    2、如果使用短路验证,则如果输入字符串,提交表单后,默认是会出现两个错误:字段转换失败、conversion验证失败。

    <!DOCTYPE validators PUBLIC
            "-//Apache Struts//XWork Validator 1.0.2//EN"
            "http://struts.apache.org/dtds/xwork-validator-1.0.2.dtd">
    
    <validators>
        <field name="age">
            <field-validator type="conversion" short-circuit="true">
                <message>Conversion error!</message>
            </field-validator>
            <field-validator type="int">
                <param name="max">180</param>
                <param name="min">1</param>
                <message key="ageErrorMsg"></message>
            </field-validator>
        </field>
    </validators>
    • 重写实现转换失败时短路:

     若类型转换失败,默认情况下会执行后边的额拦截器,还会进行验证。可以通过修改ConversionErrorInterceptor源代码的方式,实现当类型转换失败时,不再执行后边的验证拦截器,从而直接返回input的result。

      从上边的例子中我们知道,只要我们输入的字段类型不合法必定会在错误信息中包含示:类型转换失败错误信息。

      那么,怎么避免这个错误信息呢?

      首先,从struts2-core.jar根目录下的struts-default.xml中的default-stack(默认拦截器栈)中我们得知,类型转化是通过params(com.opensymphony.xwork2.interceptor.ParametersInterceptor

     * Copyright 2002-2007,2009 The Apache Software Foundation.
    package com.opensymphony.xwork2.interceptor;
    
    import com.opensymphony.xwork2.ActionContext;
    import com.opensymphony.xwork2.ActionInvocation;
    import com.opensymphony.xwork2.security.AcceptedPatternsChecker;
    import com.opensymphony.xwork2.security.ExcludedPatternsChecker;
    import com.opensymphony.xwork2.ValidationAware;
    import com.opensymphony.xwork2.XWorkConstants;
    import com.opensymphony.xwork2.conversion.impl.InstantiatingNullHandler;
    import com.opensymphony.xwork2.conversion.impl.XWorkConverter;
    import com.opensymphony.xwork2.inject.Inject;
    import com.opensymphony.xwork2.util.ClearableValueStack;
    import com.opensymphony.xwork2.util.LocalizedTextUtil;
    import com.opensymphony.xwork2.util.MemberAccessValueStack;
    import com.opensymphony.xwork2.util.ValueStack;
    import com.opensymphony.xwork2.util.ValueStackFactory;
    import com.opensymphony.xwork2.util.logging.Logger;
    import com.opensymphony.xwork2.util.logging.LoggerFactory;
    import com.opensymphony.xwork2.util.reflection.ReflectionContextState;
    
    import java.util.Collection;
    import java.util.Comparator;
    import java.util.Map;
    import java.util.TreeMap;
    
    
    /**
     * <!-- START SNIPPET: description -->
     * This interceptor sets all parameters on the value stack.
     *
     * This interceptor gets all parameters from {@link ActionContext#getParameters()} and sets them on the value stack by
     * calling {@link ValueStack#setValue(String, Object)}, typically resulting in the values submitted in a form
     * request being applied to an action in the value stack. Note that the parameter map must contain a String key and
     * often containers a String[] for the value.
     *
     * The interceptor takes one parameter named 'ordered'. When set to true action properties are guaranteed to be
     * set top-down which means that top action's properties are set first. Then it's subcomponents properties are set.
     * The reason for this order is to enable a 'factory' pattern. For example, let's assume that one has an action
     * that contains a property named 'modelClass' that allows to choose what is the underlying implementation of model.
     * By assuring that modelClass property is set before any model properties are set, it's possible to choose model
     * implementation during action.setModelClass() call. Similiarily it's possible to use action.setPrimaryKey()
     * property set call to actually load the model class from persistent storage. Without any assumption on parameter
     * order you have to use patterns like 'Preparable'.
     *
     * Because parameter names are effectively OGNL statements, it is important that security be taken in to account.
     * This interceptor will not apply any values in the parameters map if the expression contains an assignment (=),
     * multiple expressions (,), or references any objects in the context (#). This is all done in the {@link
     * #acceptableName(String)} method. In addition to this method, if the action being invoked implements the {@link
     * ParameterNameAware} interface, the action will be consulted to determine if the parameter should be set.
     *
     * In addition to these restrictions, a flag ({@link ReflectionContextState#DENY_METHOD_EXECUTION}) is set such that
     * no methods are allowed to be invoked. That means that any expression such as <i>person.doSomething()</i> or
     * <i>person.getName()</i> will be explicitely forbidden. This is needed to make sure that your application is not
     * exposed to attacks by malicious users.
     *
     * While this interceptor is being invoked, a flag ({@link ReflectionContextState#CREATE_NULL_OBJECTS}) is turned
     * on to ensure that any null reference is automatically created - if possible. See the type conversion documentation
     * and the {@link InstantiatingNullHandler} javadocs for more information.
     *
     * Finally, a third flag ({@link XWorkConverter#REPORT_CONVERSION_ERRORS}) is set that indicates any errors when
     * converting the the values to their final data type (String[] -&gt; int) an unrecoverable error occured. With this
     * flag set, the type conversion errors will be reported in the action context. See the type conversion documentation
     * and the {@link XWorkConverter} javadocs for more information.
     *
     * If you are looking for detailed logging information about your parameters, turn on DEBUG level logging for this
     * interceptor. A detailed log of all the parameter keys and values will be reported.
     *
     * <b>Note:</b> Since XWork 2.0.2, this interceptor extends {@link MethodFilterInterceptor}, therefore being
     * able to deal with excludeMethods / includeMethods parameters. See [Workflow Interceptor]
     * (class {@link DefaultWorkflowInterceptor}) for documentation and examples on how to use this feature.
     * <!-- END SNIPPET: description -->
     *
     * <u>Interceptor parameters:</u>
     *
     * <!-- START SNIPPET: parameters -->
     *
     * <ul>
     * <li>ordered - set to true if you want the top-down property setter behaviour</li>
     * <li>acceptParamNames - a comma delimited list of regular expressions to describe a whitelist of accepted parameter names.
     * Don't change the default unless you know what you are doing in terms of security implications</li>
     * <li>excludeParams - a comma delimited list of regular expressions to describe a blacklist of not allowed parameter names</li>
     * <li>paramNameMaxLength - the maximum length of parameter names; parameters with longer names will be ignored; the default is 100 characters</li>
     * </ul>
     *
     * <!-- END SNIPPET: parameters -->
     *
     *  <u>Extending the interceptor:</u>
     *
     * <!-- START SNIPPET: extending -->
     *
     *  The best way to add behavior to this interceptor is to utilize the {@link ParameterNameAware} interface in your
     * actions. However, if you wish to apply a global rule that isn't implemented in your action, then you could extend
     * this interceptor and override the {@link #acceptableName(String)} method.
     *
     * <!-- END SNIPPET: extending -->
     *
     *
     * <!-- START SNIPPET: extending-warning -->
     * Using {@link ParameterNameAware} could be dangerous as {@link ParameterNameAware#acceptableParameterName(String)} takes precedence
     * over ParametersInterceptor which means if ParametersInterceptor excluded given parameter name you can accept it with
     * {@link ParameterNameAware#acceptableParameterName(String)}.
     *
     * The best idea is to define very tight restrictions with ParametersInterceptor and relax them per action with
     * {@link ParameterNameAware#acceptableParameterName(String)}
     * <!-- END SNIPPET: extending-warning -->
     *
     *
     * <u>Example code:</u>
     *
     * <pre>
     * <!-- START SNIPPET: example -->
     * &lt;action name="someAction" class="com.examples.SomeAction"&gt;
     *     &lt;interceptor-ref name="params"/&gt;
     *     &lt;result name="success"&gt;good_result.ftl&lt;/result&gt;
     * &lt;/action&gt;
     * <!-- END SNIPPET: example -->
     * </pre>
     *
     * @author Patrick Lightbody
     */
    public class ParametersInterceptor extends MethodFilterInterceptor {
    
        private static final Logger LOG = LoggerFactory.getLogger(ParametersInterceptor.class);
    
        protected static final int PARAM_NAME_MAX_LENGTH = 100;
    
        private int paramNameMaxLength = PARAM_NAME_MAX_LENGTH;
        private boolean devMode = false;
    
        protected boolean ordered = false;
    
        private ValueStackFactory valueStackFactory;
        private ExcludedPatternsChecker excludedPatterns;
        private AcceptedPatternsChecker acceptedPatterns;
    
        @Inject
        public void setValueStackFactory(ValueStackFactory valueStackFactory) {
            this.valueStackFactory = valueStackFactory;
        }
    
        @Inject(XWorkConstants.DEV_MODE)
        public void setDevMode(String mode) {
            devMode = "true".equalsIgnoreCase(mode);
        }
    
        @Inject
        public void setExcludedPatterns(ExcludedPatternsChecker excludedPatterns) {
            this.excludedPatterns = excludedPatterns;
        }
    
        @Inject
        public void setAcceptedPatterns(AcceptedPatternsChecker acceptedPatterns) {
            this.acceptedPatterns = acceptedPatterns;
        }
    
        /**
         * If the param name exceeds the configured maximum length it will not be
         * accepted.
         *
         * @param paramNameMaxLength Maximum length of param names
         */
        public void setParamNameMaxLength(int paramNameMaxLength) {
            this.paramNameMaxLength = paramNameMaxLength;
        }
    
        static private int countOGNLCharacters(String s) {
            int count = 0;
            for (int i = s.length() - 1; i >= 0; i--) {
                char c = s.charAt(i);
                if (c == '.' || c == '[') count++;
            }
            return count;
        }
    
        /**
         * Compares based on number of '.' and '[' characters (fewer is higher)
         */
        static final Comparator<String> rbCollator = new Comparator<String>() {
            public int compare(String s1, String s2) {
                int l1 = countOGNLCharacters(s1),
                    l2 = countOGNLCharacters(s2);
                return l1 < l2 ? -1 : (l2 < l1 ? 1 : s1.compareTo(s2));
            }
    
        };
    
        @Override
        public String doIntercept(ActionInvocation invocation) throws Exception {
            Object action = invocation.getAction();
            if (!(action instanceof NoParameters)) {
                ActionContext ac = invocation.getInvocationContext();
                final Map<String, Object> parameters = retrieveParameters(ac);
    
                if (LOG.isDebugEnabled()) {
                    LOG.debug("Setting params " + getParameterLogMap(parameters));
                }
    
                if (parameters != null) {
                    Map<String, Object> contextMap = ac.getContextMap();
                    try {
                        ReflectionContextState.setCreatingNullObjects(contextMap, true);
                        ReflectionContextState.setDenyMethodExecution(contextMap, true);
                        ReflectionContextState.setReportingConversionErrors(contextMap, true);
    
                        ValueStack stack = ac.getValueStack();
                        setParameters(action, stack, parameters);
                    } finally {
                        ReflectionContextState.setCreatingNullObjects(contextMap, false);
                        ReflectionContextState.setDenyMethodExecution(contextMap, false);
                        ReflectionContextState.setReportingConversionErrors(contextMap, false);
                    }
                }
            }
            return invocation.invoke();
        }
    
        /**
         * Gets the parameter map to apply from wherever appropriate
         *
         * @param ac The action context
         * @return The parameter map to apply
         */
        protected Map<String, Object> retrieveParameters(ActionContext ac) {
            return ac.getParameters();
        }
    
    
        /**
         * Adds the parameters into context's ParameterMap
         *
         * @param ac        The action context
         * @param newParams The parameter map to apply
         *                  <p/>
         *                  In this class this is a no-op, since the parameters were fetched from the same location.
         *                  In subclasses both retrieveParameters() and addParametersToContext() should be overridden.
         */
        protected void addParametersToContext(ActionContext ac, Map<String, Object> newParams) {
        }
    
        protected void setParameters(final Object action, ValueStack stack, final Map<String, Object> parameters) {
            Map<String, Object> params;
            Map<String, Object> acceptableParameters;
            if (ordered) {
                params = new TreeMap<String, Object>(getOrderedComparator());
                acceptableParameters = new TreeMap<String, Object>(getOrderedComparator());
                params.putAll(parameters);
            } else {
                params = new TreeMap<String, Object>(parameters);
                acceptableParameters = new TreeMap<String, Object>();
            }
    
            for (Map.Entry<String, Object> entry : params.entrySet()) {
                String name = entry.getKey();
                Object value = entry.getValue();
                if (isAcceptableParameter(name, action)) {
                    acceptableParameters.put(name, entry.getValue());
                }
            }
    
            ValueStack newStack = valueStackFactory.createValueStack(stack);
            boolean clearableStack = newStack instanceof ClearableValueStack;
            if (clearableStack) {
                //if the stack's context can be cleared, do that to prevent OGNL
                //from having access to objects in the stack, see XW-641
                ((ClearableValueStack)newStack).clearContextValues();
                Map<String, Object> context = newStack.getContext();
                ReflectionContextState.setCreatingNullObjects(context, true);
                ReflectionContextState.setDenyMethodExecution(context, true);
                ReflectionContextState.setReportingConversionErrors(context, true);
    
                //keep locale from original context
                context.put(ActionContext.LOCALE, stack.getContext().get(ActionContext.LOCALE));
            }
    
            boolean memberAccessStack = newStack instanceof MemberAccessValueStack;
            if (memberAccessStack) {
                //block or allow access to properties
                //see WW-2761 for more details
                MemberAccessValueStack accessValueStack = (MemberAccessValueStack) newStack;
                accessValueStack.setAcceptProperties(acceptedPatterns.getAcceptedPatterns());
                accessValueStack.setExcludeProperties(excludedPatterns.getExcludedPatterns());
            }
    
            for (Map.Entry<String, Object> entry : acceptableParameters.entrySet()) {
                String name = entry.getKey();
                Object value = entry.getValue();
                try {
                    newStack.setParameter(name, value);
                } catch (RuntimeException e) {
                    if (devMode) {
                        notifyDeveloperParameterException(action, name, e.getMessage());
                    }
                }
            }
    
            if (clearableStack && (stack.getContext() != null) && (newStack.getContext() != null))
                stack.getContext().put(ActionContext.CONVERSION_ERRORS, newStack.getContext().get(ActionContext.CONVERSION_ERRORS));
    
            addParametersToContext(ActionContext.getContext(), acceptableParameters);
        }
    
        protected void notifyDeveloperParameterException(Object action, String property, String message) {
            String developerNotification = LocalizedTextUtil.findText(ParametersInterceptor.class, "devmode.notification",
                    ActionContext.getContext().getLocale(), "Developer Notification:
    {0}",
                    new Object[]{
                            "Unexpected Exception caught setting '" + property + "' on '" + action.getClass() + ": " + message
                    }
            );
            LOG.error(developerNotification);
            // see https://issues.apache.org/jira/browse/WW-4066
            if (action instanceof ValidationAware) {
                Collection<String> messages = ((ValidationAware) action).getActionMessages();
                messages.add(message);
                ((ValidationAware) action).setActionMessages(messages);
            }
        }
    
        /**
         * Checks if name of parameter can be accepted or thrown away
         *
         * @param name parameter name
         * @param action current action
         * @return true if parameter is accepted
         */
        protected boolean isAcceptableParameter(String name, Object action) {
            ParameterNameAware parameterNameAware = (action instanceof ParameterNameAware) ? (ParameterNameAware) action : null;
            return acceptableName(name) && (parameterNameAware == null || parameterNameAware.acceptableParameterName(name));
        }
    
        /**
         * Gets an instance of the comparator to use for the ordered sorting.  Override this
         * method to customize the ordering of the parameters as they are set to the
         * action.
         *
         * @return A comparator to sort the parameters
         */
        protected Comparator<String> getOrderedComparator() {
            return rbCollator;
        }
    
        protected String getParameterLogMap(Map<String, Object> parameters) {
            if (parameters == null) {
                return "NONE";
            }
    
            StringBuilder logEntry = new StringBuilder();
            for (Map.Entry entry : parameters.entrySet()) {
                logEntry.append(String.valueOf(entry.getKey()));
                logEntry.append(" => ");
                if (entry.getValue() instanceof Object[]) {
                    Object[] valueArray = (Object[]) entry.getValue();
                    logEntry.append("[ ");
                    if (valueArray.length > 0 ) {
                        for (int indexA = 0; indexA < (valueArray.length - 1); indexA++) {
                            Object valueAtIndex = valueArray[indexA];
                            logEntry.append(String.valueOf(valueAtIndex));
                            logEntry.append(", ");
                        }
                        logEntry.append(String.valueOf(valueArray[valueArray.length - 1]));
                    }
                    logEntry.append(" ] ");
                } else {
                    logEntry.append(String.valueOf(entry.getValue()));
                }
            }
    
            return logEntry.toString();
        }
    
        protected boolean acceptableName(String name) {
            boolean accepted = isWithinLengthLimit(name) && !isExcluded(name) && isAccepted(name);
            if (devMode && accepted) { // notify only when in devMode
                LOG.debug("Parameter [#0] was accepted and will be appended to action!", name);
            }
            return accepted;
        }
    
        protected boolean isWithinLengthLimit( String name ) {
            boolean matchLength = name.length() <= paramNameMaxLength;
            if (!matchLength) {
                notifyDeveloper("Parameter [#0] is too long, allowed length is [#1]", name, String.valueOf(paramNameMaxLength));
            }
            return matchLength;
        }
    
        protected boolean isAccepted(String paramName) {
            AcceptedPatternsChecker.IsAccepted result = acceptedPatterns.isAccepted(paramName);
            if (result.isAccepted()) {
                return true;
            }
            notifyDeveloper("Parameter [#0] didn't match accepted pattern [#1]!", paramName, result.getAcceptedPattern());
            return false;
        }
    
        protected boolean isExcluded(String paramName) {
            ExcludedPatternsChecker.IsExcluded result = excludedPatterns.isExcluded(paramName);
            if (result.isExcluded()) {
                notifyDeveloper("Parameter [#0] matches excluded pattern [#1]!", paramName, result.getExcludedPattern());
                return true;
            }
            return false;
        }
    
        private void notifyDeveloper(String message, String... parameters) {
            if (devMode) {
                LOG.warn(message, parameters);
            } else {
                if (LOG.isDebugEnabled()) {
                    LOG.debug(message, parameters);
                }
            }
        }
    
        /**
         * Whether to order the parameters or not
         *
         * @return True to order
         */
        public boolean isOrdered() {
            return ordered;
        }
    
        /**
         * Set whether to order the parameters by object depth or not
         *
         * @param ordered True to order them
         */
        public void setOrdered(boolean ordered) {
            this.ordered = ordered;
        }
    
        /**
         * Sets a comma-delimited list of regular expressions to match
         * parameters that are allowed in the parameter map (aka whitelist).
         * <p/>
         * Don't change the default unless you know what you are doing in terms
         * of security implications.
         *
         * @param commaDelim A comma-delimited list of regular expressions
         */
        public void setAcceptParamNames(String commaDelim) {
            acceptedPatterns.setAcceptedPatterns(commaDelim);
        }
    
        /**
         * Sets a comma-delimited list of regular expressions to match
         * parameters that should be removed from the parameter map.
         *
         * @param commaDelim A comma-delimited list of regular expressions
         */
        public void setExcludeParams(String commaDelim) {
            excludedPatterns.setExcludedPatterns(commaDelim);
        }
    
    }
    View Code

    )拦截器来实现的,而类型转化失败信息是被conversionError(org.apache.struts2.interceptor.StrutsConversionErrorInterceptor,它的父类是com.opensymphony.xwork2.interceptor.ConversionErrorInterceptor

    /*
     * Copyright 2002-2007,2009 The Apache Software Foundation.
     * 
     * 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 com.opensymphony.xwork2.interceptor;
    
    import com.opensymphony.xwork2.ActionContext;
    import com.opensymphony.xwork2.ActionInvocation;
    import com.opensymphony.xwork2.ValidationAware;
    import com.opensymphony.xwork2.conversion.impl.XWorkConverter;
    import com.opensymphony.xwork2.util.ValueStack;
    import org.apache.commons.lang3.StringEscapeUtils;
    
    import java.util.HashMap;
    import java.util.Map;
    
    
    /**
     * <!-- START SNIPPET: description -->
     * ConversionErrorInterceptor adds conversion errors from the ActionContext to the Action's field errors.
     *
     * <p/>
     * This interceptor adds any error found in the {@link ActionContext}'s conversionErrors map as a field error (provided
     * that the action implements {@link ValidationAware}). In addition, any field that contains a validation error has its
     * original value saved such that any subsequent requests for that value return the original value rather than the value
     * in the action. This is important because if the value "abc" is submitted and can't be converted to an int, we want to
     * display the original string ("abc") again rather than the int value (likely 0, which would make very little sense to
     * the user).
     *
     *
     * <!-- END SNIPPET: description -->
     *
     * <p/> <u>Interceptor parameters:</u>
     *
     * <!-- START SNIPPET: parameters -->
     *
     * <ul>
     *
     * <li>None</li>
     *
     * </ul>
     *
     * <!-- END SNIPPET: parameters -->
     *
     * <p/> <u>Extending the interceptor:</u>
     *
     * <p/>
     *
     * <!-- START SNIPPET: extending -->
     *
     * Because this interceptor is not web-specific, it abstracts the logic for whether an error should be added. This
     * allows for web-specific interceptors to use more complex logic in the {@link #shouldAddError} method for when a value
     * has a conversion error but is null or empty or otherwise indicates that the value was never actually entered by the
     * user.
     *
     * <!-- END SNIPPET: extending -->
     *
     * <p/> <u>Example code:</u>
     *
     * <pre>
     * <!-- START SNIPPET: example -->
     * &lt;action name="someAction" class="com.examples.SomeAction"&gt;
     *     &lt;interceptor-ref name="params"/&gt;
     *     &lt;interceptor-ref name="conversionError"/&gt;
     *     &lt;result name="success"&gt;good_result.ftl&lt;/result&gt;
     * &lt;/action&gt;
     * <!-- END SNIPPET: example -->
     * </pre>
     *
     * @author Jason Carreira
     */
    public class ConversionErrorInterceptor extends AbstractInterceptor {
    
        public static final String ORIGINAL_PROPERTY_OVERRIDE = "original.property.override";
    
        protected Object getOverrideExpr(ActionInvocation invocation, Object value) {
            return escape(value);
        }
    
        protected String escape(Object value) {
            return """ + StringEscapeUtils.escapeJava(String.valueOf(value)) + """;
        }
    
        @Override
        public String intercept(ActionInvocation invocation) throws Exception {
    
            ActionContext invocationContext = invocation.getInvocationContext();
            Map<String, Object> conversionErrors = invocationContext.getConversionErrors();
            ValueStack stack = invocationContext.getValueStack();
    
            HashMap<Object, Object> fakie = null;
    
            for (Map.Entry<String, Object> entry : conversionErrors.entrySet()) {
                String propertyName = entry.getKey();
                Object value = entry.getValue();
    
                if (shouldAddError(propertyName, value)) {
                    String message = XWorkConverter.getConversionErrorMessage(propertyName, stack);
    
                    Object action = invocation.getAction();
                    if (action instanceof ValidationAware) {
                        ValidationAware va = (ValidationAware) action;
                        va.addFieldError(propertyName, message);
                    }
    
                    if (fakie == null) {
                        fakie = new HashMap<Object, Object>();
                    }
    
                    fakie.put(propertyName, getOverrideExpr(invocation, value));
                }
            }
    
            if (fakie != null) {
                // if there were some errors, put the original (fake) values in place right before the result
                stack.getContext().put(ORIGINAL_PROPERTY_OVERRIDE, fakie);
                invocation.addPreResultListener(new PreResultListener() {
                    public void beforeResult(ActionInvocation invocation, String resultCode) {
                        Map<Object, Object> fakie = (Map<Object, Object>) invocation.getInvocationContext().get(ORIGINAL_PROPERTY_OVERRIDE);
    
                        if (fakie != null) {
                            invocation.getStack().setExprOverrides(fakie);
                        }
                    }
                });
            }
            return invocation.invoke();
        }
    
        protected boolean shouldAddError(String propertyName, Object value) {
            return true;
        }
    }
    View Code

    )拦截器处理的。因此,我们要处理上边的错误信息,必须从conversionError拦截器下手,来修改其源代码才可以解决上边提到的问题。

      在工程src下创建com.opensymphony.xwork2.interceptor包,在该包下创建ConversionErrorInterceptor.java类,把com.opensymphony.xwork2.interceptor.ConversionErrorInterceptor源代码拷贝到新建类中,修改类文件如下:

    /*
     * Copyright 2002-2007,2009 The Apache Software Foundation.
     * 
     * 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 com.opensymphony.xwork2.interceptor;
    
    import com.opensymphony.xwork2.Action;
    import com.opensymphony.xwork2.ActionContext;
    import com.opensymphony.xwork2.ActionInvocation;
    import com.opensymphony.xwork2.ValidationAware;
    import com.opensymphony.xwork2.conversion.impl.XWorkConverter;
    import com.opensymphony.xwork2.util.ValueStack;
    import org.apache.commons.lang3.StringEscapeUtils;
    
    import java.util.HashMap;
    import java.util.Map;
    
    /**
     * <!-- START SNIPPET: description --> ConversionErrorInterceptor adds
     * conversion errors from the ActionContext to the Action's field errors.
     *
     * <p/>
     * This interceptor adds any error found in the {@link ActionContext}'s
     * conversionErrors map as a field error (provided that the action implements
     * {@link ValidationAware}). In addition, any field that contains a validation
     * error has its original value saved such that any subsequent requests for that
     * value return the original value rather than the value in the action. This is
     * important because if the value "abc" is submitted and can't be converted to
     * an int, we want to display the original string ("abc") again rather than the
     * int value (likely 0, which would make very little sense to the user).
     *
     *
     * <!-- END SNIPPET: description -->
     *
     * <p/>
     * <u>Interceptor parameters:</u>
     *
     * <!-- START SNIPPET: parameters -->
     *
     * <ul>
     *
     * <li>None</li>
     *
     * </ul>
     *
     * <!-- END SNIPPET: parameters -->
     *
     * <p/>
     * <u>Extending the interceptor:</u>
     *
     * <p/>
     *
     * <!-- START SNIPPET: extending -->
     *
     * Because this interceptor is not web-specific, it abstracts the logic for
     * whether an error should be added. This allows for web-specific interceptors
     * to use more complex logic in the {@link #shouldAddError} method for when a
     * value has a conversion error but is null or empty or otherwise indicates that
     * the value was never actually entered by the user.
     *
     * <!-- END SNIPPET: extending -->
     *
     * <p/>
     * <u>Example code:</u>
     *
     * <pre>
     * <!-- START SNIPPET: example -->
     * &lt;action name="someAction" class="com.examples.SomeAction"&gt;
     *     &lt;interceptor-ref name="params"/&gt;
     *     &lt;interceptor-ref name="conversionError"/&gt;
     *     &lt;result name="success"&gt;good_result.ftl&lt;/result&gt;
     * &lt;/action&gt;
     * <!-- END SNIPPET: example -->
     * </pre>
     *
     * @author Jason Carreira
     */
    public class ConversionErrorInterceptor extends AbstractInterceptor {
    
        public static final String ORIGINAL_PROPERTY_OVERRIDE = "original.property.override";
    
        protected Object getOverrideExpr(ActionInvocation invocation, Object value) {
            return escape(value);
        }
    
        protected String escape(Object value) {
            return """ + StringEscapeUtils.escapeJava(String.valueOf(value)) + """;
        }
    
        @Override
        public String intercept(ActionInvocation invocation) throws Exception {
    
            ActionContext invocationContext = invocation.getInvocationContext();
            Map<String, Object> conversionErrors = invocationContext.getConversionErrors();
            ValueStack stack = invocationContext.getValueStack();
    
            HashMap<Object, Object> fakie = null;
    
            for (Map.Entry<String, Object> entry : conversionErrors.entrySet()) {
                String propertyName = entry.getKey();
                Object value = entry.getValue();
    
                if (shouldAddError(propertyName, value)) {
                    String message = XWorkConverter.getConversionErrorMessage(propertyName, stack);
    
                    Object action = invocation.getAction();
                    if (action instanceof ValidationAware) {
                        ValidationAware va = (ValidationAware) action;
                        va.addFieldError(propertyName, message);
                    }
    
                    if (fakie == null) {
                        fakie = new HashMap<Object, Object>();
                    }
    
                    fakie.put(propertyName, getOverrideExpr(invocation, value));
                }
            }
    
            if (fakie != null) {
                // if there were some errors, put the original (fake) values in
                // place right before the result
                stack.getContext().put(ORIGINAL_PROPERTY_OVERRIDE, fakie);
                invocation.addPreResultListener(new PreResultListener() {
                    public void beforeResult(ActionInvocation invocation, String resultCode) {
                        Map<Object, Object> fakie = (Map<Object, Object>) invocation.getInvocationContext()
                                .get(ORIGINAL_PROPERTY_OVERRIDE);
    
                        if (fakie != null) {
                            invocation.getStack().setExprOverrides(fakie);
                        }
                    }
                });
            }
    
            Object action = invocation.getAction();
            if (action instanceof ValidationAware) {
                ValidationAware validationAware = (ValidationAware) action;
                if (validationAware.hasFieldErrors() || validationAware.hasActionErrors()) {
                    return "input";
                }
            }
            
            return invocation.invoke();
        }
    
        protected boolean shouldAddError(String propertyName, Object value) {
            return true;
        }
    }
    View Code

      主要修改intercept方法,在返回值之前添加代码:

    1         Object action = invocation.getAction();
    2         if (action instanceof ValidationAware) {
    3             ValidationAware validationAware = (ValidationAware) action;
    4             if (validationAware.hasFieldErrors() || validationAware.hasActionErrors()) {
    5                 return "input";
    6             }
    7         }
    8         
    9         return invocation.invoke();

      基于短路验证和上边代码修改,则如果输入字符串,提交表单后,默认是会出现一个错误:字段转换失败。

    • 非字段验证:

     示例:验证密码和确认密码是否一直。

    index.jsp

        <s:actionerror />
    
        <s:form action="myValidation" method="post">
            <s:password name="password" label="Password"></s:password>
            <s:password name="password1" label="Password1"></s:password>
    
            <s:submit label="Submit"></s:submit>
        </s:form>

    MyValidationAction.java

    package com.dx.struts2.myvalidations;
    
    import com.opensymphony.xwork2.ActionSupport;
    
    public class MyValidationAction extends ActionSupport {
        private static final long serialVersionUID = 1L;
    
        private String password;
        private String password1;
    
        public String getPassword() {
            return password;
        }
    
        public void setPassword(String password) {
            this.password = password;
        }
    
        public String getPassword1() {
            return password1;
        }
    
        public void setPassword1(String password1) {
            this.password1 = password1;
        }
    
        @Override
        public String execute() throws Exception {
    
            System.out.println("execute...");
    
            return SUCCESS;
        }
    }

    MyValidationAction-myValidation-validation.xml

    <!DOCTYPE validators PUBLIC
            "-//Apache Struts//XWork Validator 1.0.2//EN"
            "http://struts.apache.org/dtds/xwork-validator-1.0.2.dtd">
    
    <validators>
        
        <validator type="expression">
            <param name="expression"><![CDATA[password==password1]]></param>
            <message>Password is not equals Password1</message>
        </validator>
    </validators>
  • 相关阅读:
    Python--IO模型
    python queue, pipe, manage
    python多线程,event,互斥锁,死锁,递归锁,信号量
    day34 异常处理、断言、socket之ftp协议
    day34 反射、面向对象内置方法:如__str__、面向对象的软件开发
    PHP基础入门(二)【PHP函数基础】
    PHP基础入门详解(一)【世界上最好用的编程语言】
    H5简单拖放(Drag/Drop)
    手机APP ~ MUI——创建页面方法
    bootstrap栅格系统
  • 原文地址:https://www.cnblogs.com/yy3b2007com/p/6690239.html
Copyright © 2020-2023  润新知