All Downloads are FREE. Search and download functionalities are using the official Maven repository.

org.apache.struts2.json.JSONValidationInterceptor Maven / Gradle / Ivy

There is a newer version: 7.0.0
Show newest version
/*
 * $Id$
 *
 * 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.struts2.json;

import com.opensymphony.xwork2.Action;
import com.opensymphony.xwork2.ActionInvocation;
import com.opensymphony.xwork2.ModelDriven;
import com.opensymphony.xwork2.interceptor.ValidationAware;
import com.opensymphony.xwork2.interceptor.MethodFilterInterceptor;
import org.apache.logging.log4j.Logger;
import org.apache.logging.log4j.LogManager;
import org.apache.commons.lang3.StringEscapeUtils;
import org.apache.struts2.ServletActionContext;

import javax.servlet.http.HttpServletRequest;
import javax.servlet.http.HttpServletResponse;
import java.io.IOException;
import java.util.Collection;
import java.util.List;
import java.util.Map;

/**
 * 

Serializes validation and action errors into JSON. This interceptor does not * perform any validation, so it must follow the 'validation' interceptor on the stack. *

* *

This stack (defined in struts-default.xml) shows how to use this interceptor with the * 'validation' interceptor

*
 * <interceptor-stack name="jsonValidationWorkflowStack">
 *      <interceptor-ref name="basicStack"/>
 *      <interceptor-ref name="validation">
 *            <param name="excludeMethods">input,back,cancel</param>
 *      </interceptor-ref>
 *      <interceptor-ref name="jsonValidation"/>
 *      <interceptor-ref name="workflow"/>
 * </interceptor-stack>
 * 
*

If 'validationFailedStatus' is set it will be used as the Response status * when validation fails.

* *

If the request has a parameter 'struts.validateOnly' execution will return after * validation (action won't be executed).

* *

If 'struts.validateOnly' is set to false you may want to use {@link JSONActionRedirectResult}.

* *

A request parameter named 'struts.enableJSONValidation' must be set to 'true' to * use this interceptor

* *

If the request has a parameter 'struts.JSONValidation.set.encoding' set to true * the character encoding will NOT be set on the response - is needed in portlet environment * - for more details see issue WW-3237

*/ public class JSONValidationInterceptor extends MethodFilterInterceptor { private static final Logger LOG = LogManager.getLogger(JSONValidationInterceptor.class); public static final String VALIDATE_ONLY_PARAM = "struts.validateOnly"; public static final String VALIDATE_JSON_PARAM = "struts.enableJSONValidation"; public static final String NO_ENCODING_SET_PARAM = "struts.JSONValidation.no.encoding"; public static final String DEFAULT_ENCODING = "UTF-8"; private int validationFailedStatus = -1; /** * HTTP status that will be set in the response if validation fails * * @param validationFailedStatus validation failed status */ public void setValidationFailedStatus(int validationFailedStatus) { this.validationFailedStatus = validationFailedStatus; } @Override protected String doIntercept(ActionInvocation invocation) throws Exception { HttpServletResponse response = ServletActionContext.getResponse(); HttpServletRequest request = ServletActionContext.getRequest(); Object action = invocation.getAction(); if (isJsonEnabled(request)) { if (action instanceof ValidationAware) { // generate json ValidationAware validationAware = (ValidationAware) action; if (validationAware.hasErrors()) { return generateJSON(request, response, validationAware); } } if (isValidateOnly(request)) { //there were no errors setupEncoding(response, request); response.getWriter().print("{}"); response.setContentType("application/json"); return Action.NONE; } else { return invocation.invoke(); } } else return invocation.invoke(); } private void setupEncoding(HttpServletResponse response, HttpServletRequest request) { if (isSetEncoding(request)) { if (LOG.isDebugEnabled()) { LOG.debug("Default encoding not set!"); } } else { if (LOG.isDebugEnabled()) { LOG.debug("Setting up encoding to: [" + DEFAULT_ENCODING + "]!"); } response.setCharacterEncoding(DEFAULT_ENCODING); } } private String generateJSON(HttpServletRequest request, HttpServletResponse response, ValidationAware validationAware) throws IOException { if (validationFailedStatus >= 0) { response.setStatus(validationFailedStatus); } setupEncoding(response, request); response.getWriter().print(buildResponse(validationAware)); response.setContentType("application/json"); return Action.NONE; } private boolean isJsonEnabled(HttpServletRequest request) { return "true".equals(request.getParameter(VALIDATE_JSON_PARAM)); } private boolean isValidateOnly(HttpServletRequest request) { return "true".equals(request.getParameter(VALIDATE_ONLY_PARAM)); } private boolean isSetEncoding(HttpServletRequest request) { return "true".equals(request.getParameter(NO_ENCODING_SET_PARAM)); } /** * @param validationAware the validation aware object * @return JSON string that contains the errors and field errors */ @SuppressWarnings("unchecked") protected String buildResponse(ValidationAware validationAware) { //should we use FreeMarker here? StringBuilder sb = new StringBuilder(); sb.append("{ "); if (validationAware.hasErrors()) { //action errors if (validationAware.hasActionErrors()) { sb.append("\"errors\":"); sb.append(buildArray(validationAware.getActionErrors())); } //field errors if (validationAware.hasFieldErrors()) { if (validationAware.hasActionErrors()) sb.append(","); sb.append("\"fieldErrors\": {"); Map> fieldErrors = validationAware .getFieldErrors(); for (Map.Entry> fieldError : fieldErrors .entrySet()) { sb.append("\""); //if it is model driven, remove "model." see WW-2721 String fieldErrorKey = fieldError.getKey(); sb.append(((validationAware instanceof ModelDriven) && fieldErrorKey.startsWith("model."))? fieldErrorKey.substring(6) : fieldErrorKey); sb.append("\":"); sb.append(buildArray(fieldError.getValue())); sb.append(","); } //remove trailing comma, IE creates an empty object, duh sb.deleteCharAt(sb.length() - 1); sb.append("}"); } } sb.append("}"); /*response should be something like: * { * "errors": ["this", "that"], * "fieldErrors": { * field1: "this", * field2: "that" * } * } */ return sb.toString(); } private String buildArray(Collection values) { StringBuilder sb = new StringBuilder(); sb.append("["); for (String value : values) { sb.append("\""); sb.append(StringEscapeUtils.escapeJson(value)); sb.append("\","); } if (values.size() > 0) sb.deleteCharAt(sb.length() - 1); sb.append("]"); return sb.toString(); } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy