2 * Copyright (c) 2016 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.codec.xml;
10 import static java.util.Objects.requireNonNull;
11 import static org.junit.Assert.assertEquals;
12 import static org.junit.Assert.assertNotNull;
13 import static org.opendaylight.yangtools.yang.data.impl.schema.Builders.choiceBuilder;
14 import static org.opendaylight.yangtools.yang.data.impl.schema.Builders.containerBuilder;
15 import static org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNodes.leafNode;
17 import java.io.IOException;
18 import java.io.InputStream;
19 import java.io.StringWriter;
20 import java.util.Arrays;
21 import java.util.Collection;
22 import javax.xml.stream.XMLOutputFactory;
23 import javax.xml.stream.XMLStreamWriter;
24 import javax.xml.transform.OutputKeys;
25 import javax.xml.transform.Transformer;
26 import javax.xml.transform.TransformerException;
27 import javax.xml.transform.TransformerFactory;
28 import javax.xml.transform.TransformerFactoryConfigurationError;
29 import javax.xml.transform.dom.DOMResult;
30 import javax.xml.transform.dom.DOMSource;
31 import javax.xml.transform.stream.StreamResult;
32 import org.custommonkey.xmlunit.Diff;
33 import org.custommonkey.xmlunit.ElementNameAndTextQualifier;
34 import org.custommonkey.xmlunit.IgnoreTextAndAttributeValuesDifferenceListener;
35 import org.custommonkey.xmlunit.XMLUnit;
36 import org.junit.Test;
37 import org.junit.runner.RunWith;
38 import org.junit.runners.Parameterized;
39 import org.opendaylight.yangtools.util.xml.UntrustedXML;
40 import org.opendaylight.yangtools.yang.common.QName;
41 import org.opendaylight.yangtools.yang.common.QNameModule;
42 import org.opendaylight.yangtools.yang.common.Revision;
43 import org.opendaylight.yangtools.yang.common.Uint32;
44 import org.opendaylight.yangtools.yang.common.XMLNamespace;
45 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifier;
46 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeIdentifierWithPredicates;
47 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier.NodeWithValue;
48 import org.opendaylight.yangtools.yang.data.api.schema.ContainerNode;
49 import org.opendaylight.yangtools.yang.data.api.schema.LeafNode;
50 import org.opendaylight.yangtools.yang.data.api.schema.LeafSetEntryNode;
51 import org.opendaylight.yangtools.yang.data.api.schema.MapEntryNode;
52 import org.opendaylight.yangtools.yang.data.api.schema.SystemLeafSetNode;
53 import org.opendaylight.yangtools.yang.data.api.schema.SystemMapNode;
54 import org.opendaylight.yangtools.yang.data.api.schema.builder.CollectionNodeBuilder;
55 import org.opendaylight.yangtools.yang.data.api.schema.builder.DataContainerNodeBuilder;
56 import org.opendaylight.yangtools.yang.data.api.schema.builder.ListNodeBuilder;
57 import org.opendaylight.yangtools.yang.data.api.schema.builder.NormalizedNodeBuilder;
58 import org.opendaylight.yangtools.yang.data.api.schema.stream.NormalizedNodeStreamWriter;
59 import org.opendaylight.yangtools.yang.data.api.schema.stream.NormalizedNodeWriter;
60 import org.opendaylight.yangtools.yang.data.impl.schema.Builders;
61 import org.opendaylight.yangtools.yang.data.impl.schema.ImmutableNormalizedNodeStreamWriter;
62 import org.opendaylight.yangtools.yang.data.impl.schema.NormalizationResultHolder;
63 import org.opendaylight.yangtools.yang.model.api.EffectiveModelContext;
64 import org.opendaylight.yangtools.yang.model.util.SchemaInferenceStack.Inference;
65 import org.opendaylight.yangtools.yang.test.util.YangParserTestUtils;
66 import org.w3c.dom.Document;
67 import org.w3c.dom.Node;
68 import org.xml.sax.SAXException;
70 @RunWith(Parameterized.class)
71 public class NormalizedNodeXmlTranslationTest {
72 private final EffectiveModelContext schema;
74 @Parameterized.Parameters()
75 public static Collection<Object[]> data() {
76 return Arrays.asList(new Object[][] {
77 { "/schema/augment_choice_hell.yang", "/schema/augment_choice_hell_ok.xml", augmentChoiceHell() },
78 { "/schema/augment_choice_hell.yang", "/schema/augment_choice_hell_ok2.xml", null },
79 { "/schema/augment_choice_hell.yang", "/schema/augment_choice_hell_ok3.xml", augmentChoiceHell2() },
80 { "/schema/test.yang", "/schema/simple.xml", null },
81 { "/schema/test.yang", "/schema/simple2.xml", null },
82 // TODO check attributes
83 { "/schema/test.yang", "/schema/simple_xml_with_attributes.xml", withAttributes() }
87 private static final QNameModule MODULE = QNameModule.create(
88 XMLNamespace.of("urn:opendaylight:params:xml:ns:yang:controller:test"), Revision.of("2014-03-13"));
90 private static ContainerNode augmentChoiceHell2() {
91 final NodeIdentifier container = getNodeIdentifier("container");
92 final QName augmentChoice1QName = QName.create(container.getNodeType(), "augment-choice1");
93 final QName augmentChoice2QName = QName.create(augmentChoice1QName, "augment-choice2");
94 final QName containerQName = QName.create(augmentChoice1QName, "case11-choice-case-container");
95 final QName leafQName = QName.create(augmentChoice1QName, "case11-choice-case-leaf");
97 final NodeIdentifier augmentChoice1Id = new NodeIdentifier(augmentChoice1QName);
98 final NodeIdentifier augmentChoice2Id = new NodeIdentifier(augmentChoice2QName);
99 final NodeIdentifier containerId = new NodeIdentifier(containerQName);
101 return containerBuilder().withNodeIdentifier(container)
102 .withChild(choiceBuilder().withNodeIdentifier(augmentChoice1Id)
103 .withChild(choiceBuilder().withNodeIdentifier(augmentChoice2Id)
104 .withChild(containerBuilder().withNodeIdentifier(containerId)
105 .withChild(leafNode(leafQName, "leaf-value"))
112 private static ContainerNode withAttributes() {
113 final DataContainerNodeBuilder<NodeIdentifier, ContainerNode> b = containerBuilder();
114 b.withNodeIdentifier(getNodeIdentifier("container"));
116 final CollectionNodeBuilder<MapEntryNode, SystemMapNode> listBuilder =
117 Builders.mapBuilder().withNodeIdentifier(getNodeIdentifier("list"));
119 final DataContainerNodeBuilder<NodeIdentifierWithPredicates, MapEntryNode> list1Builder = Builders
120 .mapEntryBuilder().withNodeIdentifier(NodeIdentifierWithPredicates.of(
121 getNodeIdentifier("list").getNodeType(),
122 getNodeIdentifier("uint32InList").getNodeType(), Uint32.valueOf(3)));
123 final NormalizedNodeBuilder<NodeIdentifier, Object, LeafNode<Object>> uint32InListBuilder = Builders
124 .leafBuilder().withNodeIdentifier(getNodeIdentifier("uint32InList"));
126 list1Builder.withChild(uint32InListBuilder.withValue(Uint32.valueOf(3)).build());
128 listBuilder.withChild(list1Builder.build());
129 b.withChild(listBuilder.build());
131 final NormalizedNodeBuilder<NodeIdentifier, Object, LeafNode<Object>> booleanBuilder = Builders
132 .leafBuilder().withNodeIdentifier(getNodeIdentifier("boolean"));
133 booleanBuilder.withValue(Boolean.FALSE);
134 b.withChild(booleanBuilder.build());
136 final ListNodeBuilder<Object, SystemLeafSetNode<Object>> leafListBuilder = Builders.leafSetBuilder()
137 .withNodeIdentifier(getNodeIdentifier("leafList"));
139 final NormalizedNodeBuilder<NodeWithValue, Object, LeafSetEntryNode<Object>> leafList1Builder = Builders
140 .leafSetEntryBuilder().withNodeIdentifier(
141 new NodeWithValue(getNodeIdentifier("leafList").getNodeType(), "a"));
143 leafList1Builder.withValue("a");
145 leafListBuilder.withChild(leafList1Builder.build());
146 b.withChild(leafListBuilder.build());
151 private static ContainerNode augmentChoiceHell() {
153 final DataContainerNodeBuilder<NodeIdentifier, ContainerNode> b = containerBuilder();
154 b.withNodeIdentifier(getNodeIdentifier("container"));
156 b.withChild(choiceBuilder()
157 .withNodeIdentifier(getNodeIdentifier("ch2"))
159 Builders.leafBuilder().withNodeIdentifier(getNodeIdentifier("c2Leaf")).withValue("2").build())
162 .withNodeIdentifier(getNodeIdentifier("c2DeepChoice"))
164 Builders.leafBuilder()
165 .withNodeIdentifier(getNodeIdentifier("c2DeepChoiceCase1Leaf2"))
166 .withValue("2").build()).build()).build());
168 b.withChild(choiceBuilder()
169 .withNodeIdentifier(getNodeIdentifier("ch3"))
171 Builders.leafBuilder().withNodeIdentifier(getNodeIdentifier("c3Leaf")).withValue("3").build())
174 b.withChild(Builders.leafBuilder().withNodeIdentifier(getNodeIdentifier("augLeaf")).withValue("augment")
177 b.withChild(choiceBuilder()
178 .withNodeIdentifier(getNodeIdentifier("ch"))
179 .withChild(Builders.leafBuilder().withNodeIdentifier(getNodeIdentifier("c1Leaf")).withValue("1").build())
180 .withChild(Builders.leafBuilder().withNodeIdentifier(getNodeIdentifier("c1Leaf_AnotherAugment"))
181 .withValue("1").build())
182 .withChild(choiceBuilder()
183 .withNodeIdentifier(getNodeIdentifier("deepChoice"))
184 .withChild(Builders.leafBuilder()
185 .withNodeIdentifier(getNodeIdentifier("deepLeafc1"))
186 .withValue("1").build()).build())
192 private static NodeIdentifier getNodeIdentifier(final String localName) {
193 return new NodeIdentifier(QName.create(MODULE, localName));
196 private final ContainerNode expectedNode;
197 private final String xmlPath;
199 public NormalizedNodeXmlTranslationTest(final String yangPath, final String xmlPath,
200 final ContainerNode expectedNode) {
201 schema = YangParserTestUtils.parseYangResource(yangPath);
202 this.xmlPath = xmlPath;
203 this.expectedNode = expectedNode;
207 public void testTranslationRepairing() throws Exception {
208 testTranslation(TestFactories.REPAIRING_OUTPUT_FACTORY);
212 public void testTranslation() throws Exception {
213 testTranslation(TestFactories.DEFAULT_OUTPUT_FACTORY);
216 private void testTranslation(final XMLOutputFactory factory) throws Exception {
217 final var resourceAsStream = XmlToNormalizedNodesTest.class.getResourceAsStream(xmlPath);
219 final var reader = UntrustedXML.createXMLStreamReader(resourceAsStream);
221 final var result = new NormalizationResultHolder();
222 final var streamWriter = ImmutableNormalizedNodeStreamWriter.from(result);
223 final var xmlParser = XmlParserStream.create(streamWriter,
224 Inference.ofDataTreePath(schema, QName.create(MODULE, "container")));
225 xmlParser.parse(reader);
227 final var built = result.getResult().data();
228 assertNotNull(built);
230 if (expectedNode != null) {
231 assertEquals(expectedNode, built);
234 final Document document = UntrustedXML.newDocumentBuilder().newDocument();
235 final DOMResult domResult = new DOMResult(document);
237 final XMLStreamWriter xmlStreamWriter = factory.createXMLStreamWriter(domResult);
239 final NormalizedNodeStreamWriter xmlNormalizedNodeStreamWriter = XMLStreamNormalizedNodeStreamWriter
240 .create(xmlStreamWriter, schema);
242 final NormalizedNodeWriter normalizedNodeWriter = NormalizedNodeWriter.forStreamWriter(
243 xmlNormalizedNodeStreamWriter);
245 normalizedNodeWriter.write(built);
247 final Document doc = loadDocument(xmlPath);
249 XMLUnit.setIgnoreWhitespace(true);
250 XMLUnit.setIgnoreComments(true);
251 XMLUnit.setIgnoreAttributeOrder(true);
252 XMLUnit.setNormalize(true);
254 final String expectedXml = toString(doc.getDocumentElement());
255 final String serializedXml = toString(domResult.getNode());
257 final Diff diff = new Diff(expectedXml, serializedXml);
258 diff.overrideDifferenceListener(new IgnoreTextAndAttributeValuesDifferenceListener());
259 diff.overrideElementQualifier(new ElementNameAndTextQualifier());
261 // FIXME the comparison cannot be performed, since the qualifiers supplied by XMlUnit do not work correctly in
263 // We need to implement custom qualifier so that the element ordering does not mess the DIFF
264 // dd.overrideElementQualifier(new MultiLevelElementNameAndTextQualifier(100, true));
265 // assertTrue(dd.toString(), dd.similar());
267 //new XMLTestCase() {}.assertXMLEqual(diff, true);
270 private static Document loadDocument(final String xmlPath) throws IOException, SAXException {
271 final InputStream resourceAsStream = NormalizedNodeXmlTranslationTest.class.getResourceAsStream(xmlPath);
272 return requireNonNull(readXmlToDocument(resourceAsStream));
275 private static Document readXmlToDocument(final InputStream xmlContent) throws IOException, SAXException {
276 final Document doc = UntrustedXML.newDocumentBuilder().parse(xmlContent);
277 doc.getDocumentElement().normalize();
281 private static String toString(final Node xml) {
283 final Transformer transformer = TransformerFactory.newInstance().newTransformer();
284 transformer.setOutputProperty(OutputKeys.INDENT, "yes");
285 transformer.setOutputProperty("{http://xml.apache.org/xslt}indent-amount", "4");
287 final StreamResult result = new StreamResult(new StringWriter());
288 final DOMSource source = new DOMSource(xml);
289 transformer.transform(source, result);
291 return result.getWriter().toString();
292 } catch (IllegalArgumentException | TransformerFactoryConfigurationError | TransformerException e) {
293 throw new RuntimeException("Unable to serialize xml element " + xml, e);