/**
* Copyright (c) 2014 by the original author or authors.
*
* This code is free software; you can redistribute it and/or modify it under the terms of the
* GNU Lesser General Public License as published by the Free Software Foundation; either
* version 2.1 of the License, or (at your option) any later version.
*
* The above copyright notice and this permission notice shall be included in all copies or
* substantial portions of the Software.
*
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING
* BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM,
* DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
*/
package ch.sdi;
import java.io.IOException;
import java.io.InputStream;
import java.util.ArrayList;
import java.util.List;
import java.util.Properties;
import org.apache.logging.log4j.LogManager;
import org.apache.logging.log4j.Logger;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.core.env.ConfigurableEnvironment;
import org.springframework.core.env.MutablePropertySources;
import org.springframework.core.env.PropertiesPropertySource;
import org.springframework.core.env.PropertySource;
import org.springframework.stereotype.Component;
import org.springframework.util.StringUtils;
import ch.sdi.core.annotations.SdiProps;
import ch.sdi.core.impl.cfg.ConfigUtils;
import ch.sdi.core.intf.SdiMainProperties;
import ch.sdi.core.util.ClassUtil;
/**
* Overloads the already loaded properties with properties found in property files on the classpath
* whose filenam follows this pattern "user.<name>.properties", where name is one of the property
* file names of the configuration classes which are annotated with @SdiProps (see {@link SdiProps}
* for more information).<p>
*
*
* @version 1.0 (07.11.2014)
* @author Heri
*/
@Component
public class UserPropertyOverloader
{
/** logger for this class */
private Logger myLog = LogManager.getLogger( UserPropertyOverloader.class );
@Autowired
private ConfigurableEnvironment myEnv;
/**
* Constructor
*
*/
public UserPropertyOverloader()
{
super();
}
public void overrideByUserProperties()
{
List<Class<?>> candidates = findCandidates();
for ( Class<?> clazz : candidates )
{
myLog.debug( "candidate for user property overloading: " + clazz.getName() );
String fileName = SdiMainProperties.USER_OVERRIDE_PREFIX +
ConfigUtils.makePropertyResourceName( clazz );
InputStream is = this.getClass().getResourceAsStream( "/" + fileName );
if ( is == null )
{
myLog.debug( "Skipping non existing user overloading property file: " + fileName );
continue;
} // if is == null
myLog.debug( "Found overloading property file: " + fileName );
Properties props = new Properties();
try
{
props.load( is );
}
catch ( IOException t )
{
myLog.error( "Problems loading property file " + fileName );
}
myEnv.setIgnoreUnresolvableNestedPlaceholders( true );
try
{
props.stringPropertyNames().stream()
.map( key ->
{
String origValue = myEnv.getProperty( key );
String result = "Key " + key + ": ";
return ( origValue == null || origValue.isEmpty() )
? result + "No default value found. Adding new value to environment: \""
+ props.getProperty( key ) + "\""
: result + "Overriding default value \"" + origValue + "\" with new value: \""
+ props.getProperty( key ) + "\"";
})
.forEach( msg -> myLog.debug( msg ) ) ;
}
finally
{
myEnv.setIgnoreUnresolvableNestedPlaceholders( false );
}
PropertySource<?> ps = new PropertiesPropertySource( fileName, props );
MutablePropertySources mps = myEnv.getPropertySources();
if ( mps.get( ConfigUtils.PROP_SOURCE_NAME_CMD_LINE ) != null )
{
mps.addAfter( ConfigUtils.PROP_SOURCE_NAME_CMD_LINE, ps );
}
else
{
mps.addFirst( ps );
}
myLog.debug( "PropertySources after adding overloading: " + mps );
}
}
/**
* @return
*/
private List<Class<?>> findCandidates()
{
List<Class<?>> result = new ArrayList<Class<?>>();
// we parse all classes which are below our top level package:
String pack = this.getClass().getPackage().getName();
myLog.debug( "found package: " + pack );
pack = pack.replace( '.', '/' );
String defaultRoot = pack.split( "/" )[0];
result.addAll( ClassUtil.findCandidatesByAnnotation( SdiProps.class, defaultRoot ) );
String newRoot = myEnv.getProperty( SdiMainProperties.KEY_PROPERTIESOVERRIDE_INCLUDEROOT );
if ( StringUtils.hasText( newRoot ) )
{
String[] newRoots = newRoot.split( "," );
for ( int i = 0; i < newRoots.length; i++ )
{
if ( newRoots[i].equals( defaultRoot ) )
{
continue;
}
result.addAll( ClassUtil.findCandidatesByAnnotation( SdiProps.class, newRoots[i] ) );
}
}
return result;
}
}