org.frameworkset.web.bind.BindingResultImpl Maven / Gradle / Ivy
Show all versions of bboss-mvc Show documentation
/*
* Copyright 2008 biaoping.yin
*
* 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 org.frameworkset.web.bind;
import java.util.Collections;
import java.util.EmptyStackException;
import java.util.HashMap;
import java.util.Iterator;
import java.util.LinkedList;
import java.util.List;
import java.util.Map;
import java.util.Stack;
import org.frameworkset.spi.support.bind.BindingErrorProcessor;
import org.frameworkset.spi.support.bind.DefaultBindingErrorProcessor;
import org.frameworkset.spi.support.validate.BindingResult;
import org.frameworkset.spi.support.validate.DefaultMessageCodesResolver;
import org.frameworkset.spi.support.validate.Errors;
import org.frameworkset.spi.support.validate.FieldError;
import org.frameworkset.spi.support.validate.MessageCodesResolver;
import org.frameworkset.spi.support.validate.ObjectError;
import com.frameworkset.util.StringUtil;
/**
* Title: BindingResultImpl.java
* Description:
* bboss workgroup
* Copyright (c) 2008
* @Date 2010-12-24
* @author biaoping.yin
* @version 1.0
*/
public class BindingResultImpl implements BindingResult {
// private Map errors = new HashMap();
// private Map model;
private Object target;
private MessageCodesResolver messageCodesResolver = new DefaultMessageCodesResolver();
private final List errors = new LinkedList();
private BindingErrorProcessor bindingErrorProcessor;
private String objectName;
private String nestedPath = "";
private final Stack nestedPathStack = new Stack();
public BindingResultImpl(String objectName,DefaultBindingErrorProcessor bindingErrorProcessor)
{
this.objectName = objectName;
this.bindingErrorProcessor = bindingErrorProcessor;
}
/**
* Create a new AbstractBindingResult instance.
* @param objectName the name of the target object
* @see DefaultMessageCodesResolver
*/
public BindingResultImpl(String objectName) {
this.objectName = objectName == null?"command":objectName;
this.bindingErrorProcessor = new DefaultBindingErrorProcessor();
}
/**
* Set the strategy to use for resolving errors into message codes.
* Default is DefaultMessageCodesResolver.
* @see DefaultMessageCodesResolver
*/
public void setMessageCodesResolver(MessageCodesResolver messageCodesResolver) {
this.messageCodesResolver = messageCodesResolver;
}
/**
* Return the strategy to use for resolving errors into message codes.
*/
public MessageCodesResolver getMessageCodesResolver() {
return this.messageCodesResolver;
}
//---------------------------------------------------------------------
// Implementation of the Errors interface
//---------------------------------------------------------------------
public String getObjectName() {
return this.objectName;
}
public void setNestedPath(String nestedPath) {
doSetNestedPath(nestedPath);
this.nestedPathStack.clear();
}
public String getNestedPath() {
return this.nestedPath;
}
public void pushNestedPath(String subPath) {
this.nestedPathStack.push(getNestedPath());
doSetNestedPath(getNestedPath() + subPath);
}
public void popNestedPath() throws IllegalArgumentException {
try {
String formerNestedPath = (String) this.nestedPathStack.pop();
doSetNestedPath(formerNestedPath);
}
catch (EmptyStackException ex) {
throw new IllegalStateException("Cannot pop nested path: no nested path on stack");
}
}
/**
* Actually set the nested path.
* Delegated to by setNestedPath and pushNestedPath.
*/
protected void doSetNestedPath(String nestedPath) {
if (nestedPath == null) {
nestedPath = "";
}
nestedPath = canonicalFieldName(nestedPath);
if (nestedPath.length() > 0 && !nestedPath.endsWith(Errors.NESTED_PATH_SEPARATOR)) {
nestedPath += Errors.NESTED_PATH_SEPARATOR;
}
this.nestedPath = nestedPath;
}
/**
* Transform the given field into its full path,
* regarding the nested path of this instance.
*/
protected String fixedField(String field) {
if (StringUtil.hasLength(field)) {
return getNestedPath() + canonicalFieldName(field);
}
else {
String path = getNestedPath();
return (path.endsWith(Errors.NESTED_PATH_SEPARATOR) ?
path.substring(0, path.length() - NESTED_PATH_SEPARATOR.length()) : path);
}
}
/**
* Determine the canonical field name for the given field.
* The default implementation simply returns the field name as-is.
* @param field the original field name
* @return the canonical field name
*/
protected String canonicalFieldName(String field) {
return field;
}
public void rejectValueWithErrorArgs(String field, String errorCode, Object[] errorArgs) {
rejectValueWithErrorArgs( field, errorCode, errorArgs, null);
}
public void rejectValueWithErrorArgs(String field, String errorCode, Object[] errorArgs, String defaultMessage) {
if ("".equals(getNestedPath()) && !StringUtil.hasLength(field)) {
// We're at the top of the nested object hierarchy,
// so the present level is not a field but rather the top object.
// The best we can do is register a global error here...
rejectWithErrorArgs(errorCode, errorArgs, defaultMessage);
return;
}
String fixedField = fixedField(field);
Object newVal = null;//getActualFieldValue(fixedField);
// FieldError fe = new FieldError(
// getObjectName(), fixedField, newVal, null,false,
// resolveMessageCodes(errorCode, field,null), errorArgs, defaultMessage);
FieldError fe = new FieldError(
getObjectName(), fixedField, newVal, null,false,
new String[]{errorCode}, errorArgs, defaultMessage);
addError(fe);
}
public void rejectValueWithErrorArgs(String field, String errorCode, Object[] errorArgs,String value,Class type, String defaultMessage) {
if ("".equals(getNestedPath()) && !StringUtil.hasLength(field)) {
// We're at the top of the nested object hierarchy,
// so the present level is not a field but rather the top object.
// The best we can do is register a global error here...
rejectWithErrorArgs(errorCode, errorArgs, defaultMessage);
return;
}
String fixedField = fixedField(field);
Object newVal = value;//getActualFieldValue(fixedField);
// FieldError fe = new FieldError(
// getObjectName(), fixedField, newVal,type, false,
// resolveMessageCodes(errorCode, field, type), errorArgs, defaultMessage);
FieldError fe = new FieldError(
getObjectName(), fixedField, newVal,type, false,
new String[]{errorCode}, errorArgs, defaultMessage);
addError(fe);
}
public void rejectValueWithErrorArgs(String field, String errorCode, Object[] errorArgs,String[] value,Class type, String defaultMessage) {
if ("".equals(getNestedPath()) && !StringUtil.hasLength(field)) {
// We're at the top of the nested object hierarchy,
// so the present level is not a field but rather the top object.
// The best we can do is register a global error here...
rejectWithErrorArgs(errorCode, errorArgs, defaultMessage);
return;
}
String fixedField = fixedField(field);
Object newVal = value;//getActualFieldValue(fixedField);
// FieldError fe = new FieldError(
// getObjectName(), fixedField, newVal,type, false,
// resolveMessageCodes(errorCode, field,type), errorArgs, defaultMessage);
FieldError fe = new FieldError(
getObjectName(), fixedField, newVal,type, false,
new String[]{errorCode}, errorArgs, defaultMessage);
addError(fe);
}
/* (non-Javadoc)
* @see org.frameworkset.spi.support.validate.Errors#rejectValue(java.lang.String, java.lang.String, java.lang.String, java.lang.String)
*/
public void rejectValue(String field, String errorCode, String rejectvalue,
String defaultMessage) {
rejectValueWithErrorArgs( field, errorCode, null,rejectvalue,null, defaultMessage);
}
/* (non-Javadoc)
* @see org.frameworkset.spi.support.validate.Errors#rejectValue(java.lang.String, java.lang.String, java.lang.String[], java.lang.String)
*/
public void rejectValue(String field, String errorCode,
String[] rejectvalue, String defaultMessage) {
rejectValueWithErrorArgs( field, errorCode, null,rejectvalue,null, defaultMessage);
}
/* (non-Javadoc)
* @see org.frameworkset.spi.support.validate.Errors#rejectValue(java.lang.String, java.lang.String, java.lang.String, java.lang.Class, java.lang.String)
*/
public void rejectValue(String field, String errorCode, String rejectvalue,
Class fieldtype, String defaultMessage) {
rejectValueWithErrorArgs( field, errorCode, null,rejectvalue,fieldtype, defaultMessage);
}
/* (non-Javadoc)
* @see org.frameworkset.spi.support.validate.Errors#rejectValue(java.lang.String, java.lang.String, java.lang.String[], java.lang.Class, java.lang.String)
*/
public void rejectValue(String field, String errorCode,
String[] rejectvalue, Class fieldtype, String defaultMessage) {
rejectValueWithErrorArgs( field, errorCode, null,rejectvalue,fieldtype, defaultMessage);
}
public void addError(ObjectError error) {
this.errors.add(error);
}
public List getAllErrors() {
return Collections.unmodifiableList(this.errors);
}
public void addAllErrors(Errors errors) {
if (!errors.getObjectName().equals(getObjectName())) {
throw new IllegalArgumentException("Errors object needs to have same object name");
}
this.errors.addAll(errors.getAllErrors());
}
/**
* Resolve the given error code into message codes.
* Calls the MessageCodesResolver with appropriate parameters.
* @param errorCode the error code to resolve into message codes
* @return the resolved message codes
* @see #setMessageCodesResolver
*/
public String[] resolveMessageCodes(String errorCode) {
return getMessageCodesResolver().resolveMessageCodes(errorCode, getObjectName());
}
public String[] resolveMessageCodes(String errorCode, String field,Class fieldType) {
String fixedField = fixedField(field);
// Class fieldType = getFieldType(fixedField);
return getMessageCodesResolver().resolveMessageCodes(errorCode, getObjectName(), fixedField, fieldType);
}
public boolean hasErrors() {
return !this.errors.isEmpty();
}
public int getErrorCount() {
return this.errors.size();
}
public List getGlobalErrors() {
List result = new LinkedList();
for (Iterator it = this.errors.iterator(); it.hasNext();) {
Object error = it.next();
if (!(error instanceof FieldError)) {
result.add(error);
}
}
return Collections.unmodifiableList(result);
}
// public ObjectError getGlobalError() {
// for (Iterator it = this.errors.iterator(); it.hasNext();) {
// ObjectError objectError = (ObjectError) it.next();
// if (!(objectError instanceof FieldError)) {
// return objectError;
// }
// }
// return null;
// }
public List getFieldErrors() {
List result = new LinkedList();
for (Iterator it = this.errors.iterator(); it.hasNext();) {
Object error = it.next();
if (error instanceof FieldError) {
result.add(error);
}
}
return Collections.unmodifiableList(result);
}
public FieldError getFieldError() {
for (Iterator it = this.errors.iterator(); it.hasNext();) {
Object error = it.next();
if (error instanceof FieldError) {
return (FieldError) error;
}
}
return null;
}
public List getFieldErrors(String field) {
List result = new LinkedList();
String fixedField = fixedField(field);
for (Iterator it = this.errors.iterator(); it.hasNext();) {
Object error = it.next();
if (error instanceof FieldError && isMatchingFieldError(fixedField, (FieldError) error)) {
result.add(error);
}
}
return Collections.unmodifiableList(result);
}
public FieldError getFieldError(String field) {
String fixedField = fixedField(field);
for (Iterator it = this.errors.iterator(); it.hasNext();) {
Object error = it.next();
if (error instanceof FieldError) {
FieldError fe = (FieldError) error;
if (isMatchingFieldError(fixedField, fe)) {
return fe;
}
}
}
return null;
}
/**
* Check whether the given FieldError matches the given field.
* @param field the field that we are looking up FieldErrors for
* @param fieldError the candidate FieldError
* @return whether the FieldError matches the given field
*/
protected boolean isMatchingFieldError(String field, FieldError fieldError) {
return (field.equals(fieldError.getField()) ||
(field.endsWith("*") && fieldError.getField().startsWith(field.substring(0, field.length() - 1))));
}
public String toString() {
StringBuilder sb = new StringBuilder(getClass().getName());
sb.append(": ").append(getErrorCount()).append(" errors");
Iterator it = getAllErrors().iterator();
while (it.hasNext()) {
sb.append('\n').append(it.next());
}
return sb.toString();
}
public Object getFieldValue(String field) {
FieldError fe = getFieldError(field);
// Use rejected value in case of error, current bean property value else.
Object value = null;
if (fe != null) {
value = fe.getRejectedValue();
}
// Apply formatting, but not on binding failures like type mismatches.
if (fe == null || !fe.isBindingFailure()) {
value = formatFieldValue(field, value);
}
return value;
}
//---------------------------------------------------------------------
// Implementation of BindingResult interface
//---------------------------------------------------------------------
/**
* Return a model Map for the obtained state, exposing an Errors
* instance as '{@link #MODEL_KEY_PREFIX MODEL_KEY_PREFIX} + objectName'
* and the object itself.
*
Note that the Map is constructed every time you're calling this method.
* Adding things to the map and then re-calling this method will not work.
*
The attributes in the model Map returned by this method are usually
* included in the ModelAndView for a form view that uses 's bind tag,
* which needs access to the Errors instance. 's SimpleFormController
* will do this for you when rendering its form or success view. When
* building the ModelAndView yourself, you need to include the attributes
* from the model Map returned by this method yourself.
* @see #getObjectName
* @see #MODEL_KEY_PREFIX
*/
public Map getModel() {
Map model = new HashMap(2);
// Errors instance, even if no errors.
model.put(MODEL_KEY_PREFIX + getObjectName(), this);
// Mapping from name to target object.
model.put(getObjectName(), getTarget());
return model;
}
// public Object getRawFieldValue(String field) {
// return getActualFieldValue(fixedField(field));
// }
// /**
// * This implementation delegates to the
// * {@link #getPropertyEditorRegistry() PropertyEditorRegistry}'s
// * editor lookup facility, if available.
// */
// public EditorInf findEditor(String field, Class valueType) {
// PropertyEditorRegistry editorRegistry = getPropertyEditorRegistry();
// if (editorRegistry != null) {
// Class valueTypeToUse = valueType;
//// if (valueTypeToUse == null) {
//// valueTypeToUse = getFieldType(field);
//// }
// return editorRegistry.findCustomEditor(valueTypeToUse, fixedField(field));
// }
// else {
// return null;
// }
// }
//
// /**
// * This implementation returns null
.
// */
// public PropertyEditorRegistry getPropertyEditorRegistry() {
// return null;
// }
// /**
// * Mark the specified disallowed field as suppressed.
// *
The data binder invokes this for each field value that was
// * detected to target a disallowed field.
// * @see DataBinder#setAllowedFields
// */
// public void recordSuppressedField(String field) {
// this.suppressedFields.add(field);
// }
// /**
// * Return the list of fields that were suppressed during the bind process.
// *
Can be used to determine whether any field values were targetting
// * disallowed fields.
// * @see DataBinder#setAllowedFields
// */
// public String[] getSuppressedFields() {
// return StringUtil.toStringArray(this.suppressedFields);
// }
public boolean equals(Object other) {
if (this == other) {
return true;
}
if (!(other instanceof BindingResult)) {
return false;
}
BindingResult otherResult = (BindingResult) other;
return (getObjectName().equals(otherResult.getObjectName()) &&
getTarget().equals(otherResult.getTarget()) &&
getAllErrors().equals(otherResult.getAllErrors()));
}
public int hashCode() {
return getObjectName().hashCode() * 29 + getTarget().hashCode();
}
//---------------------------------------------------------------------
// Template methods to be implemented/overridden by subclasses
//---------------------------------------------------------------------
/**
* Return the wrapped target object.
*/
public Object getTarget()
{
return this.target;
}
/**
* Format the given value for the specified field.
*
The default implementation simply returns the field value as-is.
* @param field the field to check
* @param value the value of the field (either a rejected value
* other than from a binding error, or an actual field value)
* @return the formatted value
*/
protected Object formatFieldValue(String field, Object value) {
return value;
}
public void addError(FieldError error) {
this.addError((ObjectError)error);
}
public boolean hasGlobalErrors() {
return (getGlobalErrorCount() > 0);
}
public int getGlobalErrorCount() {
return getGlobalErrors().size();
}
public ObjectError getGlobalError() {
List globalErrors = getGlobalErrors();
return (!globalErrors.isEmpty() ? (ObjectError) globalErrors.get(0) : null);
}
public boolean hasFieldErrors() {
return (getFieldErrorCount() > 0);
}
public int getFieldErrorCount() {
return getFieldErrors().size();
}
// public FieldError getFieldError() {
// List fieldErrors = getFieldErrors();
// return (!fieldErrors.isEmpty() ? (FieldError) fieldErrors.get(0) : null);
// }
public boolean hasFieldErrors(String field) {
return (getFieldErrorCount(field) > 0);
}
public int getFieldErrorCount(String field) {
return getFieldErrors(field).size();
}
// public List getFieldErrors(String field) {
// List fieldErrors = getFieldErrors();
// List result = new LinkedList();
// String fixedField = fixedField(field);
// for (Iterator it = fieldErrors.iterator(); it.hasNext();) {
// Object error = it.next();
// if (isMatchingFieldError(fixedField, (FieldError) error)) {
// result.add(error);
// }
// }
// return Collections.unmodifiableList(result);
// }
// public FieldError getFieldError(String field) {
// List fieldErrors = getFieldErrors(field);
// return (!fieldErrors.isEmpty() ? (FieldError) fieldErrors.get(0) : null);
// }
public void rejectValue(String field, String errorCode) {
rejectValue( field, errorCode,
null);
}
public void rejectValue(String field, String errorCode, String defaultMessage) {
rejectValueWithErrorArgs(field, errorCode, (Object[])null, defaultMessage);
}
public void rejectWithErrorArgs(String errorCode, Object[] errorArgs, String defaultMessage) {
// addError(new ObjectError(getObjectName(), resolveMessageCodes(errorCode), errorArgs, defaultMessage));
addError(new ObjectError(getObjectName(),new String[]{errorCode}, errorArgs, defaultMessage));
}
public void reject(String errorCode) {
reject( errorCode, null);
}
public void rejectWithErrorArgs(String errorCode, Object[] errorArgs) {
// addError(new ObjectError(getObjectName(), resolveMessageCodes(errorCode), errorArgs, defaultMessage));
addError(new ObjectError(getObjectName(),new String[]{errorCode}, errorArgs, null));
}
public void reject(String errorCode, String defaultMessage) {
rejectWithErrorArgs( errorCode, (Object[] )null, defaultMessage);
}
}