d09d05711aee23d6043bd4e70da8f1d2dd4e6262
[yangtools.git] / data / yang-data-util / src / main / java / org / opendaylight / yangtools / yang / data / util / impl / legacy / DataContainerContextNode.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.yangtools.yang.data.util.impl.legacy;
9
10 import static java.util.Objects.requireNonNull;
11
12 import java.util.concurrent.ConcurrentHashMap;
13 import java.util.concurrent.ConcurrentMap;
14 import org.eclipse.jdt.annotation.NonNull;
15 import org.eclipse.jdt.annotation.Nullable;
16 import org.opendaylight.yangtools.yang.common.QName;
17 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.PathArgument;
18 import org.opendaylight.yangtools.yang.data.util.DataSchemaContextNode;
19 import org.opendaylight.yangtools.yang.model.api.DataNodeContainer;
20 import org.opendaylight.yangtools.yang.model.api.DataSchemaNode;
21 import org.opendaylight.yangtools.yang.model.util.SchemaInferenceStack;
22
23 abstract class DataContainerContextNode extends AbstractInteriorContextNode {
24     private final ConcurrentMap<PathArgument, AbstractDataSchemaContextNode> byArg = new ConcurrentHashMap<>();
25     private final ConcurrentMap<QName, AbstractDataSchemaContextNode> byQName = new ConcurrentHashMap<>();
26     private final DataNodeContainer container;
27
28     DataContainerContextNode(final PathArgument pathArgument, final DataNodeContainer container,
29             final DataSchemaNode schema) {
30         super(pathArgument, schema);
31         this.container = requireNonNull(container);
32     }
33
34     @Override
35     public AbstractDataSchemaContextNode getChild(final PathArgument child) {
36         var potential = byArg.get(child);
37         if (potential != null) {
38             return potential;
39         }
40         potential = fromLocalSchema(child);
41         return register(potential);
42     }
43
44     @Override
45     public AbstractDataSchemaContextNode getChild(final QName child) {
46         AbstractDataSchemaContextNode potential = byQName.get(child);
47         if (potential != null) {
48             return potential;
49         }
50         potential = fromLocalSchemaAndQName(container, child);
51         return register(potential);
52     }
53
54     @Override
55     protected final DataSchemaContextNode enterChild(final QName child, final SchemaInferenceStack stack) {
56         return pushToStack(getChild(child), stack);
57     }
58
59     @Override
60     protected final DataSchemaContextNode enterChild(final PathArgument child, final SchemaInferenceStack stack) {
61         return pushToStack(getChild(child), stack);
62     }
63
64     private static @Nullable DataSchemaContextNode pushToStack(final @Nullable AbstractDataSchemaContextNode child,
65             final @NonNull SchemaInferenceStack stack) {
66         if (child != null) {
67             child.pushToStack(stack);
68         }
69         return child;
70     }
71
72     private AbstractDataSchemaContextNode fromLocalSchema(final PathArgument child) {
73         return fromSchemaAndQNameChecked(container, child.getNodeType());
74     }
75
76     protected AbstractDataSchemaContextNode fromLocalSchemaAndQName(final DataNodeContainer schema,
77             final QName child) {
78         return fromSchemaAndQNameChecked(schema, child);
79     }
80
81     private AbstractDataSchemaContextNode register(final AbstractDataSchemaContextNode potential) {
82         if (potential != null) {
83             // FIXME: use putIfAbsent() to make sure we do not perform accidental overrwrites
84             byArg.put(potential.pathArgument(), potential);
85             for (QName qname : potential.qnameIdentifiers()) {
86                 byQName.put(qname, potential);
87             }
88         }
89         return potential;
90     }
91 }