blob: 37bd902f0d4d8f1c0b6cd74fb7deb52872df37fb [file] [log] [blame]
/*
* 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.sling.feature.extension.apiregions;
import java.util.Iterator;
import java.util.LinkedHashSet;
import java.util.Map;
import java.util.Set;
import org.apache.sling.feature.Extension;
import org.apache.sling.feature.Feature;
import org.apache.sling.feature.builder.HandlerContext;
import org.apache.sling.feature.builder.MergeHandler;
import org.apache.sling.feature.extension.apiregions.api.config.ConfigurableEntity;
import org.apache.sling.feature.extension.apiregions.api.config.ConfigurableEntityAddition;
import org.apache.sling.feature.extension.apiregions.api.config.ConfigurationApi;
import org.apache.sling.feature.extension.apiregions.api.config.ConfigurationDescription;
import org.apache.sling.feature.extension.apiregions.api.config.ConfigurationDescriptionAddition;
import org.apache.sling.feature.extension.apiregions.api.config.FactoryConfigurationDescription;
import org.apache.sling.feature.extension.apiregions.api.config.FactoryConfigurationDescriptionAddition;
import org.apache.sling.feature.extension.apiregions.api.config.FrameworkPropertyDescription;
import org.apache.sling.feature.extension.apiregions.api.config.PropertyDescription;
import org.apache.sling.feature.extension.apiregions.api.config.PropertyDescriptionAddition;
import org.apache.sling.feature.extension.apiregions.api.config.Region;
/**
* Merge the configuration api extension
*/
public class ConfigurationApiMergeHandler implements MergeHandler {
@Override
public boolean canMerge(final Extension extension) {
return ConfigurationApi.EXTENSION_NAME.equals(extension.getName());
}
@Override
public void merge(final HandlerContext context,
final Feature targetFeature,
final Feature sourceFeature,
final Extension targetExtension,
final Extension sourceExtension) {
if ( targetExtension == null ) {
// no target available yet, just copy source and update cache
final ConfigurationApi sourceApi = ConfigurationApi.getConfigurationApi(sourceExtension);
sourceApi.getFeatureToRegionCache().put(sourceFeature.getId(), sourceApi.detectRegion());
ConfigurationApi.setConfigurationApi(targetFeature, sourceApi);
} else {
final ConfigurationApi sourceApi = ConfigurationApi.getConfigurationApi(sourceExtension);
final ConfigurationApi targetApi = ConfigurationApi.getConfigurationApi(targetExtension);
// region merging
if ( context.isInitialMerge() ) {
targetApi.setRegion(sourceApi.getRegion());
targetApi.setMode(sourceApi.getMode());
} else {
// region merging is different for prototypes
if ( sourceApi.getRegion() != targetApi.getRegion() ) {
if ( context.isPrototypeMerge() ) {
if ( sourceApi.getRegion() != null ) {
targetApi.setRegion(sourceApi.getRegion());
}
} else {
targetApi.setRegion(Region.GLOBAL);
}
}
if ( targetApi.getMode().ordinal() > sourceApi.getMode().ordinal() ) {
targetApi.setMode(sourceApi.getMode());
}
}
// merge - but throw on duplicates
for(final Map.Entry<String, ConfigurationDescription> entry : sourceApi.getConfigurationDescriptions().entrySet()) {
if ( targetApi.getConfigurationDescriptions().containsKey(entry.getKey())) {
throw new IllegalStateException("Duplicate configuration description " + entry.getKey());
}
targetApi.getConfigurationDescriptions().put(entry.getKey(), entry.getValue());
}
for(final Map.Entry<String, FactoryConfigurationDescription> entry : sourceApi.getFactoryConfigurationDescriptions().entrySet()) {
if ( targetApi.getFactoryConfigurationDescriptions().containsKey(entry.getKey())) {
throw new IllegalStateException("Duplicate factory configuration description " + entry.getKey());
}
targetApi.getFactoryConfigurationDescriptions().put(entry.getKey(), entry.getValue());
}
for(final Map.Entry<String, FrameworkPropertyDescription> entry : sourceApi.getFrameworkPropertyDescriptions().entrySet()) {
if ( targetApi.getFrameworkPropertyDescriptions().containsKey(entry.getKey())) {
throw new IllegalStateException("Duplicate framework property description " + entry.getKey());
}
targetApi.getFrameworkPropertyDescriptions().put(entry.getKey(), entry.getValue());
}
targetApi.getInternalConfigurations().addAll(sourceApi.getInternalConfigurations());
targetApi.getInternalFactoryConfigurations().addAll(sourceApi.getInternalFactoryConfigurations());
targetApi.getInternalFrameworkProperties().addAll(sourceApi.getInternalFrameworkProperties());
// update cache
if ( !context.isPrototypeMerge() ) {
targetApi.getFeatureToRegionCache().put(sourceFeature.getId(), sourceApi.detectRegion());
targetApi.getFeatureToRegionCache().putAll(sourceApi.getFeatureToRegionCache());
}
// check for additions
processAdditions(targetApi, targetApi, false);
processAdditions(targetApi, sourceApi, true);
ConfigurationApi.setConfigurationApi(targetFeature, targetApi);
}
}
/**
* Process/add the additions from another configuration api
*/
private void processAdditions(final ConfigurationApi targetApi, final ConfigurationApi sourceApi, final boolean keep) {
// configuration additions
final Iterator<Map.Entry<String, ConfigurationDescriptionAddition>> itc = sourceApi.getConfigurationDescriptionAdditions().entrySet().iterator();
while ( itc.hasNext() ) {
final Map.Entry<String, ConfigurationDescriptionAddition> entry = itc.next();
final ConfigurationDescription cd = targetApi.getConfigurationDescriptions().get(entry.getKey());
if ( cd == null && keep ) {
// if configuration is not found, keep the addition
if ( targetApi.getConfigurationDescriptionAdditions().get(entry.getKey()) == null ) {
targetApi.getConfigurationDescriptionAdditions().put(entry.getKey(), entry.getValue());
} else {
throw new IllegalStateException("Duplicate configuration description addition " + entry.getKey());
}
} else if ( cd != null ) {
processAddition("configuration ".concat(entry.getKey()), cd, entry.getValue());
if ( !keep ) {
itc.remove();
}
}
}
// factory configuration additions
final Iterator<Map.Entry<String, FactoryConfigurationDescriptionAddition>> itf = sourceApi.getFactoryConfigurationDescriptionAdditions().entrySet().iterator();
while ( itf.hasNext() ) {
final Map.Entry<String, FactoryConfigurationDescriptionAddition> entry = itf.next();
final FactoryConfigurationDescription cd = targetApi.getFactoryConfigurationDescriptions().get(entry.getKey());
if ( cd == null && keep ) {
// if configuration is not found, keep the addition
if ( targetApi.getFactoryConfigurationDescriptionAdditions().get(entry.getKey()) == null ) {
targetApi.getFactoryConfigurationDescriptionAdditions().put(entry.getKey(), entry.getValue());
} else {
throw new IllegalStateException("Duplicate factory configuration description addition " + entry.getKey());
}
} else if ( cd != null ) {
processAddition("factory configuration ".concat(entry.getKey()), cd, entry.getValue());
if ( !keep ) {
itf.remove();
}
}
}
}
private void processAddition(final String descId,
final ConfigurableEntity entity,
final ConfigurableEntityAddition addition) {
for(final Map.Entry<String, PropertyDescriptionAddition> entry : addition.getPropertyDescriptionAdditions().entrySet()) {
final String propName = entry.getKey();
final PropertyDescription pd = entity.getPropertyDescriptions().get(propName);
if ( pd == null ) {
throw new IllegalStateException("Property named '" + propName + "' is not defined in " + descId);
}
if ( entry.getValue().getIncludes() != null ) {
final Set<String> includes = new LinkedHashSet<>();
if ( pd.getIncludes() != null ) {
for(final String v : pd.getIncludes()) {
includes.add(v);
}
}
for(final String v : entry.getValue().getIncludes()) {
includes.add(v);
}
pd.setIncludes(includes.toArray(new String[includes.size()]));
}
}
if ( entity instanceof FactoryConfigurationDescription ) {
final FactoryConfigurationDescription fcd = (FactoryConfigurationDescription)entity;
final FactoryConfigurationDescriptionAddition fcda = (FactoryConfigurationDescriptionAddition)addition;
for(final String n : fcda.getInternalNames()) {
fcd.getInternalNames().add(n);
}
}
}
}