X-Git-Url: https://git.opendaylight.org/gerrit/gitweb?a=blobdiff_plain;f=binding%2Fmdsal-binding-dom-adapter%2Fsrc%2Fmain%2Fjava%2Forg%2Fopendaylight%2Fmdsal%2Fbinding%2Fdom%2Fadapter%2FActionProviderServiceAdapter.java;h=df1b933e1ad18f830e65f86b7c84040a15575533;hb=11408d627adca7eb71ac956c3ad01f75b6b91596;hp=11bc55dc1ac51cae9e6983575de7aa60e160f8c1;hpb=b4f58d7fb693ae61cefc69a991f8b5a1af71ab83;p=mdsal.git diff --git a/binding/mdsal-binding-dom-adapter/src/main/java/org/opendaylight/mdsal/binding/dom/adapter/ActionProviderServiceAdapter.java b/binding/mdsal-binding-dom-adapter/src/main/java/org/opendaylight/mdsal/binding/dom/adapter/ActionProviderServiceAdapter.java index 11bc55dc1a..df1b933e1a 100644 --- a/binding/mdsal-binding-dom-adapter/src/main/java/org/opendaylight/mdsal/binding/dom/adapter/ActionProviderServiceAdapter.java +++ b/binding/mdsal-binding-dom-adapter/src/main/java/org/opendaylight/mdsal/binding/dom/adapter/ActionProviderServiceAdapter.java @@ -9,49 +9,90 @@ package org.opendaylight.mdsal.binding.dom.adapter; import static java.util.Objects.requireNonNull; -import com.google.common.annotations.Beta; +import com.google.common.collect.ClassToInstanceMap; import com.google.common.collect.ImmutableSet; -import com.google.common.util.concurrent.FluentFuture; +import com.google.common.util.concurrent.Futures; +import com.google.common.util.concurrent.ListenableFuture; +import java.util.List; import java.util.Set; +import java.util.stream.Collectors; import org.eclipse.jdt.annotation.NonNullByDefault; import org.opendaylight.mdsal.binding.api.ActionProviderService; +import org.opendaylight.mdsal.binding.api.ActionSpec; import org.opendaylight.mdsal.binding.api.DataTreeIdentifier; -import org.opendaylight.mdsal.binding.dom.codec.api.BindingNormalizedNodeSerializer; +import org.opendaylight.mdsal.binding.dom.adapter.BindingDOMAdapterBuilder.Factory; import org.opendaylight.mdsal.common.api.LogicalDatastoreType; import org.opendaylight.mdsal.dom.api.DOMActionImplementation; +import org.opendaylight.mdsal.dom.api.DOMActionInstance; import org.opendaylight.mdsal.dom.api.DOMActionProviderService; import org.opendaylight.mdsal.dom.api.DOMActionResult; import org.opendaylight.mdsal.dom.api.DOMDataTreeIdentifier; +import org.opendaylight.mdsal.dom.api.DOMService; +import org.opendaylight.mdsal.dom.spi.SimpleDOMActionResult; import org.opendaylight.yangtools.concepts.AbstractObjectRegistration; import org.opendaylight.yangtools.concepts.ObjectRegistration; import org.opendaylight.yangtools.yang.binding.Action; import org.opendaylight.yangtools.yang.binding.DataObject; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier; +import org.opendaylight.yangtools.yang.common.ErrorTag; +import org.opendaylight.yangtools.yang.common.ErrorType; import org.opendaylight.yangtools.yang.common.RpcResult; +import org.opendaylight.yangtools.yang.common.RpcResultBuilder; import org.opendaylight.yangtools.yang.common.YangConstants; +import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier; import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier; import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode; -import org.opendaylight.yangtools.yang.model.api.SchemaPath; +import org.opendaylight.yangtools.yang.model.api.stmt.SchemaNodeIdentifier.Absolute; +import org.slf4j.Logger; +import org.slf4j.LoggerFactory; -@Beta @NonNullByDefault -final class ActionProviderServiceAdapter extends AbstractBindingAdapter +public final class ActionProviderServiceAdapter extends AbstractBindingAdapter implements ActionProviderService { - ActionProviderServiceAdapter(final BindingToNormalizedNodeCodec codec, final DOMActionProviderService delegate) { - super(codec, delegate); + private static final class Builder extends BindingDOMAdapterBuilder { + Builder(final AdapterContext adapterContext) { + super(adapterContext); + } + + @Override + protected ActionProviderService createInstance(final ClassToInstanceMap delegates) { + return new ActionProviderServiceAdapter(adapterContext(), + delegates.getInstance(DOMActionProviderService.class)); + } + + @Override + public Set> getRequiredDelegates() { + return ImmutableSet.of(DOMActionProviderService.class); + } + } + + private static final Logger LOG = LoggerFactory.getLogger(ActionProviderServiceAdapter.class); + + static final Factory BUILDER_FACTORY = Builder::new; + + ActionProviderServiceAdapter(final AdapterContext adapterContext, final DOMActionProviderService delegate) { + super(adapterContext, delegate); } @Override - public , - T extends org.opendaylight.yangtools.yang.binding.Action, S extends T> - ObjectRegistration registerImplementation(final Class actionInterface, final S implementation, - final LogicalDatastoreType datastore, final Set> validNodes) { - final SchemaPath path = getCodec().getActionPath(actionInterface); - final ObjectRegistration reg = getDelegate().registerActionImplementation( - new Impl(getCodec(), - NodeIdentifier.create(YangConstants.operationOutputQName(path.getLastComponent().getModule())), - actionInterface, implementation), ImmutableSet.of()); - return new AbstractObjectRegistration(implementation) { + public

, ?, ?>, S extends A> + ObjectRegistration registerImplementation(final ActionSpec spec, final S implementation, + final LogicalDatastoreType datastore, final Set> validNodes) { + final CurrentAdapterSerializer serializer = currentSerializer(); + final Absolute actionPath = serializer.getActionPath(spec); + final Impl impl = new Impl(adapterContext(), actionPath, spec.type(), implementation); + final DOMActionInstance instance = validNodes.isEmpty() + // Register on the entire datastore + ? DOMActionInstance.of(actionPath, new DOMDataTreeIdentifier(datastore, YangInstanceIdentifier.empty())) + // Register on specific instances + : DOMActionInstance.of(actionPath, validNodes.stream() + .map(node -> serializer.toDOMDataTreeIdentifier(DataTreeIdentifier.create(datastore, node))) + .collect(Collectors.toUnmodifiableSet())); + + + final ObjectRegistration reg = getDelegate().registerActionImplementation(impl, instance); + + return new AbstractObjectRegistration<>(implementation) { @Override protected void removeRegistration() { reg.close(); @@ -61,24 +102,41 @@ final class ActionProviderServiceAdapter extends AbstractBindingAdapter> actionInterface; + private final AdapterContext adapterContext; + @SuppressWarnings("rawtypes") private final Action implementation; - private final BindingNormalizedNodeSerializer codec; private final NodeIdentifier outputName; - Impl(final BindingNormalizedNodeSerializer codec, final NodeIdentifier outputName, + Impl(final AdapterContext adapterContext, final Absolute actionPath, final Class> actionInterface, final Action implementation) { - this.codec = requireNonNull(codec); - this.outputName = requireNonNull(outputName); + this.adapterContext = requireNonNull(adapterContext); + outputName = NodeIdentifier.create( + YangConstants.operationOutputQName(actionPath.lastNodeIdentifier().getModule())); this.actionInterface = requireNonNull(actionInterface); this.implementation = requireNonNull(implementation); } @Override @SuppressWarnings({ "rawtypes", "unchecked" }) - public FluentFuture invokeAction(final SchemaPath type, + public ListenableFuture invokeAction(final Absolute type, final DOMDataTreeIdentifier path, final ContainerNode input) { - final FluentFuture> userFuture = implementation.invoke( - codec.fromYangInstanceIdentifier(path.getRootIdentifier()), + final CurrentAdapterSerializer codec = adapterContext.currentSerializer(); + final InstanceIdentifier instance = codec.fromYangInstanceIdentifier(path.getRootIdentifier()); + if (instance == null) { + // Not representable: return an error + LOG.debug("Path {} is not representable in binding, rejecting invocation", path); + return Futures.immediateFuture(new SimpleDOMActionResult(List.of(RpcResultBuilder.newError( + ErrorType.APPLICATION, ErrorTag.INVALID_VALUE, "Supplied path cannot be represented")))); + } + if (instance.isWildcarded()) { + // A wildcard path: return an error + LOG.debug("Path {} maps to a wildcard {}, rejecting invocation", path, instance); + return Futures.immediateFuture(new SimpleDOMActionResult(List.of(RpcResultBuilder.newError( + ErrorType.APPLICATION, ErrorTag.INVALID_VALUE, + "Supplied path does not identify a concrete instance")))); + } + + final ListenableFuture> userFuture = implementation.invoke(instance, codec.fromNormalizedNodeActionInput(actionInterface, input)); if (userFuture instanceof BindingOperationFluentFuture) { // If we are looping back through our future we can skip wrapping. This can happen if application @@ -86,7 +144,7 @@ final class ActionProviderServiceAdapter extends AbstractBindingAdapter