/* * JBoss, Home of Professional Open Source. * Copyright 2012, Red Hat, Inc., and individual contributors * as indicated by the @author tags. See the copyright.txt file in the * distribution for a full listing of individual contributors. * * This 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. * * This software is distributed in the hope that it will be useful, * but WITHOUT ANY WARRANTY; without even the implied warranty of * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU * Lesser General Public License for more details. * * You should have received a copy of the GNU Lesser General Public * License along with this software; if not, write to the Free * Software Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA * 02110-1301 USA, or see the FSF site: http://www.fsf.org. */ package org.jboss.as.clustering.jgroups.subsystem; import java.util.ArrayList; import java.util.EnumSet; import java.util.LinkedList; import java.util.List; import java.util.Set; import java.util.function.UnaryOperator; import java.util.stream.Collectors; import org.jboss.as.clustering.controller.CapabilityReference; import org.jboss.as.clustering.controller.CommonUnaryRequirement; import org.jboss.as.clustering.controller.Definable; import org.jboss.as.clustering.controller.Operations; import org.jboss.as.clustering.controller.ResourceServiceBuilderFactory; import org.jboss.as.clustering.controller.WriteAttributeStepHandler; import org.jboss.as.clustering.controller.transform.ChainedOperationTransformer; import org.jboss.as.clustering.controller.transform.LegacyPropertyAddOperationTransformer; import org.jboss.as.clustering.controller.transform.LegacyPropertyMapGetOperationTransformer; import org.jboss.as.clustering.controller.transform.LegacyPropertyResourceTransformer; import org.jboss.as.clustering.controller.transform.LegacyPropertyWriteOperationTransformer; import org.jboss.as.clustering.controller.transform.PathAddressTransformer; import org.jboss.as.clustering.controller.transform.RequiredChildResourceDiscardPolicy; import org.jboss.as.clustering.controller.transform.SimpleDescribeOperationTransformer; import org.jboss.as.clustering.controller.transform.SimpleOperationTransformer; import org.jboss.as.clustering.controller.transform.SimplePathOperationTransformer; import org.jboss.as.clustering.controller.transform.SimpleReadAttributeOperationTransformer; import org.jboss.as.clustering.controller.transform.SimpleRemoveOperationTransformer; import org.jboss.as.clustering.jgroups.logging.JGroupsLogger; import org.jboss.as.controller.AttributeDefinition; import org.jboss.as.controller.ModelVersion; import org.jboss.as.controller.OperationContext; import org.jboss.as.controller.OperationFailedException; import org.jboss.as.controller.OperationStepHandler; import org.jboss.as.controller.PathAddress; import org.jboss.as.controller.PathElement; import org.jboss.as.controller.SimpleAttributeDefinitionBuilder; import org.jboss.as.controller.access.management.SensitiveTargetAccessConstraintDefinition; import org.jboss.as.controller.capability.RuntimeCapability; import org.jboss.as.controller.descriptions.ModelDescriptionConstants; import org.jboss.as.controller.operations.global.MapOperations; import org.jboss.as.controller.registry.AliasEntry; import org.jboss.as.controller.registry.AttributeAccess; import org.jboss.as.controller.registry.Resource; import org.jboss.as.controller.transform.OperationTransformer; import org.jboss.as.controller.transform.ResourceTransformationContext; import org.jboss.as.controller.transform.ResourceTransformer; import org.jboss.as.controller.transform.description.AttributeConverter.DefaultValueAttributeConverter; import org.jboss.as.controller.transform.description.ResourceTransformationDescriptionBuilder; import org.jboss.dmr.ModelNode; import org.jboss.dmr.ModelType; import org.jgroups.protocols.TP; import org.wildfly.clustering.jgroups.spi.ChannelFactory; import org.wildfly.clustering.jgroups.spi.TransportConfiguration; /** * Resource description for /subsystem=jgroups/stack=X/transport=* * * @author Richard Achmatowicz (c) 2011 Red Hat Inc. * @author Paul Ferraro */ public class TransportResourceDefinition<T extends TP> extends AbstractProtocolResourceDefinition<T, TransportConfiguration<T>> { static final PathElement LEGACY_PATH = pathElement("TRANSPORT"); static final PathElement WILDCARD_PATH = pathElement(PathElement.WILDCARD_VALUE); public static PathElement pathElement(String name) { return PathElement.pathElement("transport", name); } enum Capability implements org.jboss.as.clustering.controller.Capability { TRANSPORT("org.wildfly.clustering.jgroups.transport"), ; private final RuntimeCapability<Void> definition; Capability(String name) { this.definition = RuntimeCapability.Builder.of(name, true).build(); } @Override public RuntimeCapability<Void> getDefinition() { return this.definition; } @Override public RuntimeCapability<Void> resolve(PathAddress address) { return this.definition.fromBaseCapability(address.getParent().getLastElement().getValue()); } } enum Attribute implements org.jboss.as.clustering.controller.Attribute { @Deprecated SHARED("shared", ModelType.BOOLEAN, builder -> builder .setDefaultValue(new ModelNode(false)) .setDeprecated(JGroupsModel.VERSION_4_0_0.getVersion())), SOCKET_BINDING("socket-binding", ModelType.STRING, builder -> builder .setAccessConstraints(SensitiveTargetAccessConstraintDefinition.SOCKET_BINDING_REF) .setCapabilityReference(new CapabilityReference(Capability.TRANSPORT, CommonUnaryRequirement.SOCKET_BINDING))), DIAGNOSTICS_SOCKET_BINDING("diagnostics-socket-binding", ModelType.STRING, builder -> builder .setAccessConstraints(SensitiveTargetAccessConstraintDefinition.SOCKET_BINDING_REF) .setCapabilityReference(new CapabilityReference(Capability.TRANSPORT, CommonUnaryRequirement.SOCKET_BINDING))), SITE("site", ModelType.STRING), RACK("rack", ModelType.STRING), MACHINE("machine", ModelType.STRING), ; private final AttributeDefinition definition; Attribute(String name, ModelType type) { this(name, type, UnaryOperator.identity()); } Attribute(String name, ModelType type, UnaryOperator<SimpleAttributeDefinitionBuilder> configurator) { this.definition = configurator.apply(new SimpleAttributeDefinitionBuilder(name, type) .setAllowExpression(true) .setRequired(false) .setFlags(AttributeAccess.Flag.RESTART_RESOURCE_SERVICES) ).build(); } @Override public AttributeDefinition getDefinition() { return this.definition; } } @Deprecated enum ThreadingAttribute implements org.jboss.as.clustering.controller.Attribute { DEFAULT_EXECUTOR("default-executor"), OOB_EXECUTOR("oob-executor"), TIMER_EXECUTOR("timer-executor"), THREAD_FACTORY("thread-factory"), ; private final AttributeDefinition definition; ThreadingAttribute(String name) { this.definition = new SimpleAttributeDefinitionBuilder(name, ModelType.STRING) .setAllowExpression(false) .setRequired(false) .setDeprecated(JGroupsModel.VERSION_3_0_0.getVersion()) .setFlags(AttributeAccess.Flag.RESTART_NONE) .build(); } @Override public AttributeDefinition getDefinition() { return this.definition; } } static void buildTransformation(ModelVersion version, ResourceTransformationDescriptionBuilder parent) { ResourceTransformationDescriptionBuilder builder = parent.addChildResource(WILDCARD_PATH); AbstractProtocolResourceDefinition.addTransformations(version, builder); if (JGroupsModel.VERSION_3_0_0.requiresTransformation(version)) { builder.getAttributeBuilder().setValueConverter(new DefaultValueAttributeConverter(Attribute.SHARED.getDefinition()), Attribute.SHARED.getDefinition()); builder.setCustomResourceTransformer(new ResourceTransformer() { @Override public void transformResource(ResourceTransformationContext context, PathAddress address, Resource resource) throws OperationFailedException { new LegacyPropertyResourceTransformer().transformResource(context, LEGACY_ADDRESS_TRANSFORMER.transform(address), resource); } }); builder.addOperationTransformationOverride(ModelDescriptionConstants.ADD).setCustomOperationTransformer(new SimpleOperationTransformer(new org.jboss.as.clustering.controller.transform.OperationTransformer() { @Override public ModelNode transformOperation(final ModelNode operation) { operation.get(ModelDescriptionConstants.OP_ADDR).set(LEGACY_ADDRESS_TRANSFORMER.transform(Operations.getPathAddress(operation)).toModelNode()); return new LegacyPropertyAddOperationTransformer().transformOperation(operation); } })).inheritResourceAttributeDefinitions(); builder.addOperationTransformationOverride(ModelDescriptionConstants.REMOVE).setCustomOperationTransformer(new SimpleRemoveOperationTransformer(LEGACY_ADDRESS_TRANSFORMER)); builder.addOperationTransformationOverride(ModelDescriptionConstants.READ_ATTRIBUTE_OPERATION).setCustomOperationTransformer(new SimpleReadAttributeOperationTransformer(LEGACY_ADDRESS_TRANSFORMER)); builder.addOperationTransformationOverride(ModelDescriptionConstants.DESCRIBE).setCustomOperationTransformer(new SimpleDescribeOperationTransformer(LEGACY_ADDRESS_TRANSFORMER)); List<OperationTransformer> getOpTransformerChain = new LinkedList<>(); getOpTransformerChain.add(new SimplePathOperationTransformer(LEGACY_ADDRESS_TRANSFORMER)); getOpTransformerChain.add(new SimpleOperationTransformer(new LegacyPropertyMapGetOperationTransformer())); ChainedOperationTransformer getOpChainedTransformer = new ChainedOperationTransformer(getOpTransformerChain, false); builder.addRawOperationTransformationOverride(MapOperations.MAP_GET_DEFINITION.getName(), getOpChainedTransformer); List<OperationTransformer> writeOpTransformerChain = new LinkedList<>(); writeOpTransformerChain.add(new SimplePathOperationTransformer(LEGACY_ADDRESS_TRANSFORMER)); writeOpTransformerChain.add(new LegacyPropertyWriteOperationTransformer()); ChainedOperationTransformer writeOpChainedTransformer = new ChainedOperationTransformer(writeOpTransformerChain, false); for (String opName : Operations.getAllWriteAttributeOperationNames()) { builder.addOperationTransformationOverride(opName) .inheritResourceAttributeDefinitions() .setCustomOperationTransformer(writeOpChainedTransformer); } // Reject thread pool configuration, discard if undefined, support EAP 6.x slaves using deprecated attributes builder.addChildResource(ThreadPoolResourceDefinition.WILDCARD_PATH, RequiredChildResourceDiscardPolicy.REJECT_AND_WARN); } else { EnumSet.allOf(ThreadPoolResourceDefinition.class).forEach(p -> p.buildTransformation(version, parent)); } } // Transform /subsystem=jgroups/stack=*/transport=* -> /subsystem=jgroups/stack=*/transport=TRANSPORT static final PathAddressTransformer LEGACY_ADDRESS_TRANSFORMER = new PathAddressTransformer() { @Override public PathAddress transform(PathAddress address) { return address.subAddress(0, address.size() - 1).append(LEGACY_PATH); } }; TransportResourceDefinition(ResourceServiceBuilderFactory<TransportConfiguration<T>> builderFactory, ResourceServiceBuilderFactory<ChannelFactory> parentBuilderFactory) { this(new Parameters(WILDCARD_PATH, new JGroupsResourceDescriptionResolver(WILDCARD_PATH, ProtocolResourceDefinition.WILDCARD_PATH)), builderFactory, parentBuilderFactory); } TransportResourceDefinition(PathElement path, ResourceServiceBuilderFactory<TransportConfiguration<T>> builderFactory, ResourceServiceBuilderFactory<ChannelFactory> parentBuilderFactory) { this(new Parameters(path, new JGroupsResourceDescriptionResolver(path, WILDCARD_PATH, ProtocolResourceDefinition.WILDCARD_PATH)), builderFactory, parentBuilderFactory); } private TransportResourceDefinition(Parameters parameters, ResourceServiceBuilderFactory<TransportConfiguration<T>> builderFactory, ResourceServiceBuilderFactory<ChannelFactory> parentBuilderFactory) { super(parameters, descriptor -> descriptor .addAttributes(Attribute.class) .addCapabilities(Capability.class) .addExtraParameters(ThreadingAttribute.class) .addRequiredChildren(ThreadPoolResourceDefinition.class) , builderFactory, parentBuilderFactory, (parent, registration) -> { new WriteAttributeStepHandler(() -> EnumSet.allOf(ThreadingAttribute.class).stream().map(Definable::getDefinition).collect(Collectors.toList())) { @Override protected void validateUpdatedModel(OperationContext context, Resource model) throws OperationFailedException { // Add a new step to validate instead of doing it directly in this method. // This allows a composite op to change both attributes and then the // validation occurs after both have done their work. context.addStep(new OperationStepHandler() { @Override public void execute(OperationContext context, ModelNode operation) throws OperationFailedException { ModelNode conf = context.readResource(PathAddress.EMPTY_ADDRESS).getModel(); // TODO doesn't cover the admin-only modes if (context.getProcessType().isServer()) { for (ThreadingAttribute attribute : EnumSet.allOf(ThreadingAttribute.class)) { if (conf.hasDefined(attribute.getName())) { // That is not supported. throw new OperationFailedException(JGroupsLogger.ROOT_LOGGER.threadsAttributesUsedInRuntime()); } } } } }, OperationContext.Stage.MODEL); } }.register(registration); if (registration.getPathAddress().getLastElement().isWildcard()) { EnumSet.allOf(ThreadPoolResourceDefinition.class).forEach(pool -> pool.register(registration)); } if (registration.getPathAddress().getLastElement().isWildcard()) { parent.registerAlias(LEGACY_PATH, new AliasEntry(registration) { @Override public PathAddress convertToTargetAddress(PathAddress aliasAddress, AliasContext aliasContext) { PathAddress target = this.getTargetAddress(); List<PathElement> result = new ArrayList<>(aliasAddress.size()); for (int i = 0; i < aliasAddress.size(); ++i) { PathElement element = aliasAddress.getElement(i); if (i == target.size() - 1) { final ModelNode operation = aliasContext.getOperation(); final String stackName; if (ModelDescriptionConstants.ADD.equals(Operations.getName(operation)) && operation.hasDefined("type")) { stackName = operation.get("type").asString(); } else { Resource root = null; try { root = aliasContext.readResourceFromRoot(PathAddress.pathAddress(result)); } catch (Resource.NoSuchResourceException ignored) { } if (root == null) { stackName = "*"; } else { Set<String> names = root.getChildrenNames("transport"); if (names.size() > 1) { throw new AssertionError("There should be at most one child"); } else if (names.size() == 0) { stackName = "*"; } else { stackName = names.iterator().next(); } } } result.add(PathElement.pathElement("transport", stackName)); } else if (i < target.size()) { PathElement targetElement = target.getElement(i); result.add(targetElement.isWildcard() ? PathElement.pathElement(targetElement.getKey(), element.getValue()) : targetElement); } else { result.add(element); } } return PathAddress.pathAddress(result); } }); } }); } }