/* * ModeShape (http://www.modeshape.org) * * 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.modeshape.jboss.subsystem; import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.ADDRESS; import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.OP; import static org.jboss.as.controller.descriptions.ModelDescriptionConstants.OPERATION_HEADERS; import java.util.Properties; import org.jboss.as.controller.AbstractAddStepHandler; import org.jboss.as.controller.AttributeDefinition; import org.jboss.as.controller.OperationContext; import org.jboss.as.controller.OperationFailedException; import org.jboss.dmr.ModelNode; import org.jboss.dmr.Property; import org.jboss.msc.service.ServiceBuilder; import org.jboss.msc.service.ServiceController; import org.jboss.msc.service.ServiceTarget; import org.modeshape.common.util.StringUtil; import org.modeshape.jboss.service.IndexProviderService; import org.modeshape.jcr.JcrRepository; import org.modeshape.jcr.ModeShapeEngine; import org.modeshape.jcr.RepositoryConfiguration; import org.modeshape.jcr.RepositoryConfiguration.FieldName; public class AddIndexProvider extends AbstractAddStepHandler { public static final AddIndexProvider INSTANCE = new AddIndexProvider(); private AddIndexProvider() { } @Override protected void populateModel( ModelNode operation, ModelNode model ) throws OperationFailedException { populate(operation, model); } static void populate( ModelNode operation, ModelNode model ) throws OperationFailedException { for (AttributeDefinition attribute : ModelAttributes.INDEX_PROVIDER_ATTRIBUTES) { attribute.validateAndSet(operation, model); } } @Override protected void performRuntime( final OperationContext context, final ModelNode operation, final ModelNode model) throws OperationFailedException { ServiceTarget target = context.getServiceTarget(); Properties props = new Properties(); final AddressContext addressContext = AddressContext.forOperation(operation); final String repositoryName = addressContext.repositoryName(); final String providerName = addressContext.lastPathElementValue(); // Record the properties ... props.put(FieldName.NAME, providerName); String path = null; String relativeTo = null; for (String key : operation.keys()) { if (key.equals(ADDRESS) || key.equals(OP) || key.equals(OPERATION_HEADERS)) { // Ignore these ... continue; } ModelNode node = operation.get(key); if (!node.isDefined()) continue; if (key.equals(ModelKeys.CLASSNAME) && ModelAttributes.CLASSNAME.isMarshallable(operation)) { props.put(FieldName.CLASSNAME, node.asString()); } else if (key.equals(ModelKeys.MODULE) && ModelAttributes.MODULE.isMarshallable(operation)) { props.put(FieldName.CLASSLOADER, node.asString()); } else if (key.equals(ModelKeys.RELATIVE_TO) && ModelAttributes.RELATIVE_TO.isMarshallable(operation)) { // Optional field, but it is a JBoss convention that this might be a variable. Try to resolve this ... relativeTo = context.resolveExpressions(node).asString(); if (relativeTo.equalsIgnoreCase(ModeShapeExtension.JBOSS_DATA_DIR_VARIABLE)) { // the relative-to path should be the default jboss-data-dir. relativeTo = context.resolveExpressions(new ModelNode("${" + relativeTo + "}")).asString(); } props.put(ModelKeys.RELATIVE_TO, relativeTo); } else if (key.equals(ModelKeys.PATH) && ModelAttributes.PATH.isMarshallable(operation)) { // Optional field, but it is a JBoss convention that this might be a variable. Try to resolve this ... path = context.resolveExpressions(node).asString(); props.put(ModelKeys.PATH, path); } else if (key.equals(ModelKeys.PROPERTIES)) { for (Property property : node.asPropertyList()) { // Try resolving it in case it's an expression ... String value = context.resolveExpressions(property.getValue()).asString(); props.put(property.getName(), value); } } else { props.put(key, node.asString()); } } ensureClassLoadingPropertyIsSet(props); IndexProviderService providerService = new IndexProviderService(repositoryName, props); ServiceBuilder<JcrRepository> providerBuilder = target.addService(ModeShapeServiceNames.indexProviderServiceName(repositoryName, providerName), providerService); providerBuilder.addDependency(ModeShapeServiceNames.ENGINE, ModeShapeEngine.class, providerService.getModeShapeEngineInjector()); providerBuilder.addDependency(ModeShapeServiceNames.repositoryServiceName(repositoryName), JcrRepository.class, providerService.getJcrRepositoryInjector()); providerBuilder.setInitialMode(ServiceController.Mode.ACTIVE).install(); } private void ensureClassLoadingPropertyIsSet( Properties providerProperties ) throws OperationFailedException { // could be already set if the "module" element is present in the xml (AddIndexProvider) if (providerProperties.containsKey(FieldName.CLASSLOADER)) { return; } String providerClassName = providerProperties.getProperty(FieldName.CLASSNAME); if (StringUtil.isBlank(providerClassName)) { throw new OperationFailedException( String.format("Required property: %s not found among the index provider properties: %s", FieldName.CLASSNAME, providerProperties)); } // try to see if an alias is configured String fqProviderClass = RepositoryConfiguration.getBuiltInIndexProviderClassName(providerClassName); if (fqProviderClass == null) { fqProviderClass = providerClassName; } String providerModuleName = ModuleNamesProvider.moduleNameFor(fqProviderClass); providerProperties.setProperty(FieldName.CLASSLOADER, providerModuleName); } }