Expose DefaultQuery class
[mdsal.git] / binding / mdsal-binding-dom-adapter / src / main / java / org / opendaylight / mdsal / binding / dom / adapter / LazySerializedContainerNode.java
1 /*
2  * Copyright (c) 2015 Cisco Systems, Inc. and others. All rights reserved.
3  *
4  * This program and the accompanying materials are made available under the
5  * terms of the Eclipse Public License v1.0 which accompanies this distribution,
6  * and is available at http://www.eclipse.org/legal/epl-v10.html
7  */
8 package org.opendaylight.mdsal.binding.dom.adapter;
9
10 import static java.util.Objects.requireNonNull;
11
12 import java.util.Optional;
13 import org.opendaylight.mdsal.binding.dom.codec.api.BindingNormalizedNodeSerializer;
14 import org.opendaylight.mdsal.binding.dom.codec.spi.AbstractBindingLazyContainerNode;
15 import org.opendaylight.yangtools.yang.binding.DataObject;
16 import org.opendaylight.yangtools.yang.common.QName;
17 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier;
18 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.PathArgument;
19 import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode;
20 import org.opendaylight.yangtools.yang.data.api.schema.DataContainerChild;
21 import org.opendaylight.yangtools.yang.data.api.schema.LeafNode;
22
23 /*
24  * FIXME: This is a bit of functionality which should really live in binding-dom-codec, but for to happen we need
25  *        to extend BindingNormalizedNodeSerializer with the concept of a routing context -- which would be deprecated,
26  *        as we want to move to actions in the long term.
27  *
28  *        Even then, BindingNormalizedNodeCodecRegistry provides background updates to the context used in
29  *        deserialization, which is currently being used.
30  */
31 class LazySerializedContainerNode
32         extends AbstractBindingLazyContainerNode<DataObject, BindingNormalizedNodeSerializer> {
33
34     private LazySerializedContainerNode(final QName identifier, final DataObject binding,
35             final BindingNormalizedNodeSerializer codec) {
36         super(NodeIdentifier.create(identifier), binding, requireNonNull(codec));
37     }
38
39     static ContainerNode create(final QName rpcName, final DataObject data,
40             final BindingNormalizedNodeSerializer codec) {
41         return data == null ? null : new LazySerializedContainerNode(rpcName, data, codec);
42     }
43
44     static ContainerNode withContextRef(final QName rpcName, final DataObject data,
45             final LeafNode<?> contextRef, final BindingNormalizedNodeSerializer serializer) {
46         return new WithContextRef(rpcName, data, contextRef, serializer);
47     }
48
49     @Override
50     protected final ContainerNode computeContainerNode(final BindingNormalizedNodeSerializer context) {
51         return context.toNormalizedNodeRpcData(getDataObject());
52     }
53
54     /**
55      * Lazy Serialized Node with pre-cached serialized leaf holding routing information.
56      */
57     private static final class WithContextRef extends LazySerializedContainerNode {
58         private final LeafNode<?> contextRef;
59
60         protected WithContextRef(final QName identifier, final DataObject binding, final LeafNode<?> contextRef,
61                 final BindingNormalizedNodeSerializer codec) {
62             super(identifier, binding, codec);
63             this.contextRef = requireNonNull(contextRef);
64         }
65
66         @Override
67         public Optional<DataContainerChild<? extends PathArgument, ?>> getChild(final PathArgument child) {
68             // Use pre-cached value of routing field and do not run full serialization if we are accessing it.
69             return contextRef.getIdentifier().equals(child) ?  Optional.of(contextRef) : super.getChild(child);
70         }
71     }
72 }