001 /** 002 * Copyright 2005-2013 The Kuali Foundation 003 * 004 * Licensed under the Educational Community License, Version 2.0 (the "License"); 005 * you may not use this file except in compliance with the License. 006 * You may obtain a copy of the License at 007 * 008 * http://www.opensource.org/licenses/ecl2.php 009 * 010 * Unless required by applicable law or agreed to in writing, software 011 * distributed under the License is distributed on an "AS IS" BASIS, 012 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. 013 * See the License for the specific language governing permissions and 014 * limitations under the License. 015 */ 016 package org.kuali.rice.krad.service.impl; 017 018 import org.apache.commons.lang.StringUtils; 019 import org.apache.commons.logging.Log; 020 import org.apache.commons.logging.LogFactory; 021 import org.apache.log4j.Logger; 022 import org.kuali.rice.core.api.config.property.ConfigContext; 023 import org.kuali.rice.core.api.config.property.ConfigurationService; 024 import org.kuali.rice.core.api.util.Truth; 025 import org.kuali.rice.kns.web.struts.action.KualiPropertyMessageResources; 026 import org.kuali.rice.kns.web.struts.action.KualiPropertyMessageResourcesFactory; 027 import org.kuali.rice.krad.exception.DuplicateKeyException; 028 import org.kuali.rice.krad.exception.PropertiesException; 029 030 import java.io.IOException; 031 import java.io.InputStream; 032 import java.net.URL; 033 import java.util.Collections; 034 import java.util.Iterator; 035 import java.util.Map; 036 import java.util.Properties; 037 038 /** 039 * Implementation of the {@link ConfigurationService} that loads messages from the configured rice resource 040 * files and stores them in an internal property holder 041 * 042 * @author Kuali Rice Team (rice.collab@kuali.org) 043 */ 044 public class ConfigurationServiceImpl implements ConfigurationService { 045 private final PropertyHolder propertyHolder = new PropertyHolder(); 046 047 /** 048 * Default constructor 049 */ 050 public ConfigurationServiceImpl() { 051 this.propertyHolder.getHeldProperties().putAll(ConfigContext.getCurrentContextConfig().getProperties()); 052 053 KualiPropertyMessageResourcesFactory propertyMessageFactory = new KualiPropertyMessageResourcesFactory(); 054 055 // create default KualiPropertyMessageResources 056 KualiPropertyMessageResources messageResources = 057 (KualiPropertyMessageResources) propertyMessageFactory.createResources(""); 058 059 //Add Kuali Properties to property holder 060 this.propertyHolder.getHeldProperties().putAll(messageResources.getKualiProperties(null)); 061 } 062 063 /** 064 * @see org.kuali.rice.core.api.config.property.ConfigurationService#getPropertyValueAsString(java.lang.String) 065 */ 066 @Override 067 public String getPropertyValueAsString(String key) { 068 if (key == null) { 069 throw new IllegalArgumentException("invalid (null) key"); 070 } 071 072 return this.propertyHolder.getProperty(key); 073 } 074 075 /** 076 * @see org.kuali.rice.core.api.config.property.ConfigurationService#getPropertyValueAsBoolean(java.lang.String) 077 */ 078 @Override 079 public boolean getPropertyValueAsBoolean(String key) { 080 if (key == null) { 081 throw new IllegalArgumentException("invalid (null) key"); 082 } 083 084 String property = this.propertyHolder.getProperty(key); 085 Boolean b = Truth.strToBooleanIgnoreCase(property); 086 if (b == null) { 087 return false; 088 } 089 090 return b; 091 } 092 093 /** 094 * @see org.kuali.rice.core.api.config.property.ConfigurationService#getAllProperties() 095 */ 096 @Override 097 public Map<String, String> getAllProperties() { 098 return (Map) Collections.unmodifiableMap(propertyHolder.getHeldProperties()); 099 } 100 101 /** 102 * Interface for a source for properties 103 */ 104 protected static interface PropertySource { 105 106 /** 107 * @return Properties loaded from this PropertySource 108 * @throws org.kuali.rice.krad.exception.PropertiesException if there's a problem loading the properties 109 */ 110 public Properties loadProperties(); 111 } 112 113 /** 114 * This class is a Property container. It is able to load properties from various property-sources. 115 */ 116 protected static class PropertyHolder { 117 private static Logger LOG = Logger.getLogger(PropertyHolder.class); 118 119 Properties heldProperties; 120 121 /** 122 * Default constructor. 123 */ 124 public PropertyHolder() { 125 this.heldProperties = new Properties(); 126 } 127 128 /** 129 * @return true if this container currently has no properties 130 */ 131 public boolean isEmpty() { 132 return this.heldProperties.isEmpty(); 133 } 134 135 /** 136 * @param key 137 * @return true if a property with the given key exists in this container 138 * @throws IllegalArgumentException if the given key is null 139 */ 140 public boolean containsKey(String key) { 141 validateKey(key); 142 143 return this.heldProperties.containsKey(key); 144 } 145 146 /** 147 * @param key 148 * @return the current value of the property with the given key, or null if no property exists with that key 149 * @throws IllegalArgumentException if the given key is null 150 */ 151 public String getProperty(String key) { 152 validateKey(key); 153 154 return this.heldProperties.getProperty(key); 155 } 156 157 /** 158 * Associates the given value with the given key 159 * 160 * @param key 161 * @param value 162 * @throws IllegalArgumentException if the given key is null 163 * @throws IllegalArgumentException if the given value is null 164 * @throws org.kuali.rice.krad.exception.DuplicateKeyException if a property with the given key already exists 165 */ 166 public void setProperty(String key, String value) { 167 setProperty(null, key, value); 168 } 169 170 /** 171 * Associates the given value with the given key 172 * 173 * @param source 174 * @param key 175 * @param value 176 * @throws IllegalArgumentException if the given key is null 177 * @throws IllegalArgumentException if the given value is null 178 * @throws org.kuali.rice.krad.exception.DuplicateKeyException if a property with the given key already exists 179 */ 180 public void setProperty(PropertySource source, String key, String value) { 181 validateKey(key); 182 validateValue(value); 183 184 if (containsKey(key)) { 185 if (source != null && source instanceof FilePropertySource && ((FilePropertySource) source) 186 .isAllowOverrides()) { 187 LOG.info("Duplicate Key: Override is enabled [key=" 188 + key 189 + ", new value=" 190 + value 191 + ", old value=" 192 + this.heldProperties.getProperty(key) 193 + "]"); 194 } else { 195 throw new DuplicateKeyException("duplicate key '" + key + "'"); 196 } 197 } 198 this.heldProperties.setProperty(key, value); 199 } 200 201 /** 202 * Removes the property with the given key from this container 203 * 204 * @param key 205 * @throws IllegalArgumentException if the given key is null 206 */ 207 public void clearProperty(String key) { 208 validateKey(key); 209 210 this.heldProperties.remove(key); 211 } 212 213 /** 214 * Copies all name,value pairs from the given PropertySource instance into this container. 215 * 216 * @param source 217 * @throws IllegalStateException if the source is invalid (improperly initialized) 218 * @throws org.kuali.rice.krad.exception.DuplicateKeyException the first time a given property has the same key 219 * as an existing property 220 * @throws org.kuali.rice.krad.exception.PropertiesException if unable to load properties from the given source 221 */ 222 public void loadProperties(PropertySource source) { 223 if (source == null) { 224 throw new IllegalArgumentException("invalid (null) source"); 225 } 226 227 Properties newProperties = source.loadProperties(); 228 229 for (Iterator i = newProperties.keySet().iterator(); i.hasNext(); ) { 230 String key = (String) i.next(); 231 setProperty(source, key, newProperties.getProperty(key)); 232 } 233 } 234 235 /** 236 * Removes all properties from this container. 237 */ 238 public void clearProperties() { 239 this.heldProperties.clear(); 240 } 241 242 /** 243 * @return iterator over the keys of all properties in this container 244 */ 245 public Iterator getKeys() { 246 return this.heldProperties.keySet().iterator(); 247 } 248 249 /** 250 * @param key 251 * @throws IllegalArgumentException if the given key is null 252 */ 253 private void validateKey(String key) { 254 if (key == null) { 255 throw new IllegalArgumentException("invalid (null) key"); 256 } 257 } 258 259 /** 260 * @throws IllegalArgumentException if the given value is null 261 */ 262 private void validateValue(String value) { 263 if (value == null) { 264 throw new IllegalArgumentException("invalid (null) value"); 265 } 266 } 267 268 public Properties getHeldProperties() { 269 return heldProperties; 270 } 271 272 public void setHeldProperties(Properties heldProperties) { 273 this.heldProperties = heldProperties; 274 } 275 } 276 277 /** 278 * Used to obtain properties from a properties file 279 */ 280 protected static class FilePropertySource implements PropertySource { 281 private static Log log = LogFactory.getLog(FilePropertySource.class); 282 283 private String fileName; 284 private boolean allowOverrides; 285 286 public void setFileName(String fileName) { 287 this.fileName = fileName; 288 } 289 290 public String getFileName() { 291 return this.fileName; 292 } 293 294 public boolean isAllowOverrides() { 295 return this.allowOverrides; 296 } 297 298 public void setAllowOverrides(boolean allowOverrides) { 299 this.allowOverrides = allowOverrides; 300 } 301 302 /** 303 * Attempts to load properties from a properties file which has the current fileName and is located on the 304 * classpath 305 * 306 * @throws IllegalStateException if the fileName is null or empty 307 * @see org.kuali.rice.krad.service.impl.ConfigurationServiceImpl.PropertySource#loadProperties() 308 */ 309 public Properties loadProperties() { 310 if (StringUtils.isBlank(getFileName())) { 311 throw new IllegalStateException("invalid (blank) fileName"); 312 } 313 314 Properties properties = new Properties(); 315 316 ClassLoader loader = Thread.currentThread().getContextClassLoader(); 317 URL url = loader.getResource(getFileName()); 318 if (url == null) { 319 throw new PropertiesException("unable to locate properties file '" + getFileName() + "'"); 320 } 321 322 InputStream in = null; 323 324 try { 325 in = url.openStream(); 326 properties.load(in); 327 } catch (IOException e) { 328 throw new PropertiesException("error loading from properties file '" + getFileName() + "'", e); 329 } finally { 330 if (in != null) { 331 try { 332 in.close(); 333 } catch (IOException e) { 334 log.error("caught exception closing InputStream: " + e); 335 } 336 337 } 338 } 339 340 return properties; 341 } 342 } 343 }