Expose EffectiveModuleStatement prefix mapping
[yangtools.git] / yang / yang-parser-rfc7950 / src / main / java / org / opendaylight / yangtools / yang / parser / rfc7950 / stmt / module / ModuleEffectiveStatementImpl.java
1 /*
2  * Copyright (c) 2015 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.parser.rfc7950.stmt.module;
9
10 import static com.google.common.base.Verify.verifyNotNull;
11
12 import com.google.common.collect.ImmutableMap;
13 import com.google.common.collect.ImmutableMap.Builder;
14 import com.google.common.collect.Maps;
15 import java.util.Map;
16 import java.util.Map.Entry;
17 import java.util.Objects;
18 import java.util.Optional;
19 import org.eclipse.jdt.annotation.NonNull;
20 import org.opendaylight.yangtools.yang.common.QNameModule;
21 import org.opendaylight.yangtools.yang.model.api.meta.EffectiveStatement;
22 import org.opendaylight.yangtools.yang.model.api.meta.IdentifierNamespace;
23 import org.opendaylight.yangtools.yang.model.api.stmt.ImportEffectiveStatement;
24 import org.opendaylight.yangtools.yang.model.api.stmt.ModuleEffectiveStatement;
25 import org.opendaylight.yangtools.yang.model.api.stmt.ModuleStatement;
26 import org.opendaylight.yangtools.yang.model.api.stmt.PrefixEffectiveStatement;
27 import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.AbstractEffectiveModule;
28 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext;
29 import org.opendaylight.yangtools.yang.parser.spi.source.ImportPrefixToModuleCtx;
30 import org.opendaylight.yangtools.yang.parser.spi.source.ModuleCtxToModuleQName;
31
32 final class ModuleEffectiveStatementImpl extends AbstractEffectiveModule<ModuleStatement>
33         implements ModuleEffectiveStatement {
34     private final Map<String, ModuleEffectiveStatement> prefixToModule;
35     private final Map<QNameModule, String> namespaceToPrefix;
36     private final @NonNull QNameModule qnameModule;
37
38     ModuleEffectiveStatementImpl(
39             final StmtContext<String, ModuleStatement, EffectiveStatement<String, ModuleStatement>> ctx) {
40         super(ctx);
41         qnameModule = verifyNotNull(ctx.getFromNamespace(ModuleCtxToModuleQName.class, ctx));
42
43         final String localPrefix = findFirstEffectiveSubstatementArgument(PrefixEffectiveStatement.class).get();
44         final Builder<String, ModuleEffectiveStatement> prefixToModuleBuilder = ImmutableMap.builder();
45         prefixToModuleBuilder.put(localPrefix, this);
46
47         streamEffectiveSubstatements(ImportEffectiveStatement.class)
48                 .map(imp -> imp.findFirstEffectiveSubstatementArgument(PrefixEffectiveStatement.class).get())
49                 .forEach(prefix -> {
50                     final StmtContext<?, ?, ?> importedCtx =
51                             verifyNotNull(ctx.getFromNamespace(ImportPrefixToModuleCtx.class, prefix),
52                                 "Failed to resolve prefix %s", prefix);
53                     prefixToModuleBuilder.put(prefix, (ModuleEffectiveStatement) importedCtx.buildEffective());
54                 });
55         prefixToModule = prefixToModuleBuilder.build();
56
57         final Map<QNameModule, String> tmp = Maps.newLinkedHashMapWithExpectedSize(prefixToModule.size() + 1);
58         tmp.put(qnameModule, localPrefix);
59         for (Entry<String, ModuleEffectiveStatement> e : prefixToModule.entrySet()) {
60             tmp.putIfAbsent(e.getValue().localQNameModule(), e.getKey());
61         }
62         namespaceToPrefix = ImmutableMap.copyOf(tmp);
63     }
64
65     @Override
66     public @NonNull QNameModule localQNameModule() {
67         return qnameModule;
68     }
69
70     @Override
71     public @NonNull QNameModule getQNameModule() {
72         return qnameModule;
73     }
74
75     @Override
76     @SuppressWarnings("unchecked")
77     public <K, V, N extends IdentifierNamespace<K, V>> Optional<? extends Map<K, V>> getNamespaceContents(
78             final @NonNull Class<N> namespace) {
79         if (PrefixToEffectiveModuleNamespace.class.equals(namespace)) {
80             return Optional.of((Map<K, V>) prefixToModule);
81         }
82         if (QNameModuleToPrefixNamespace.class.equals(namespace)) {
83             return Optional.of((Map<K, V>) namespaceToPrefix);
84         }
85         return super.getNamespaceContents(namespace);
86     }
87
88     @Override
89     public int hashCode() {
90         final int prime = 31;
91         int result = 1;
92         result = prime * result + Objects.hashCode(getName());
93         result = prime * result + Objects.hashCode(getYangVersion());
94         result = prime * result + Objects.hashCode(qnameModule);
95         return result;
96     }
97
98     @Override
99     public boolean equals(final Object obj) {
100         if (this == obj) {
101             return true;
102         }
103         if (!(obj instanceof ModuleEffectiveStatementImpl)) {
104             return false;
105         }
106         ModuleEffectiveStatementImpl other = (ModuleEffectiveStatementImpl) obj;
107         if (!Objects.equals(getName(), other.getName())) {
108             return false;
109         }
110         if (!qnameModule.equals(other.qnameModule)) {
111             return false;
112         }
113         if (!Objects.equals(getYangVersion(), other.getYangVersion())) {
114             return false;
115         }
116         return true;
117     }
118 }