Do not pretty-print body class
[yangtools.git] / yang / yang-data-impl / src / main / java / org / opendaylight / yangtools / yang / data / impl / schema / builder / impl / ImmutableMapEntryNodeBuilder.java
1 /*
2  * Copyright (c) 2013 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.impl.schema.builder.impl;
9
10 import java.util.Collection;
11 import java.util.LinkedHashMap;
12 import java.util.Map;
13 import java.util.Map.Entry;
14 import org.eclipse.jdt.annotation.NonNull;
15 import org.opendaylight.yangtools.yang.common.QName;
16 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.AugmentationIdentifier;
17 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifierWithPredicates;
18 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.PathArgument;
19 import org.opendaylight.yangtools.yang.data.api.schema.DataContainerChild;
20 import org.opendaylight.yangtools.yang.data.api.schema.LeafNode;
21 import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode;
22 import org.opendaylight.yangtools.yang.data.api.schema.builder.DataContainerNodeBuilder;
23 import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes;
24 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.valid.DataValidationException;
25 import org.opendaylight.yangtools.yang.data.impl.schema.nodes.AbstractImmutableDataContainerNode;
26 import org.slf4j.Logger;
27 import org.slf4j.LoggerFactory;
28
29 public class ImmutableMapEntryNodeBuilder
30         extends AbstractImmutableDataContainerNodeBuilder<NodeIdentifierWithPredicates, MapEntryNode> {
31     private static final Logger LOG = LoggerFactory.getLogger(ImmutableMapEntryNodeBuilder.class);
32     protected final Map<QName, PathArgument> childrenQNamesToPaths;
33
34     protected ImmutableMapEntryNodeBuilder() {
35         this.childrenQNamesToPaths = new LinkedHashMap<>();
36     }
37
38     protected ImmutableMapEntryNodeBuilder(final int sizeHint) {
39         super(sizeHint);
40         this.childrenQNamesToPaths = new LinkedHashMap<>(sizeHint);
41     }
42
43     protected ImmutableMapEntryNodeBuilder(final ImmutableMapEntryNode node) {
44         super(node);
45         this.childrenQNamesToPaths = new LinkedHashMap<>();
46         fillQNames(node.body(), childrenQNamesToPaths);
47     }
48
49     public static @NonNull DataContainerNodeBuilder<NodeIdentifierWithPredicates, MapEntryNode> create() {
50         return new ImmutableMapEntryNodeBuilder();
51     }
52
53     public static @NonNull DataContainerNodeBuilder<NodeIdentifierWithPredicates, MapEntryNode> create(
54             final int sizeHint) {
55         return new ImmutableMapEntryNodeBuilder(sizeHint);
56     }
57
58     public static @NonNull DataContainerNodeBuilder<NodeIdentifierWithPredicates, MapEntryNode> create(
59             final MapEntryNode node) {
60         if (!(node instanceof ImmutableMapEntryNode)) {
61             throw new UnsupportedOperationException(String.format("Cannot initialize from class %s", node.getClass()));
62         }
63
64         return new ImmutableMapEntryNodeBuilder((ImmutableMapEntryNode)node);
65     }
66
67     private static void fillQNames(final Iterable<DataContainerChild> iterable, final Map<QName, PathArgument> out) {
68         for (final DataContainerChild child : iterable) {
69             putQName(out, child);
70         }
71     }
72
73     private static void putQName(final Map<QName, PathArgument> map, final DataContainerChild child) {
74         // Augmentation nodes cannot be keys, and do not have to be present in childrenQNamesToPaths map
75         final PathArgument identifier = child.getIdentifier();
76         if (!(identifier instanceof AugmentationIdentifier)) {
77             map.put(identifier.getNodeType(), identifier);
78         }
79     }
80
81     @Override
82     public DataContainerNodeBuilder<NodeIdentifierWithPredicates, MapEntryNode> withValue(
83             final Collection<DataContainerChild> withValue) {
84         fillQNames(withValue, childrenQNamesToPaths);
85         return super.withValue(withValue);
86     }
87
88     @Override
89     public DataContainerNodeBuilder<NodeIdentifierWithPredicates, MapEntryNode> withChild(
90             final DataContainerChild child) {
91         putQName(childrenQNamesToPaths, child);
92         return super.withChild(child);
93     }
94
95     @Override
96     public MapEntryNode build() {
97         for (final Entry<QName, Object> key : getNodeIdentifier().entrySet()) {
98             final DataContainerChild childNode = getChild(childrenQNamesToPaths.get(key.getKey()));
99
100             // We have enough information to fill-in missing leaf nodes, so let's do that
101             if (childNode == null) {
102                 LeafNode<Object> leaf = ImmutableNodes.leafNode(key.getKey(), key.getValue());
103                 LOG.debug("Adding leaf {} implied by key {}", leaf, key);
104                 withChild(leaf);
105             } else {
106                 DataValidationException.checkListKey(getNodeIdentifier(), key.getKey(), key.getValue(),
107                     childNode.body());
108             }
109         }
110
111         return new ImmutableMapEntryNode(getNodeIdentifier(), buildValue());
112     }
113
114     private static final class ImmutableMapEntryNode
115             extends AbstractImmutableDataContainerNode<NodeIdentifierWithPredicates, MapEntryNode>
116             implements MapEntryNode {
117
118         ImmutableMapEntryNode(final NodeIdentifierWithPredicates nodeIdentifier,
119                 final Map<PathArgument, Object> children) {
120             super(children, nodeIdentifier);
121         }
122
123         @Override
124         protected Class<MapEntryNode> implementedType() {
125             return MapEntryNode.class;
126         }
127     }
128 }