/** * Licensed to the Apache Software Foundation (ASF) under one or more * contributor license agreements. See the NOTICE file distributed with * this work for additional information regarding copyright ownership. * The ASF licenses this file to You 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.apache.camel.blueprint; import java.lang.reflect.Method; import java.util.ArrayList; import java.util.LinkedHashSet; import java.util.List; import java.util.Map; import java.util.Properties; import java.util.Set; import org.apache.aries.blueprint.ExtendedBeanMetadata; import org.apache.aries.blueprint.ext.AbstractPropertyPlaceholder; import org.apache.aries.blueprint.ext.PropertyPlaceholder; import org.apache.camel.component.properties.DefaultPropertiesParser; import org.apache.camel.component.properties.PropertiesComponent; import org.apache.camel.component.properties.PropertiesParser; import org.apache.camel.util.ObjectHelper; import org.osgi.service.blueprint.container.BlueprintContainer; import org.osgi.service.blueprint.reflect.ComponentMetadata; /** * Blueprint {@link PropertiesParser} which supports looking up * property placeholders from the Blueprint Property Placeholder Service. * <p/> * This implementation will sit on top of any existing configured * {@link PropertiesParser} and will delegate to those in case Blueprint could not * resolve the property. */ public class BlueprintPropertiesParser extends DefaultPropertiesParser { private final PropertiesComponent propertiesComponent; private final BlueprintContainer container; private final PropertiesParser delegate; private final Set<AbstractPropertyPlaceholder> placeholders = new LinkedHashSet<AbstractPropertyPlaceholder>(); private Method method; public BlueprintPropertiesParser(PropertiesComponent propertiesComponent, BlueprintContainer container, PropertiesParser delegate) { super(propertiesComponent); this.propertiesComponent = propertiesComponent; this.container = container; this.delegate = delegate; } /** * Lookup the ids of the Blueprint property placeholder services in the * Blueprint container. * * @return the ids, will be an empty array if none found. */ public String[] lookupPropertyPlaceholderIds() { List<String> ids = new ArrayList<String>(); for (Object componentId : container.getComponentIds()) { String id = (String) componentId; ComponentMetadata meta = container.getComponentMetadata(id); if (meta instanceof ExtendedBeanMetadata) { Class<?> clazz = ((ExtendedBeanMetadata) meta).getRuntimeClass(); if (clazz != null && AbstractPropertyPlaceholder.class.isAssignableFrom(clazz)) { ids.add(id); } } } return ids.toArray(new String[ids.size()]); } /** * Adds the given Blueprint property placeholder service with the given id * * @param id id of the Blueprint property placeholder service to add. */ public void addPropertyPlaceholder(String id) { Object component = container.getComponentInstance(id); if (component instanceof AbstractPropertyPlaceholder) { AbstractPropertyPlaceholder placeholder = (AbstractPropertyPlaceholder) component; placeholders.add(placeholder); log.debug("Adding Blueprint PropertyPlaceholder: {}", id); if (method == null) { try { method = AbstractPropertyPlaceholder.class.getDeclaredMethod("retrieveValue", String.class); method.setAccessible(true); } catch (NoSuchMethodException e) { throw new IllegalStateException("Cannot add blueprint property placeholder: " + id + " as the method retrieveValue is not accessible", e); } } } } @Override public String parseProperty(String key, String value, Properties properties) { log.trace("Parsing property key: {} with value: {}", key, value); String answer = null; // prefer any override properties // this logic is special for BlueprintPropertiesParser as we otherwise prefer // to use the AbstractPropertyPlaceholder from OSGi blueprint config admins // service to lookup otherwise if (key != null && propertiesComponent.getOverrideProperties() != null) { answer = (String) propertiesComponent.getOverrideProperties().get(key); } // lookup key in blueprint and return its value if (answer == null && key != null) { for (AbstractPropertyPlaceholder placeholder : placeholders) { boolean isDefault = false; if (placeholders.size() > 1) { // okay we have multiple placeholders and we want to return the answer that // is not the default placeholder if there is multiple keys if (placeholder instanceof PropertyPlaceholder) { Map map = ((PropertyPlaceholder) placeholder).getDefaultProperties(); isDefault = map != null && map.containsKey(key); } log.trace("Blueprint property key: {} is part of default properties: {}", key, isDefault); } try { String candidate = (String) ObjectHelper.invokeMethod(method, placeholder, key); if (candidate != null) { if (answer == null || !isDefault) { log.trace("Blueprint parsed candidate property key: {} as value: {}", key, answer); answer = candidate; } } } catch (Exception ex) { // Here we just catch the exception and try to use other candidate } } log.debug("Blueprint parsed property key: {} as value: {}", key, answer); } // if there is a delegate then let it parse the current answer as it may be jasypt which // need to decrypt values if (delegate != null) { String delegateAnswer = delegate.parseProperty(key, answer != null ? answer : value, properties); if (delegateAnswer != null) { answer = delegateAnswer; log.debug("Delegate property parser parsed the property key: {} as value: {}", key, answer); } } log.trace("Returning parsed property key: {} as value: {}", key, answer); return answer; } }