Fix identity-ref value parsing/serialization
[yangtools.git] / codec / yang-data-codec-xml / src / main / java / org / opendaylight / yangtools / yang / data / codec / xml / XmlStringInstanceIdentifierCodec.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 package org.opendaylight.yangtools.yang.data.codec.xml;
9
10 import static java.util.Objects.requireNonNull;
11
12 import java.util.ArrayDeque;
13 import java.util.Deque;
14 import java.util.Iterator;
15 import javax.xml.namespace.NamespaceContext;
16 import javax.xml.stream.XMLStreamException;
17 import javax.xml.stream.XMLStreamWriter;
18 import org.eclipse.jdt.annotation.NonNull;
19 import org.opendaylight.yangtools.yang.common.XMLNamespace;
20 import org.opendaylight.yangtools.yang.data.api.YangInstanceIdentifier;
21 import org.opendaylight.yangtools.yang.data.util.AbstractModuleStringInstanceIdentifierCodec;
22 import org.opendaylight.yangtools.yang.data.util.DataSchemaContextTree;
23 import org.opendaylight.yangtools.yang.model.api.DataSchemaNode;
24 import org.opendaylight.yangtools.yang.model.api.EffectiveModelContext;
25 import org.opendaylight.yangtools.yang.model.api.LeafListSchemaNode;
26 import org.opendaylight.yangtools.yang.model.api.LeafSchemaNode;
27 import org.opendaylight.yangtools.yang.model.api.Module;
28 import org.opendaylight.yangtools.yang.model.util.LeafrefResolver;
29
30 final class XmlStringInstanceIdentifierCodec extends AbstractModuleStringInstanceIdentifierCodec
31         implements XmlCodec<YangInstanceIdentifier> {
32
33     private static final ThreadLocal<Deque<NamespaceContext>> TL_CONTEXT = new ThreadLocal<>();
34
35     private final @NonNull DataSchemaContextTree dataContextTree;
36     private final @NonNull XmlCodecFactory codecFactory;
37     private final @NonNull EffectiveModelContext context;
38
39     XmlStringInstanceIdentifierCodec(final EffectiveModelContext context, final XmlCodecFactory xmlCodecFactory) {
40         this.context = requireNonNull(context);
41         this.dataContextTree = DataSchemaContextTree.from(context);
42         this.codecFactory = requireNonNull(xmlCodecFactory);
43     }
44
45     @Override
46     protected Module moduleForPrefix(final String prefix) {
47         final String prefixedNS = getNamespaceContext().getNamespaceURI(prefix);
48         final Iterator<? extends Module> modules = context.findModules(XMLNamespace.of(prefixedNS)).iterator();
49         return modules.hasNext() ? modules.next() : null;
50     }
51
52     @Override
53     protected String prefixForNamespace(final XMLNamespace namespace) {
54         final Iterator<? extends Module> modules = context.findModules(namespace).iterator();
55         return modules.hasNext() ? modules.next().getName() : null;
56     }
57
58     @Override
59     protected DataSchemaContextTree getDataContextTree() {
60         return dataContextTree;
61     }
62
63     @Override
64     protected Object deserializeKeyValue(final DataSchemaNode schemaNode, final LeafrefResolver resolver,
65             final String value) {
66         requireNonNull(schemaNode, "schemaNode cannot be null");
67         final XmlCodec<?> objectXmlCodec;
68         if (schemaNode instanceof LeafSchemaNode leafSchemaNode) {
69             objectXmlCodec = codecFactory.codecFor(leafSchemaNode, resolver);
70         } else if (schemaNode instanceof LeafListSchemaNode leafListSchemaNode) {
71             objectXmlCodec = codecFactory.codecFor(leafListSchemaNode, resolver);
72         } else {
73             throw new IllegalArgumentException("schemaNode " + schemaNode
74                     + " must be of type LeafSchemaNode or LeafListSchemaNode");
75         }
76         return objectXmlCodec.parseValue(getNamespaceContext(), value);
77     }
78
79     @Override
80     public Class<YangInstanceIdentifier> getDataType() {
81         return YangInstanceIdentifier.class;
82     }
83
84     @Override
85     public YangInstanceIdentifier parseValue(final NamespaceContext ctx, final String str) {
86         pushNamespaceContext(ctx);
87         try {
88             return deserialize(str);
89         } finally {
90             popNamespaceContext();
91         }
92     }
93
94     @Override
95     public void writeValue(final XMLStreamWriter ctx, final YangInstanceIdentifier value)
96             throws XMLStreamException {
97         ctx.writeCharacters(serialize(value));
98     }
99
100     private static NamespaceContext getNamespaceContext() {
101         return TL_CONTEXT.get().getFirst();
102     }
103
104     private static void popNamespaceContext() {
105         final Deque<NamespaceContext> stack = TL_CONTEXT.get();
106         stack.pop();
107         if (stack.isEmpty()) {
108             TL_CONTEXT.set(null);
109         }
110     }
111
112     private static void pushNamespaceContext(final NamespaceContext context) {
113         Deque<NamespaceContext> stack = TL_CONTEXT.get();
114         if (stack == null) {
115             stack = new ArrayDeque<>(1);
116             TL_CONTEXT.set(stack);
117         }
118         stack.push(context);
119     }
120 }