2 * Copyright (c) 2014 Cisco Systems, Inc. and others. All rights reserved.
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
8 package org.opendaylight.yangtools.yang.data.impl.schema;
10 import com.google.common.base.Preconditions;
11 import java.util.ArrayDeque;
12 import java.util.Collection;
13 import java.util.Deque;
14 import javax.annotation.Nonnull;
15 import javax.xml.transform.dom.DOMSource;
16 import org.opendaylight.yangtools.yang.common.QName;
17 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.AugmentationIdentifier;
18 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier;
19 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifierWithPredicates;
20 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.PathArgument;
21 import org.opendaylight.yangtools.yang.data.api.schema.AnyXmlNode;
22 import org.opendaylight.yangtools.yang.data.api.schema.ChoiceNode;
23 import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode;
24 import org.opendaylight.yangtools.yang.data.api.schema.LeafNode;
25 import org.opendaylight.yangtools.yang.data.api.schema.LeafSetEntryNode;
26 import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode;
27 import org.opendaylight.yangtools.yang.data.api.schema.MapNode;
28 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
29 import org.opendaylight.yangtools.yang.data.api.schema.OrderedMapNode;
30 import org.opendaylight.yangtools.yang.data.api.schema.UnkeyedListEntryNode;
31 import org.opendaylight.yangtools.yang.data.api.schema.UnkeyedListNode;
32 import org.opendaylight.yangtools.yang.data.api.schema.YangModeledAnyXmlNode;
33 import org.opendaylight.yangtools.yang.data.api.schema.stream.NormalizedNodeStreamWriter;
34 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.CollectionNodeBuilder;
35 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.DataContainerNodeAttrBuilder;
36 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.DataContainerNodeBuilder;
37 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.ListNodeBuilder;
38 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.NormalizedNodeBuilder;
39 import org.opendaylight.yangtools.yang.data.impl.schema.builder.api.NormalizedNodeContainerBuilder;
40 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableAnyXmlNodeBuilder;
41 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableAugmentationNodeBuilder;
42 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableChoiceNodeBuilder;
43 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableContainerNodeBuilder;
44 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableLeafSetNodeBuilder;
45 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableMapEntryNodeBuilder;
46 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableMapNodeBuilder;
47 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableOrderedLeafSetNodeBuilder;
48 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableOrderedMapNodeBuilder;
49 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableUnkeyedListEntryNodeBuilder;
50 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableUnkeyedListNodeBuilder;
51 import org.opendaylight.yangtools.yang.data.impl.schema.builder.impl.ImmutableYangModeledAnyXmlNodeBuilder;
52 import org.opendaylight.yangtools.yang.data.util.LeafInterner;
53 import org.opendaylight.yangtools.yang.model.api.DataSchemaNode;
54 import org.opendaylight.yangtools.yang.model.api.LeafSchemaNode;
55 import org.opendaylight.yangtools.yang.model.api.YangModeledAnyXmlSchemaNode;
59 * Implementation of {@link NormalizedNodeStreamWriter}, which constructs
60 * immutable instances of {@link NormalizedNode}s.
62 * This writer supports two modes of behaviour one is using {@link #from(NormalizedNodeResult)}
63 * where resulting NormalizedNode will be stored in supplied result object.
65 * Other mode of operation is using {@link #from(NormalizedNodeContainerBuilder)},
66 * where all created nodes will be written to this builder.
70 public class ImmutableNormalizedNodeStreamWriter implements NormalizedNodeStreamWriter {
72 @SuppressWarnings("rawtypes")
73 private final Deque<NormalizedNodeContainerBuilder> builders = new ArrayDeque<>();
74 private DataSchemaNode nextSchema;
76 @SuppressWarnings("rawtypes")
77 protected ImmutableNormalizedNodeStreamWriter(final NormalizedNodeContainerBuilder topLevelBuilder) {
78 builders.push(topLevelBuilder);
81 protected ImmutableNormalizedNodeStreamWriter(final NormalizedNodeResult result) {
82 this(new NormalizedNodeResultBuilder(result));
86 * Creates a {@link NormalizedNodeStreamWriter} which creates instances of supplied
87 * {@link NormalizedNode}s and writes them to supplied builder as child nodes.
89 * Type of supplied {@link NormalizedNodeContainerBuilder} affects,
90 * which events could be emitted in order to ensure proper construction of
93 * @param builder Builder to which data will be written.
94 * @return {@link NormalizedNodeStreamWriter} which writes data
96 public static NormalizedNodeStreamWriter from(final NormalizedNodeContainerBuilder<?, ?, ?, ?> builder) {
97 return new ImmutableNormalizedNodeStreamWriter(builder);
101 * Creates a {@link NormalizedNodeStreamWriter} which creates one instance of top
102 * level {@link NormalizedNode} (type of NormalizedNode) is determined by first
105 * Result is built when {@link #endNode()} associated with that start event
108 * Writer properly creates also nested {@link NormalizedNode} instances,
109 * if their are supported inside the scope of first event.
111 * This method is useful for clients, which knows there will be one
112 * top level node written, but does not know which type of {@link NormalizedNode}
115 * @param result {@link NormalizedNodeResult} object which will hold result value.
116 * @return {@link NormalizedNodeStreamWriter} which will write item to supplied result holder.
118 public static NormalizedNodeStreamWriter from(final NormalizedNodeResult result) {
119 return new ImmutableNormalizedNodeStreamWriter(result);
122 protected Deque<NormalizedNodeContainerBuilder> getBuilders() {
126 @SuppressWarnings("rawtypes")
127 protected NormalizedNodeContainerBuilder getCurrent() {
128 return builders.peek();
131 @SuppressWarnings("rawtypes")
132 private void enter(final NormalizedNodeContainerBuilder next) {
137 @SuppressWarnings("unchecked")
138 protected void writeChild(final NormalizedNode<?, ?> child) {
139 getCurrent().addChild(child);
143 @SuppressWarnings({"rawtypes","unchecked"})
144 public void endNode() {
145 final NormalizedNodeContainerBuilder finishedBuilder = builders.poll();
146 Preconditions.checkState(finishedBuilder != null, "Node which should be closed does not exists.");
147 final NormalizedNodeContainerBuilder current = getCurrent();
148 Preconditions.checkState(current != null, "Reached top level node, which could not be closed in this writer.");
149 final NormalizedNode<PathArgument, ?> product = finishedBuilder.build();
150 current.addChild(product);
155 public void leafNode(final NodeIdentifier name, final Object value) {
156 checkDataNodeContainer();
158 final LeafNode<Object> sample = ImmutableNodes.leafNode(name, value);
159 final LeafNode<?> node;
160 if (nextSchema instanceof LeafSchemaNode) {
161 node = LeafInterner.forSchema((LeafSchemaNode)nextSchema).intern(sample);
171 public void startLeafSet(final NodeIdentifier name, final int childSizeHint) {
172 checkDataNodeContainer();
173 final ListNodeBuilder<Object, LeafSetEntryNode<Object>> builder = UNKNOWN_SIZE == childSizeHint ?
174 InterningLeafSetNodeBuilder.create(nextSchema) :
175 InterningLeafSetNodeBuilder.create(nextSchema, childSizeHint);
176 builder.withNodeIdentifier(name);
181 public void leafSetEntryNode(final QName name, final Object value) {
182 if (getCurrent() instanceof ImmutableOrderedLeafSetNodeBuilder) {
183 @SuppressWarnings("unchecked")
184 ListNodeBuilder<Object, LeafSetEntryNode<Object>> builder = ((ImmutableOrderedLeafSetNodeBuilder<Object>) getCurrent());
185 builder.withChildValue(value);
186 } else if (getCurrent() instanceof ImmutableLeafSetNodeBuilder) {
187 @SuppressWarnings("unchecked")
188 ListNodeBuilder<Object, LeafSetEntryNode<Object>> builder = ((ImmutableLeafSetNodeBuilder<Object>) getCurrent());
189 builder.withChildValue(value);
191 throw new IllegalArgumentException("LeafSetEntryNode is not valid for parent " + getCurrent());
198 public void startOrderedLeafSet(final NodeIdentifier name, final int childSizeHint) {
199 checkDataNodeContainer();
200 final ListNodeBuilder<Object, LeafSetEntryNode<Object>> builder = Builders.orderedLeafSetBuilder();
201 builder.withNodeIdentifier(name);
206 public void anyxmlNode(final NodeIdentifier name, final Object value) {
207 checkDataNodeContainer();
209 final AnyXmlNode node = ImmutableAnyXmlNodeBuilder.create().withNodeIdentifier(name)
210 .withValue((DOMSource) value).build();
217 public void startContainerNode(final NodeIdentifier name, final int childSizeHint) {
218 checkDataNodeContainer();
220 final DataContainerNodeAttrBuilder<NodeIdentifier, ContainerNode> builder = UNKNOWN_SIZE == childSizeHint ?
221 ImmutableContainerNodeBuilder.create() : ImmutableContainerNodeBuilder.create(childSizeHint);
222 enter(builder.withNodeIdentifier(name));
226 public void startYangModeledAnyXmlNode(final NodeIdentifier name, final int childSizeHint) {
227 checkDataNodeContainer();
229 Preconditions.checkArgument(nextSchema instanceof YangModeledAnyXmlSchemaNode,
230 "Schema of this node should be instance of YangModeledAnyXmlSchemaNode");
231 final DataContainerNodeAttrBuilder<NodeIdentifier, YangModeledAnyXmlNode> builder = UNKNOWN_SIZE == childSizeHint ? ImmutableYangModeledAnyXmlNodeBuilder
232 .create((YangModeledAnyXmlSchemaNode) nextSchema) : ImmutableYangModeledAnyXmlNodeBuilder.create(
233 (YangModeledAnyXmlSchemaNode) nextSchema, childSizeHint);
234 enter(builder.withNodeIdentifier(name));
238 public void startUnkeyedList(final NodeIdentifier name, final int childSizeHint) {
239 checkDataNodeContainer();
241 final CollectionNodeBuilder<UnkeyedListEntryNode, UnkeyedListNode> builder = UNKNOWN_SIZE == childSizeHint ?
242 ImmutableUnkeyedListNodeBuilder.create() : ImmutableUnkeyedListNodeBuilder.create(childSizeHint);
243 enter(builder.withNodeIdentifier(name));
247 public void startUnkeyedListItem(final NodeIdentifier name, final int childSizeHint) {
248 Preconditions.checkArgument((getCurrent() instanceof NormalizedNodeResultBuilder)
249 || getCurrent() instanceof ImmutableUnkeyedListNodeBuilder);
250 final DataContainerNodeAttrBuilder<NodeIdentifier, UnkeyedListEntryNode> builder = UNKNOWN_SIZE == childSizeHint ?
251 ImmutableUnkeyedListEntryNodeBuilder.create() : ImmutableUnkeyedListEntryNodeBuilder.create(childSizeHint);
252 enter(builder.withNodeIdentifier(name));
256 public void startMapNode(final NodeIdentifier name, final int childSizeHint) {
257 checkDataNodeContainer();
259 final CollectionNodeBuilder<MapEntryNode, MapNode> builder = UNKNOWN_SIZE == childSizeHint ?
260 ImmutableMapNodeBuilder.create() : ImmutableMapNodeBuilder.create(childSizeHint);
261 enter(builder.withNodeIdentifier(name));
265 public void startMapEntryNode(final NodeIdentifierWithPredicates identifier, final int childSizeHint) {
266 if (!(getCurrent() instanceof NormalizedNodeResultBuilder)) {
267 Preconditions.checkArgument(getCurrent() instanceof ImmutableMapNodeBuilder || getCurrent() instanceof ImmutableOrderedMapNodeBuilder);
270 final DataContainerNodeAttrBuilder<NodeIdentifierWithPredicates, MapEntryNode> builder = UNKNOWN_SIZE == childSizeHint ?
271 ImmutableMapEntryNodeBuilder.create() : ImmutableMapEntryNodeBuilder.create(childSizeHint);
272 enter(builder.withNodeIdentifier(identifier));
276 public void startOrderedMapNode(final NodeIdentifier name, final int childSizeHint) {
277 checkDataNodeContainer();
279 final CollectionNodeBuilder<MapEntryNode, OrderedMapNode> builder = UNKNOWN_SIZE == childSizeHint ?
280 ImmutableOrderedMapNodeBuilder.create() : ImmutableOrderedMapNodeBuilder.create(childSizeHint);
281 enter(builder.withNodeIdentifier(name));
285 public void startChoiceNode(final NodeIdentifier name, final int childSizeHint) {
286 checkDataNodeContainer();
288 final DataContainerNodeBuilder<NodeIdentifier, ChoiceNode> builder = UNKNOWN_SIZE == childSizeHint ?
289 ImmutableChoiceNodeBuilder.create() : ImmutableChoiceNodeBuilder.create(childSizeHint);
290 enter(builder.withNodeIdentifier(name));
294 public void startAugmentationNode(final AugmentationIdentifier identifier) {
295 checkDataNodeContainer();
296 Preconditions.checkArgument(!(getCurrent() instanceof ImmutableAugmentationNodeBuilder));
297 enter(Builders.augmentationBuilder().withNodeIdentifier(identifier));
300 private void checkDataNodeContainer() {
301 @SuppressWarnings("rawtypes")
303 NormalizedNodeContainerBuilder current = getCurrent();
304 if (!(current instanceof NormalizedNodeResultBuilder)) {
305 Preconditions.checkArgument(current instanceof DataContainerNodeBuilder<?, ?>, "Invalid nesting of data.");
309 @SuppressWarnings("rawtypes")
310 protected static final class NormalizedNodeResultBuilder implements NormalizedNodeContainerBuilder {
312 private final NormalizedNodeResult result;
314 public NormalizedNodeResultBuilder(final NormalizedNodeResult result) {
315 this.result = result;
319 public NormalizedNodeBuilder withValue(final Object value) {
320 throw new UnsupportedOperationException();
324 public NormalizedNode build() {
325 throw new IllegalStateException("Can not close NormalizedNodeResult");
329 public NormalizedNodeContainerBuilder withNodeIdentifier(final PathArgument nodeIdentifier) {
330 throw new UnsupportedOperationException();
334 public NormalizedNodeContainerBuilder withValue(final Collection value) {
335 throw new UnsupportedOperationException();
339 public NormalizedNodeContainerBuilder addChild(final NormalizedNode child) {
340 result.setResult(child);
345 public NormalizedNodeContainerBuilder removeChild(final PathArgument key) {
346 throw new UnsupportedOperationException();
352 public void flush() {
357 public void close() {
362 public void nextDataSchemaNode(@Nonnull final DataSchemaNode schema) {
363 nextSchema = Preconditions.checkNotNull(schema);