YANGTOOLS-813: add parent schemapath to error report
[yangtools.git] / yang / yang-data-codec-xml / src / test / java / org / opendaylight / yangtools / yang / data / codec / xml / XmlToNormalizedNodesTest.java
1 /*
2  * Copyright (c) 2016 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
9 package org.opendaylight.yangtools.yang.data.codec.xml;
10
11 import static org.junit.Assert.assertEquals;
12 import static org.junit.Assert.assertNotNull;
13 import static org.junit.Assert.assertTrue;
14 import static org.junit.Assert.fail;
15
16 import com.google.common.collect.ImmutableList;
17 import java.io.IOException;
18 import java.io.InputStream;
19 import java.net.URI;
20 import java.net.URISyntaxException;
21 import java.util.Collections;
22 import java.util.HashMap;
23 import java.util.Map;
24 import javax.xml.parsers.ParserConfigurationException;
25 import javax.xml.stream.XMLInputFactory;
26 import javax.xml.stream.XMLStreamException;
27 import javax.xml.stream.XMLStreamReader;
28 import org.junit.BeforeClass;
29 import org.junit.Test;
30 import org.opendaylight.yangtools.yang.common.QName;
31 import org.opendaylight.yangtools.yang.common.QNameModule;
32 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.AugmentationIdentifier;
33 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier;
34 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifierWithPredicates;
35 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeWithValue;
36 import org.opendaylight.yangtools.yang.data.api.schema.AugmentationNode;
37 import org.opendaylight.yangtools.yang.data.api.schema.ChoiceNode;
38 import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode;
39 import org.opendaylight.yangtools.yang.data.api.schema.LeafNode;
40 import org.opendaylight.yangtools.yang.data.api.schema.LeafSetNode;
41 import org.opendaylight.yangtools.yang.data.api.schema.MapNode;
42 import org.opendaylight.yangtools.yang.data.api.schema.NormalizedNode;
43 import org.opendaylight.yangtools.yang.data.api.schema.stream.NormalizedNodeStreamWriter;
44 import org.opendaylight.yangtools.yang.data.impl.schema.Builders;
45 import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNormalizedNodeStreamWriter;
46 import org.opendaylight.yangtools.yang.data.impl.schema.NormalizedNodeResult;
47 import org.opendaylight.yangtools.yang.model.api.ContainerSchemaNode;
48 import org.opendaylight.yangtools.yang.model.api.SchemaContext;
49 import org.opendaylight.yangtools.yang.model.util.SchemaContextUtil;
50 import org.opendaylight.yangtools.yang.parser.spi.meta.ReactorException;
51 import org.opendaylight.yangtools.yang.test.util.YangParserTestUtils;
52 import org.xml.sax.SAXException;
53
54 public class XmlToNormalizedNodesTest {
55
56     private static SchemaContext schemaContext;
57     private static ContainerSchemaNode outerContainerSchema;
58     private static ContainerSchemaNode parentContainerSchema;
59
60     private static QNameModule fooModule;
61     private static QName parentContainer;
62
63     private static QNameModule bazModule;
64     private static QName outerContainer;
65
66     private static QName myContainer1;
67     private static QName myKeyedList;
68     private static QName myKeyLeaf;
69     private static QName myLeafInList1;
70     private static QName myLeafInList2;
71     private static QName myLeaf1;
72     private static QName myLeafList;
73
74     private static QName myContainer2;
75     private static QName innerContainer;
76     private static QName myLeaf2;
77     private static QName myLeaf3;
78     private static QName myChoice;
79     private static QName myLeafInCase2;
80
81     private static QName myContainer3;
82     private static QName myDoublyKeyedList;
83     private static QName myFirstKeyLeaf;
84     private static QName mySecondKeyLeaf;
85     private static QName myLeafInList3;
86
87     @BeforeClass
88     public static void setup() {
89         fooModule = QNameModule.create(URI.create("foo-namespace"));
90         parentContainer = QName.create(fooModule, "parent-container");
91
92         bazModule = QNameModule.create(URI.create("baz-namespace"));
93         outerContainer = QName.create(bazModule, "outer-container");
94
95         myContainer1 = QName.create(bazModule, "my-container-1");
96         myKeyedList = QName.create(bazModule, "my-keyed-list");
97         myKeyLeaf = QName.create(bazModule, "my-key-leaf");
98         myLeafInList1 = QName.create(bazModule, "my-leaf-in-list-1");
99         myLeafInList2 = QName.create(bazModule, "my-leaf-in-list-2");
100         myLeaf1 = QName.create(bazModule, "my-leaf-1");
101         myLeafList = QName.create(bazModule, "my-leaf-list");
102
103         myContainer2 = QName.create(bazModule, "my-container-2");
104         innerContainer = QName.create(bazModule, "inner-container");
105         myLeaf2 = QName.create(bazModule, "my-leaf-2");
106         myLeaf3 = QName.create(bazModule, "my-leaf-3");
107         myChoice = QName.create(bazModule, "my-choice");
108         myLeafInCase2 = QName.create(bazModule, "my-leaf-in-case-2");
109
110         myContainer3 = QName.create(bazModule, "my-container-3");
111         myDoublyKeyedList = QName.create(bazModule, "my-doubly-keyed-list");
112         myFirstKeyLeaf = QName.create(bazModule, "my-first-key-leaf");
113         mySecondKeyLeaf = QName.create(bazModule, "my-second-key-leaf");
114         myLeafInList3 = QName.create(bazModule, "my-leaf-in-list-3");
115
116         schemaContext = YangParserTestUtils.parseYangResourceDirectory("/");
117         parentContainerSchema = (ContainerSchemaNode) SchemaContextUtil.findNodeInSchemaContext(schemaContext,
118                 ImmutableList.of(parentContainer));
119         outerContainerSchema = (ContainerSchemaNode) SchemaContextUtil.findNodeInSchemaContext(schemaContext,
120                 ImmutableList.of(outerContainer));
121     }
122
123     @Test
124     public void testComplexXmlParsing() throws IOException, URISyntaxException, ReactorException, XMLStreamException,
125             ParserConfigurationException, SAXException {
126         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/baz.xml");
127
128         final XMLInputFactory factory = XMLInputFactory.newInstance();
129         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
130
131         final NormalizedNodeResult result = new NormalizedNodeResult();
132         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
133
134         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, outerContainerSchema);
135         xmlParser.parse(reader);
136
137         xmlParser.flush();
138         xmlParser.close();
139
140         final NormalizedNode<?, ?> transformedInput = result.getResult();
141         assertNotNull(transformedInput);
142
143         final NormalizedNode<?, ?> expectedNormalizedNode = buildOuterContainerNode();
144         assertNotNull(expectedNormalizedNode);
145
146         assertEquals(expectedNormalizedNode, transformedInput);
147     }
148
149     @Test
150     public void testSimpleXmlParsing() throws IOException, URISyntaxException, ReactorException, XMLStreamException,
151             ParserConfigurationException, SAXException {
152         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/foo.xml");
153
154         final XMLInputFactory factory = XMLInputFactory.newInstance();
155         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
156
157         final NormalizedNodeResult result = new NormalizedNodeResult();
158         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
159
160         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, parentContainerSchema);
161         xmlParser.parse(reader);
162
163         final NormalizedNode<?, ?> transformedInput = result.getResult();
164         assertNotNull(transformedInput);
165     }
166
167     @Test
168     public void shouldFailOnDuplicateLeaf() throws ReactorException, XMLStreamException, IOException,
169             ParserConfigurationException, SAXException, URISyntaxException {
170         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/invalid-foo.xml");
171
172         final XMLInputFactory factory = XMLInputFactory.newInstance();
173         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
174
175         final NormalizedNodeResult result = new NormalizedNodeResult();
176         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
177
178         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, parentContainerSchema);
179         try {
180             xmlParser.parse(reader);
181             fail("IllegalStateException should have been thrown because of duplicate leaf.");
182         } catch (IllegalStateException ex) {
183             assertTrue(ex.getMessage().contains("Duplicate element \"decimal64-leaf\" in XML input"));
184         }
185
186     }
187
188     @Test
189     public void shouldFailOnDuplicateAnyXml() throws ReactorException, XMLStreamException, IOException,
190             ParserConfigurationException, SAXException, URISyntaxException {
191         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/invalid-foo-2.xml");
192
193         final XMLInputFactory factory = XMLInputFactory.newInstance();
194         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
195
196         final NormalizedNodeResult result = new NormalizedNodeResult();
197         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
198
199         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, parentContainerSchema);
200         try {
201             xmlParser.parse(reader);
202             fail("IllegalStateException should have been thrown because of duplicate anyxml");
203         } catch (IllegalStateException ex) {
204             assertTrue(ex.getMessage().contains("Duplicate element \"my-anyxml\" in XML input"));
205         }
206     }
207
208     @Test
209     public void shouldFailOnDuplicateContainer() throws ReactorException, XMLStreamException, IOException,
210             ParserConfigurationException, SAXException, URISyntaxException {
211         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/invalid-foo-3.xml");
212
213         final XMLInputFactory factory = XMLInputFactory.newInstance();
214         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
215
216         final NormalizedNodeResult result = new NormalizedNodeResult();
217         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
218
219         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, parentContainerSchema);
220         try {
221             xmlParser.parse(reader);
222             fail("IllegalStateException should have been thrown because of duplicate container");
223         } catch (IllegalStateException ex) {
224             assertTrue(ex.getMessage().contains("Duplicate element \"leaf-container\" in XML input"));
225         }
226     }
227
228     @Test
229     public void shouldFailOnUnterminatedLeafElement() throws ReactorException, XMLStreamException, IOException,
230             ParserConfigurationException, SAXException, URISyntaxException {
231         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/invalid-baz.xml");
232
233         final XMLInputFactory factory = XMLInputFactory.newInstance();
234         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
235
236         final NormalizedNodeResult result = new NormalizedNodeResult();
237         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
238
239         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, outerContainerSchema);
240         try {
241             xmlParser.parse(reader);
242             fail("XMLStreamException should have been thrown because of unterminated leaf element.");
243         } catch (XMLStreamException ex) {
244             assertTrue(ex.getMessage().contains("elementGetText() function expects text only elment but "
245                         + "START_ELEMENT was encountered."));
246         }
247     }
248
249     @Test
250     public void shouldFailOnUnterminatedLeafElement2() throws ReactorException, XMLStreamException, IOException,
251             ParserConfigurationException, SAXException, URISyntaxException {
252         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/invalid-baz-2.xml");
253
254         final XMLInputFactory factory = XMLInputFactory.newInstance();
255         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
256
257         final NormalizedNodeResult result = new NormalizedNodeResult();
258         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
259
260         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, outerContainerSchema);
261         try {
262             xmlParser.parse(reader);
263             fail("XMLStreamException should have been thrown because of unterminated leaf element.");
264         } catch (XMLStreamException ex) {
265             assertTrue(ex.getMessage().contains("The element type \"my-leaf-1\" must be terminated by the matching "
266                         + "end-tag \"</my-leaf-1>\"."));
267         }
268     }
269
270     @Test
271     public void shouldFailOnUnterminatedContainerElement() throws ReactorException, XMLStreamException, IOException,
272             ParserConfigurationException, SAXException, URISyntaxException {
273         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/invalid-baz-4.xml");
274
275         final XMLInputFactory factory = XMLInputFactory.newInstance();
276         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
277
278         final NormalizedNodeResult result = new NormalizedNodeResult();
279         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
280
281         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, outerContainerSchema);
282         try {
283             xmlParser.parse(reader);
284             fail("XMLStreamException should have been thrown because of unterminated container element.");
285         } catch (XMLStreamException ex) {
286             assertTrue(ex.getMessage().contains("The element type \"my-container-1\" must be terminated by the "
287                         + "matching end-tag \"</my-container-1>\"."));
288         }
289     }
290
291     @Test
292     public void shouldFailOnUnknownChildNode() throws ReactorException, XMLStreamException, IOException,
293             ParserConfigurationException, SAXException, URISyntaxException {
294         final InputStream resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream("/invalid-baz-3.xml");
295
296         final XMLInputFactory factory = XMLInputFactory.newInstance();
297         final XMLStreamReader reader = factory.createXMLStreamReader(resourceAsStream);
298
299         final NormalizedNodeResult result = new NormalizedNodeResult();
300         final NormalizedNodeStreamWriter streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
301
302         final XmlParserStream xmlParser = XmlParserStream.create(streamWriter, schemaContext, outerContainerSchema);
303         try {
304             xmlParser.parse(reader);
305             fail("IllegalStateException should have been thrown because of an unknown child node.");
306         } catch (IllegalStateException ex) {
307             assertEquals("Schema for node with name my-container-1 and namespace baz-namespace doesn't exist at "
308                     + "AbsoluteSchemaPath{path=[(baz-namespace)outer-container, (baz-namespace)my-container-1]}",
309                     ex.getMessage());
310         }
311     }
312
313     private NormalizedNode<?, ?> buildOuterContainerNode() {
314         // my-container-1
315         MapNode myKeyedListNode = Builders.mapBuilder().withNodeIdentifier(new NodeIdentifier(myKeyedList))
316                 .withChild(Builders.mapEntryBuilder().withNodeIdentifier(
317                         new NodeIdentifierWithPredicates(myKeyedList, myKeyLeaf, "listkeyvalue1"))
318                         .withChild(Builders.leafBuilder().withNodeIdentifier(new NodeIdentifier(myLeafInList1))
319                                 .withValue("listleafvalue1").build())
320                         .withChild(Builders.leafBuilder().withNodeIdentifier(new NodeIdentifier(myLeafInList2))
321                                 .withValue("listleafvalue2").build()).build())
322                 .withChild(Builders.mapEntryBuilder().withNodeIdentifier(
323                         new NodeIdentifierWithPredicates(myKeyedList, myKeyLeaf, "listkeyvalue2"))
324                         .withChild(Builders.leafBuilder().withNodeIdentifier(new NodeIdentifier(myLeafInList1))
325                                 .withValue("listleafvalue12").build())
326                         .withChild(Builders.leafBuilder().withNodeIdentifier(new NodeIdentifier(myLeafInList2))
327                                 .withValue("listleafvalue22").build()).build()).build();
328
329         LeafNode<?> myLeaf1Node = Builders.leafBuilder().withNodeIdentifier(new NodeIdentifier(myLeaf1))
330                 .withValue("value1").build();
331
332         LeafSetNode<?> myLeafListNode = Builders.leafSetBuilder().withNodeIdentifier(new NodeIdentifier(myLeafList))
333                 .withChild(Builders.leafSetEntryBuilder().withNodeIdentifier(
334                         new NodeWithValue<>(myLeafList, "lflvalue1")).withValue("lflvalue1").build())
335                 .withChild(Builders.leafSetEntryBuilder().withNodeIdentifier(
336                         new NodeWithValue<>(myLeafList, "lflvalue2")).withValue("lflvalue2").build()).build();
337
338         ContainerNode myContainer1Node = Builders.containerBuilder().withNodeIdentifier(
339                 new NodeIdentifier(myContainer1))
340                 .withChild(myKeyedListNode)
341                 .withChild(myLeaf1Node)
342                 .withChild(myLeafListNode).build();
343
344         // my-container-2
345         ContainerNode innerContainerNode = Builders.containerBuilder().withNodeIdentifier(
346                 new NodeIdentifier(innerContainer))
347                 .withChild(Builders.leafBuilder().withNodeIdentifier(new NodeIdentifier(myLeaf2))
348                         .withValue("value2").build()).build();
349
350         LeafNode<?> myLeaf3Node = Builders.leafBuilder().withNodeIdentifier(new NodeIdentifier(myLeaf3))
351                 .withValue("value3").build();
352
353         ChoiceNode myChoiceNode = Builders.choiceBuilder().withNodeIdentifier(new NodeIdentifier(myChoice))
354                 .withChild(Builders.leafBuilder().withNodeIdentifier(new NodeIdentifier(myLeafInCase2))
355                         .withValue("case2value").build()).build();
356
357         ContainerNode myContainer2Node = Builders.containerBuilder().withNodeIdentifier(
358                 new NodeIdentifier(myContainer2))
359                 .withChild(innerContainerNode)
360                 .withChild(myLeaf3Node)
361                 .withChild(myChoiceNode).build();
362
363         // my-container-3
364         Map<QName, Object> keys = new HashMap<>();
365         keys.put(myFirstKeyLeaf, "listkeyvalue1");
366         keys.put(mySecondKeyLeaf, "listkeyvalue2");
367
368         MapNode myDoublyKeyedListNode = Builders.mapBuilder().withNodeIdentifier(new NodeIdentifier(myDoublyKeyedList))
369                 .withChild(Builders.mapEntryBuilder().withNodeIdentifier(
370                         new NodeIdentifierWithPredicates(myDoublyKeyedList, keys))
371                         .withChild(Builders.leafBuilder().withNodeIdentifier(
372                                 new NodeIdentifier(myLeafInList3)).withValue("listleafvalue1").build()).build())
373                 .build();
374
375         AugmentationNode myDoublyKeyedListAugNode = Builders.augmentationBuilder().withNodeIdentifier(
376                 new AugmentationIdentifier(Collections.singleton(myDoublyKeyedList)))
377                 .withChild(myDoublyKeyedListNode).build();
378
379         ContainerNode myContainer3Node = Builders.containerBuilder().withNodeIdentifier(
380                 new NodeIdentifier(myContainer3))
381                 .withChild(myDoublyKeyedListAugNode).build();
382
383         AugmentationNode myContainer3AugNode = Builders.augmentationBuilder().withNodeIdentifier(
384                 new AugmentationIdentifier(Collections.singleton(myContainer3)))
385                 .withChild(myContainer3Node).build();
386
387         ContainerNode outerContainerNode = Builders.containerBuilder().withNodeIdentifier(
388                 new NodeIdentifier(outerContainer))
389                 .withChild(myContainer1Node)
390                 .withChild(myContainer2Node)
391                 .withChild(myContainer3AugNode).build();
392
393         return outerContainerNode;
394     }
395 }