Open Source Repository

Home /spring/spring-context-3.0.5 | Repository Home



org/springframework/validation/AbstractErrors.java
/*
 * Copyright 2002-2008 the original author or authors.
 *
 * 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.springframework.validation;

import java.io.Serializable;
import java.util.Collections;
import java.util.EmptyStackException;
import java.util.LinkedList;
import java.util.List;
import java.util.Stack;

import org.springframework.util.StringUtils;

/**
 * Abstract implementation of the {@link Errors} interface. Provides common
 * access to evaluated errors; however, does not define concrete management
 * of {@link ObjectError ObjectErrors} and {@link FieldError FieldErrors}.
 *
 @author Juergen Hoeller
 @since 2.5.3
 */
public abstract class AbstractErrors implements Errors, Serializable {

  private String nestedPath = "";

  private final Stack<String> nestedPathStack = new Stack<String>();


  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 = 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() && !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 (StringUtils.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.
   <p>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 reject(String errorCode) {
    reject(errorCode, null, null);
  }

  public void reject(String errorCode, String defaultMessage) {
    reject(errorCode, null, defaultMessage);
  }

  public void rejectValue(String field, String errorCode) {
    rejectValue(field, errorCode, null, null);
  }

  public void rejectValue(String field, String errorCode, String defaultMessage) {
    rejectValue(field, errorCode, null, defaultMessage);
  }


  public boolean hasErrors() {
    return !getAllErrors().isEmpty();
  }

  public int getErrorCount() {
    return getAllErrors().size();
  }

  public List<ObjectError> getAllErrors() {
    List<ObjectError> result = new LinkedList<ObjectError>();
    result.addAll(getGlobalErrors());
    result.addAll(getFieldErrors());
    return Collections.unmodifiableList(result);
  }

  public boolean hasGlobalErrors() {
    return (getGlobalErrorCount() 0);
  }

  public int getGlobalErrorCount() {
    return getGlobalErrors().size();
  }

  public ObjectError getGlobalError() {
    List globalErrors = getGlobalErrors();
    return (!globalErrors.isEmpty() (ObjectErrorglobalErrors.get(0null);
  }

  public boolean hasFieldErrors() {
    return (getFieldErrorCount() 0);
  }

  public int getFieldErrorCount() {
    return getFieldErrors().size();
  }

  public FieldError getFieldError() {
    List fieldErrors = getFieldErrors();
    return (!fieldErrors.isEmpty() (FieldErrorfieldErrors.get(0null);
  }

  public boolean hasFieldErrors(String field) {
    return (getFieldErrorCount(field0);
  }

  public int getFieldErrorCount(String field) {
    return getFieldErrors(field).size();
  }

  public List<FieldError> getFieldErrors(String field) {
    List<FieldError> fieldErrors = getFieldErrors();
    List<FieldError> result = new LinkedList<FieldError>();
    String fixedField = fixedField(field);
    for (FieldError error : fieldErrors) {
      if (isMatchingFieldError(fixedField, error)) {
        result.add(error);
      }
    }
    return Collections.unmodifiableList(result);
  }

  public FieldError getFieldError(String field) {
    List fieldErrors = getFieldErrors(field);
    return (!fieldErrors.isEmpty() (FieldErrorfieldErrors.get(0null);
  }


  public Class getFieldType(String field) {
    Object value = getFieldValue(field);
    if (value != null) {
      return value.getClass();
    }
    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))));
  }


  @Override
  public String toString() {
    StringBuilder sb = new StringBuilder(getClass().getName());
    sb.append(": ").append(getErrorCount()).append(" errors");
    for (ObjectError error : getAllErrors()) {
      sb.append('\n').append(error);
    }
    return sb.toString();
  }

}