Fix action invocation and registration
[mdsal.git] / binding / mdsal-binding-dom-adapter / src / main / java / org / opendaylight / mdsal / binding / dom / adapter / ActionServiceAdapter.java
index a371947420858d6d227449fbb0015e91e8a62cff..77416b5f1e6c8a5b1590bffa477f95cf71d43e40 100644 (file)
@@ -8,70 +8,63 @@
 package org.opendaylight.mdsal.binding.dom.adapter;
 
 import static com.google.common.base.Preconditions.checkArgument;
-import static com.google.common.base.Preconditions.checkState;
-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 java.lang.reflect.Proxy;
 import java.util.Set;
 import org.eclipse.jdt.annotation.NonNullByDefault;
 import org.opendaylight.mdsal.binding.api.ActionService;
+import org.opendaylight.mdsal.binding.api.ActionSpec;
 import org.opendaylight.mdsal.binding.api.DataTreeIdentifier;
+import org.opendaylight.mdsal.binding.dom.adapter.BindingDOMAdapterBuilder.Factory;
 import org.opendaylight.mdsal.binding.spec.reflect.BindingReflections;
 import org.opendaylight.mdsal.dom.api.DOMActionService;
-import org.opendaylight.yangtools.concepts.Delegator;
+import org.opendaylight.mdsal.dom.api.DOMService;
 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.binding.RpcInput;
-import org.opendaylight.yangtools.yang.binding.RpcOutput;
-import org.opendaylight.yangtools.yang.common.RpcResult;
 
-@Beta
 @NonNullByDefault
 final class ActionServiceAdapter
-        extends AbstractBindingLoadingAdapter<DOMActionService, Class<? extends Action<?, ?, ?>>, ActionAdapter>
+        extends AbstractBindingLoadingAdapter<DOMActionService, ActionSpec<?, ?>, ActionAdapter>
         implements ActionService {
-    private static final class ConstrainedAction implements Delegator<Action<?, ?, ?>>,
-            Action<InstanceIdentifier<?>, RpcInput, RpcOutput> {
-        private final Action<InstanceIdentifier<?>, RpcInput, RpcOutput> delegate;
-        private final Set<? extends DataTreeIdentifier<?>> nodes;
-
-        ConstrainedAction(final Action<?, ?, ?> delegate, final Set<? extends DataTreeIdentifier<?>> nodes) {
-            this.delegate = requireNonNull((Action) delegate);
-            this.nodes = requireNonNull(nodes);
+    private static final class Builder extends BindingDOMAdapterBuilder<ActionService> {
+        Builder(final AdapterContext adapterContext) {
+            super(adapterContext);
         }
 
         @Override
-        public FluentFuture<RpcResult<RpcOutput>> invoke(final InstanceIdentifier<?> path, final RpcInput input) {
-            checkState(nodes.contains(path), "Cannot service %s", path);
-            return delegate.invoke(path, input);
+        public Set<? extends Class<? extends DOMService>> getRequiredDelegates() {
+            return ImmutableSet.of(DOMActionService.class);
         }
 
         @Override
-        public Action<?, ?, ?> getDelegate() {
-            return delegate;
+        protected ActionService createInstance(final ClassToInstanceMap<DOMService> delegates) {
+            return new ActionServiceAdapter(adapterContext(), delegates.getInstance(DOMActionService.class));
         }
     }
 
-    ActionServiceAdapter(final BindingToNormalizedNodeCodec codec, final DOMActionService delegate) {
-        super(codec, delegate);
+    static final Factory<ActionService> BUILDER_FACTORY = Builder::new;
+
+    ActionServiceAdapter(final AdapterContext adapterContext, final DOMActionService delegate) {
+        super(adapterContext, delegate);
     }
 
     @Override
-    public <O extends DataObject, T extends Action<?, ?, ?>> T getActionHandle(final Class<T> actionInterface,
-            final Set<DataTreeIdentifier<O>> nodes) {
-        return !nodes.isEmpty() ? (T) new ConstrainedAction(getActionHandle(actionInterface, ImmutableSet.of()), nodes)
-                : (T) Proxy.newProxyInstance(actionInterface.getClassLoader(), new Class[] { actionInterface },
-                    getAdapter(actionInterface));
+    public <P extends DataObject, A extends Action<InstanceIdentifier<P>, ?, ?>> A getActionHandle(
+            final ActionSpec<A, P> spec, final Set<DataTreeIdentifier<P>> nodes) {
+        final var type = spec.type();
+        final var adapter = getAdapter(spec);
+        return type.cast(Proxy.newProxyInstance(type.getClassLoader(), new Class[] { type },
+            nodes.isEmpty() ? adapter : new ActionAdapterFilter(adapter, Set.copyOf(nodes))));
     }
 
     @Override
-    ActionAdapter loadAdapter(final Class<? extends Action<?, ?, ?>> key) {
-        checkArgument(BindingReflections.isBindingClass(key));
-        checkArgument(key.isInterface(), "Supplied Action type must be an interface.");
-        return new ActionAdapter(getCodec(), getDelegate(), key);
+    ActionAdapter loadAdapter(final ActionSpec<?, ?> key) {
+        final var type = key.type();
+        checkArgument(BindingReflections.isBindingClass(type));
+        checkArgument(type.isInterface(), "Supplied Action type must be an interface.");
+        return new ActionAdapter(adapterContext(), getDelegate(), key);
     }
 }