Open Source Repository

Home /spring/spring-core-3.0.5 | Repository Home



org/springframework/core/io/support/ResourceArrayPropertyEditor.java
/*
 * Copyright 2002-2010 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.core.io.support;

import java.beans.PropertyEditorSupport;
import java.io.IOException;
import java.util.ArrayList;
import java.util.Arrays;
import java.util.Collection;
import java.util.List;

import org.apache.commons.logging.Log;
import org.apache.commons.logging.LogFactory;

import org.springframework.core.io.Resource;
import org.springframework.util.SystemPropertyUtils;

/**
 * Editor for {@link org.springframework.core.io.Resource} arrays, to
 * automatically convert <code>String</code> location patterns
 * (e.g. <code>"file:C:/my*.txt"</code> or <code>"classpath*:myfile.txt"</code>)
 * to <code>Resource</code> array properties. Can also translate a collection
 * or array of location patterns into a merged Resource array.
 *
 <p>The path may contain <code>${...}</code> placeholders,
 * to be resolved as system properties: e.g. <code>${user.dir}</code>.
 * Unresolvable placeholder are ignored by default.
 *
 <p>Delegates to a {@link ResourcePatternResolver},
 * by default using a {@link PathMatchingResourcePatternResolver}.
 *
 @author Juergen Hoeller
 @since 1.1.2
 @see org.springframework.core.io.Resource
 @see ResourcePatternResolver
 @see PathMatchingResourcePatternResolver
 @see org.springframework.util.SystemPropertyUtils#resolvePlaceholders
 @see System#getProperty(String)
 */
public class ResourceArrayPropertyEditor extends PropertyEditorSupport {

  private static final Log logger = LogFactory.getLog(ResourceArrayPropertyEditor.class);

  private final ResourcePatternResolver resourcePatternResolver;

  private final boolean ignoreUnresolvablePlaceholders;


  /**
   * Create a new ResourceArrayPropertyEditor with a default
   * PathMatchingResourcePatternResolver.
   @see PathMatchingResourcePatternResolver
   */
  public ResourceArrayPropertyEditor() {
    this(new PathMatchingResourcePatternResolver());
  }

  /**
   * Create a new ResourceArrayPropertyEditor with the given ResourcePatternResolver.
   @param resourcePatternResolver the ResourcePatternResolver to use
   */
  public ResourceArrayPropertyEditor(ResourcePatternResolver resourcePatternResolver) {
    this(resourcePatternResolver, true);
  }

  /**
   * Create a new ResourceArrayPropertyEditor with the given ResourcePatternResolver.
   @param resourcePatternResolver the ResourcePatternResolver to use
   @param ignoreUnresolvablePlaceholders whether to ignore unresolvable placeholders
   * if no corresponding system property could be found
   */
  public ResourceArrayPropertyEditor(ResourcePatternResolver resourcePatternResolver, boolean ignoreUnresolvablePlaceholders) {
    this.resourcePatternResolver = resourcePatternResolver;
    this.ignoreUnresolvablePlaceholders = ignoreUnresolvablePlaceholders;
  }


  /**
   * Treat the given text as location pattern and convert it to a Resource array.
   */
  @Override
  public void setAsText(String text) {
    String pattern = resolvePath(text).trim();
    try {
      setValue(this.resourcePatternResolver.getResources(pattern));
    }
    catch (IOException ex) {
      throw new IllegalArgumentException(
          "Could not resolve resource location pattern [" + pattern + "]: " + ex.getMessage());
    }
  }

  /**
   * Treat the given value as collection or array and convert it to a Resource array.
   * Considers String elements as location patterns, and takes Resource elements as-is.
   */
  @Override
  public void setValue(Object valuethrows IllegalArgumentException {
    if (value instanceof Collection || (value instanceof Object[] && !(value instanceof Resource[]))) {
      Collection input = (value instanceof Collection ? (Collectionvalue : Arrays.asList((Object[]) value));
      List<Resource> merged = new ArrayList<Resource>();
      for (Object element : input) {
        if (element instanceof String) {
          // A location pattern: resolve it into a Resource array.
          // Might point to a single resource or to multiple resources.
          String pattern = resolvePath((Stringelement).trim();
          try {
            Resource[] resources = this.resourcePatternResolver.getResources(pattern);
            for (Resource resource : resources) {
              if (!merged.contains(resource)) {
                merged.add(resource);
              }
            }
          }
          catch (IOException ex) {
            // ignore - might be an unresolved placeholder or non-existing base directory
            if (logger.isDebugEnabled()) {
              logger.debug("Could not retrieve resources for pattern '" + pattern + "': " + ex);
            }
          }
        }
        else if (element instanceof Resource) {
          // A Resource object: add it to the result.
          Resource resource = (Resourceelement;
          if (!merged.contains(resource)) {
            merged.add(resource);
          }
        }
        else {
          throw new IllegalArgumentException("Cannot convert element [" + element + "] to [" +
              Resource.class.getName() "]: only location String and Resource object supported");
        }
      }
      super.setValue(merged.toArray(new Resource[merged.size()]));
    }

    else {
      // An arbitrary value: probably a String or a Resource array.
      // setAsText will be called for a String; a Resource array will be used as-is.
      super.setValue(value);
    }
  }

  /**
   * Resolve the given path, replacing placeholders with
   * corresponding system property values if necessary.
   @param path the original file path
   @return the resolved file path
   @see org.springframework.util.SystemPropertyUtils#resolvePlaceholders
   */
  protected String resolvePath(String path) {
    return SystemPropertyUtils.resolvePlaceholders(path, this.ignoreUnresolvablePlaceholders);
  }

}