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