/* * Copyright (c) 2014 Cisco Systems, Inc. and others. All rights reserved. * * This program and the accompanying materials are made available under the * terms of the Eclipse Public License v1.0 which accompanies this distribution, * and is available at http://www.eclipse.org/legal/epl-v10.html */ 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.cache.CacheBuilder; import com.google.common.cache.CacheLoader; import com.google.common.cache.LoadingCache; import com.google.common.collect.ImmutableBiMap; import com.google.common.collect.ImmutableSet; import com.google.common.collect.Iterators; import java.lang.reflect.Method; import java.time.Instant; import java.util.AbstractMap.SimpleEntry; import java.util.Collection; import java.util.HashSet; import java.util.Map.Entry; import java.util.Optional; import java.util.Set; import java.util.concurrent.TimeUnit; import java.util.function.Function; import java.util.stream.Collectors; import org.eclipse.jdt.annotation.NonNull; import org.opendaylight.mdsal.binding.api.DataTreeIdentifier; import org.opendaylight.mdsal.binding.dom.codec.api.BindingCodecTree; import org.opendaylight.mdsal.binding.dom.codec.api.BindingCodecTreeFactory; import org.opendaylight.mdsal.binding.dom.codec.api.BindingCodecTreeNode; import org.opendaylight.mdsal.binding.dom.codec.api.BindingDataObjectCodecTreeNode; import org.opendaylight.mdsal.binding.dom.codec.api.BindingLazyContainerNode; import org.opendaylight.mdsal.binding.dom.codec.api.BindingNormalizedNodeSerializer; import org.opendaylight.mdsal.binding.dom.codec.impl.BindingNormalizedNodeCodecRegistry; import org.opendaylight.mdsal.binding.dom.codec.impl.MissingSchemaException; import org.opendaylight.mdsal.binding.generator.api.ClassLoadingStrategy; import org.opendaylight.mdsal.binding.generator.util.BindingRuntimeContext; import org.opendaylight.mdsal.binding.spec.naming.BindingMapping; import org.opendaylight.mdsal.binding.spec.reflect.BindingReflections; import org.opendaylight.mdsal.dom.api.DOMDataTreeIdentifier; import org.opendaylight.mdsal.dom.api.DOMSchemaService; import org.opendaylight.yangtools.concepts.ListenerRegistration; import org.opendaylight.yangtools.yang.binding.Action; import org.opendaylight.yangtools.yang.binding.DataContainer; import org.opendaylight.yangtools.yang.binding.DataObject; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier; import org.opendaylight.yangtools.yang.binding.InstanceIdentifier.PathArgument; import org.opendaylight.yangtools.yang.binding.Notification; import org.opendaylight.yangtools.yang.binding.RpcInput; import org.opendaylight.yangtools.yang.binding.RpcOutput; import org.opendaylight.yangtools.yang.binding.RpcService; import org.opendaylight.yangtools.yang.common.QNameModule; 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.data.api.schema.NormalizedNode; import org.opendaylight.yangtools.yang.data.impl.codec.DeserializationException; import org.opendaylight.yangtools.yang.data.impl.schema.Builders; import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes; import org.opendaylight.yangtools.yang.model.api.ActionDefinition; import org.opendaylight.yangtools.yang.model.api.DocumentedNode.WithStatus; import org.opendaylight.yangtools.yang.model.api.ListSchemaNode; import org.opendaylight.yangtools.yang.model.api.Module; import org.opendaylight.yangtools.yang.model.api.NotificationDefinition; import org.opendaylight.yangtools.yang.model.api.RpcDefinition; import org.opendaylight.yangtools.yang.model.api.SchemaContext; import org.opendaylight.yangtools.yang.model.api.SchemaContextListener; import org.opendaylight.yangtools.yang.model.api.SchemaPath; import org.slf4j.Logger; import org.slf4j.LoggerFactory; /** * A combinations of {@link BindingCodecTreeFactory} and {@link BindingNormalizedNodeSerializer}, with internal * caching of instance identifiers. * *

* NOTE: this class is non-final to allow controller adapter migration without duplicated code. */ public class BindingToNormalizedNodeCodec implements BindingCodecTreeFactory, BindingNormalizedNodeSerializer, SchemaContextListener, AutoCloseable { private static final long WAIT_DURATION_SEC = 5; private static final Logger LOG = LoggerFactory.getLogger(BindingToNormalizedNodeCodec.class); private final LoadingCache, YangInstanceIdentifier> iiCache = CacheBuilder.newBuilder() .softValues().build(new CacheLoader, YangInstanceIdentifier>() { @Override public YangInstanceIdentifier load(final InstanceIdentifier key) { return toYangInstanceIdentifierBlocking(key); } }); private final BindingNormalizedNodeCodecRegistry codecRegistry; private final ClassLoadingStrategy classLoadingStrategy; private final FutureSchema futureSchema; private ListenerRegistration listenerRegistration; public BindingToNormalizedNodeCodec(final ClassLoadingStrategy classLoadingStrategy, final BindingNormalizedNodeCodecRegistry codecRegistry) { this(classLoadingStrategy, codecRegistry, false); } public BindingToNormalizedNodeCodec(final ClassLoadingStrategy classLoadingStrategy, final BindingNormalizedNodeCodecRegistry codecRegistry, final boolean waitForSchema) { this.classLoadingStrategy = requireNonNull(classLoadingStrategy, "classLoadingStrategy"); this.codecRegistry = requireNonNull(codecRegistry, "codecRegistry"); this.futureSchema = FutureSchema.create(WAIT_DURATION_SEC, TimeUnit.SECONDS, waitForSchema); } public static BindingToNormalizedNodeCodec newInstance(final ClassLoadingStrategy classLoadingStrategy, final DOMSchemaService schemaService) { final BindingNormalizedNodeCodecRegistry codecRegistry = new BindingNormalizedNodeCodecRegistry(); BindingToNormalizedNodeCodec instance = new BindingToNormalizedNodeCodec( classLoadingStrategy, codecRegistry, true); instance.listenerRegistration = schemaService.registerSchemaContextListener(instance); return instance; } protected YangInstanceIdentifier toYangInstanceIdentifierBlocking( final InstanceIdentifier binding) { try { return codecRegistry.toYangInstanceIdentifier(binding); } catch (final MissingSchemaException e) { waitForSchema(decompose(binding), e); return codecRegistry.toYangInstanceIdentifier(binding); } } /** * Translates supplied Binding Instance Identifier into NormalizedNode * instance identifier. * * @param binding * Binding Instance Identifier * @return DOM Instance Identifier * @throws IllegalArgumentException * If supplied Instance Identifier is not valid. */ public final YangInstanceIdentifier toNormalized(final InstanceIdentifier binding) { return codecRegistry.toYangInstanceIdentifier(binding); } @Override public final YangInstanceIdentifier toYangInstanceIdentifier(final InstanceIdentifier binding) { return codecRegistry.toYangInstanceIdentifier(binding); } protected YangInstanceIdentifier toYangInstanceIdentifierCached(final InstanceIdentifier binding) { return iiCache.getUnchecked(binding); } @Override public final Entry> toNormalizedNode( final InstanceIdentifier path, final T data) { try { return codecRegistry.toNormalizedNode(path, data); } catch (final MissingSchemaException e) { waitForSchema(decompose(path), e); return codecRegistry.toNormalizedNode(path, data); } } /** * Converts Binding Map.Entry to DOM Map.Entry. * *

* Same as {@link #toNormalizedNode(InstanceIdentifier, DataObject)}. * * @param binding Map Entry with InstanceIdentifier as key and DataObject as value. * @return DOM Map Entry with {@link YangInstanceIdentifier} as key and {@link NormalizedNode} * as value. */ @SuppressWarnings({"unchecked", "rawtypes"}) public final Entry> toNormalizedNode( final Entry, DataObject> binding) { return toNormalizedNode((InstanceIdentifier) binding.getKey(), binding.getValue()); } @Override public final Entry, DataObject> fromNormalizedNode(final YangInstanceIdentifier path, final NormalizedNode data) { return codecRegistry.fromNormalizedNode(path, data); } @Override public final Notification fromNormalizedNodeNotification(final SchemaPath path, final ContainerNode data) { return codecRegistry.fromNormalizedNodeNotification(path, data); } @Override public final Notification fromNormalizedNodeNotification(final SchemaPath path, final ContainerNode data, final Instant eventInstant) { return codecRegistry.fromNormalizedNodeNotification(path, data, eventInstant); } @Override public final DataObject fromNormalizedNodeRpcData(final SchemaPath path, final ContainerNode data) { return codecRegistry.fromNormalizedNodeRpcData(path, data); } @Override public T fromNormalizedNodeActionInput(final Class> action, final ContainerNode input) { return codecRegistry.fromNormalizedNodeActionInput(action, input); } @Override public T fromNormalizedNodeActionOutput(final Class> action, final ContainerNode output) { return codecRegistry.fromNormalizedNodeActionOutput(action, output); } @Override public final InstanceIdentifier fromYangInstanceIdentifier(final YangInstanceIdentifier dom) { return codecRegistry.fromYangInstanceIdentifier(dom); } @Override public final ContainerNode toNormalizedNodeNotification(final Notification data) { return codecRegistry.toNormalizedNodeNotification(data); } @Override public final ContainerNode toNormalizedNodeRpcData(final DataContainer data) { return codecRegistry.toNormalizedNodeRpcData(data); } @Override public ContainerNode toNormalizedNodeActionInput(final Class> action, final RpcInput input) { return codecRegistry.toNormalizedNodeActionInput(action, input); } @Override public ContainerNode toNormalizedNodeActionOutput(final Class> action, final RpcOutput output) { return codecRegistry.toNormalizedNodeActionOutput(action, output); } @Override public BindingLazyContainerNode toLazyNormalizedNodeActionInput( final Class> action, final NodeIdentifier identifier, final RpcInput input) { return codecRegistry.toLazyNormalizedNodeActionInput(action, identifier, input); } @Override public BindingLazyContainerNode toLazyNormalizedNodeActionOutput( final Class> action, final NodeIdentifier identifier, final RpcOutput output) { return codecRegistry.toLazyNormalizedNodeActionOutput(action, identifier, output); } /** * Returns a Binding-Aware instance identifier from normalized * instance-identifier if it is possible to create representation. * *

* Returns Optional.empty for cases where target is mixin node except * augmentation. */ public final Optional> toBinding(final YangInstanceIdentifier normalized) throws DeserializationException { try { return Optional.ofNullable(codecRegistry.fromYangInstanceIdentifier(normalized)); } catch (final IllegalArgumentException e) { return Optional.empty(); } } public final Optional, DataObject>> toBinding( final @NonNull Entry> normalized) throws DeserializationException { try { /* * This cast is required, due to generics behaviour in openjdk / oracle javac. * *

* InstanceIdentifier has definition InstanceIdentifier, * this means '?' is always  . Eclipse compiler * is able to determine this relationship and treats * Entry, DataObject> and Entry * as assignable. However openjdk / oracle javac treats this two types * as incompatible and issues a compile error. * *

* It is safe to lose generic information and cast it to other generic signature. */ @SuppressWarnings("unchecked") final Entry, DataObject> binding = Entry.class.cast( codecRegistry.fromNormalizedNode(normalized.getKey(), normalized.getValue())); return Optional.ofNullable(binding); } catch (final IllegalArgumentException e) { return Optional.empty(); } } @Override public void onGlobalContextUpdated(final SchemaContext context) { final BindingRuntimeContext runtimeContext = BindingRuntimeContext.create(classLoadingStrategy, context); codecRegistry.onBindingRuntimeContextUpdated(runtimeContext); futureSchema.onRuntimeContextUpdated(runtimeContext); } /** * Deprecated. * * @deprecated Use {@link BindingNormalizedNodeCodecRegistry#deserializeFunction} instead. */ @Deprecated public final Function>, Optional> deserializeFunction( final InstanceIdentifier path) { return codecRegistry.deserializeFunction(path)::apply; } public final BindingNormalizedNodeCodecRegistry getCodecRegistry() { return codecRegistry; } @Override public void close() { if (listenerRegistration != null) { listenerRegistration.close(); } } public final BindingNormalizedNodeCodecRegistry getCodecFactory() { return codecRegistry; } // FIXME: This should be probably part of Binding Runtime context public final ImmutableBiMap getRpcMethodToSchemaPath(final Class key) { final Module module = getModuleBlocking(key); final ImmutableBiMap.Builder ret = ImmutableBiMap.builder(); try { for (final RpcDefinition rpcDef : module.getRpcs()) { final Method method = findRpcMethod(key, rpcDef); ret.put(method, rpcDef.getPath()); } } catch (final NoSuchMethodException e) { throw new IllegalStateException("Rpc defined in model does not have representation in generated class.", e); } return ret.build(); } protected ImmutableBiMap getRpcMethodToSchema(final Class key) { final Module module = getModuleBlocking(key); final ImmutableBiMap.Builder ret = ImmutableBiMap.builder(); try { for (final RpcDefinition rpcDef : module.getRpcs()) { final Method method = findRpcMethod(key, rpcDef); ret.put(method, rpcDef); } } catch (final NoSuchMethodException e) { throw new IllegalStateException("Rpc defined in model does not have representation in generated class.", e); } return ret.build(); } private Module getModuleBlocking(final Class modeledClass) { final QNameModule moduleName = BindingReflections.getQNameModule(modeledClass); BindingRuntimeContext localRuntimeContext = runtimeContext(); Module module = localRuntimeContext == null ? null : localRuntimeContext.getSchemaContext().findModule(moduleName).orElse(null); if (module == null && futureSchema.waitForSchema(moduleName)) { localRuntimeContext = runtimeContext(); checkState(localRuntimeContext != null, "BindingRuntimeContext is not available."); module = localRuntimeContext.getSchemaContext().findModule(moduleName).orElse(null); } if (module != null) { return module; } LOG.debug("Schema for {} is not available; expected module name: {}; BindingRuntimeContext: {}", modeledClass, moduleName, localRuntimeContext); throw new IllegalStateException(String.format("Schema for %s is not available; expected module name: %s; " + "full BindingRuntimeContext available in debug log", modeledClass, moduleName)); } private void waitForSchema(final Collection> binding, final MissingSchemaException exception) { LOG.warn("Blocking thread to wait for schema convergence updates for {} {}", futureSchema.getDuration(), futureSchema.getUnit()); if (!futureSchema.waitForSchema(binding)) { throw exception; } } private Method findRpcMethod(final Class key, final RpcDefinition rpcDef) throws NoSuchMethodException { final String methodName = BindingMapping.getMethodName(rpcDef.getQName()); final Class inputClz = runtimeContext().getClassForSchema(rpcDef.getInput()); return key.getMethod(methodName, inputClz); } @Override public final BindingCodecTree create(final BindingRuntimeContext context) { return codecRegistry.create(context); } @Override public final BindingCodecTree create(final SchemaContext context, final Class... bindingClasses) { return codecRegistry.create(context, bindingClasses); } protected @NonNull Entry, BindingDataObjectCodecTreeNode> getSubtreeCodec( final YangInstanceIdentifier domIdentifier) { final BindingCodecTree currentCodecTree = codecRegistry.getCodecContext(); final InstanceIdentifier bindingPath = codecRegistry.fromYangInstanceIdentifier(domIdentifier); checkArgument(bindingPath != null); /** * If we are able to deserialize YANG instance identifier, getSubtreeCodec must * return non-null value. */ final BindingDataObjectCodecTreeNode codecContext = currentCodecTree.getSubtreeCodec(bindingPath); return new SimpleEntry<>(bindingPath, codecContext); } @SuppressWarnings("unchecked") public final Set> getNotificationClasses(final Set interested) { final Set> result = new HashSet<>(); final BindingRuntimeContext runtimeContext = runtimeContext(); for (final NotificationDefinition notification : runtimeContext.getSchemaContext().getNotifications()) { if (interested.contains(notification.getPath())) { try { result.add((Class) runtimeContext.getClassForSchema(notification)); } catch (final IllegalStateException e) { // Ignore LOG.warn("Class for {} is currently not known.", notification.getPath(), e); } } } return result; } SchemaPath getActionPath(final Class> type) { final ActionDefinition schema = runtimeContext().getActionDefinition(type); checkArgument(schema != null, "Failed to find schema for %s", type); return schema.getPath(); } private BindingRuntimeContext runtimeContext() { return futureSchema.runtimeContext(); } private static Collection> decompose(final InstanceIdentifier path) { return ImmutableSet.copyOf(Iterators.transform(path.getPathArguments().iterator(), PathArgument::getType)); } protected NormalizedNode instanceIdentifierToNode(final YangInstanceIdentifier parentPath) { return ImmutableNodes.fromInstanceId(runtimeContext().getSchemaContext(), parentPath); } /** * This method creates an empty list container of a particular type. * * @deprecated This method is not generally useful, as empty lists do not convey information in YANG (they are * equivalent to non-present lists). It also leaks implementation details to a broader scope and should * never have been public in the first place. */ @Deprecated public NormalizedNode getDefaultNodeFor(final YangInstanceIdentifier parentMapPath) { final BindingCodecTreeNode mapCodec = requireNonNull( codecRegistry.getCodecContext().getSubtreeCodec(parentMapPath), "Codec not found for yang instance identifier: " + parentMapPath); final WithStatus schema = mapCodec.getSchema(); if (schema instanceof ListSchemaNode) { final ListSchemaNode castedSchema = (ListSchemaNode) schema; return castedSchema.isUserOrdered() ? Builders.orderedMapBuilder(castedSchema).build() : Builders.mapBuilder(castedSchema).build(); } throw new IllegalArgumentException("Path does not point to list schema node"); } protected Collection toDOMDataTreeIdentifiers( final Collection> subtrees) { return subtrees.stream().map(this::toDOMDataTreeIdentifier).collect(Collectors.toSet()); } protected DOMDataTreeIdentifier toDOMDataTreeIdentifier(final DataTreeIdentifier path) { final YangInstanceIdentifier domPath = toYangInstanceIdentifierBlocking(path.getRootIdentifier()); return new DOMDataTreeIdentifier(path.getDatastoreType(), domPath); } }