Fix isSupportedToBuildEffective() propagation
[yangtools.git] / parser / yang-parser-reactor / src / main / java / org / opendaylight / yangtools / yang / parser / stmt / reactor / StatementDefinitionContext.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.stmt.reactor;
9
10 import static java.util.Objects.requireNonNull;
11
12 import com.google.common.base.MoreObjects;
13 import java.util.HashMap;
14 import java.util.Map;
15 import java.util.Optional;
16 import org.eclipse.jdt.annotation.NonNull;
17 import org.opendaylight.yangtools.yang.common.QName;
18 import org.opendaylight.yangtools.yang.common.QNameModule;
19 import org.opendaylight.yangtools.yang.model.api.meta.ArgumentDefinition;
20 import org.opendaylight.yangtools.yang.model.api.meta.DeclaredStatement;
21 import org.opendaylight.yangtools.yang.model.api.meta.EffectiveStatement;
22 import org.opendaylight.yangtools.yang.model.api.meta.StatementDefinition;
23 import org.opendaylight.yangtools.yang.parser.spi.meta.ImplicitParentAwareStatementSupport;
24 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelProcessingPhase;
25 import org.opendaylight.yangtools.yang.parser.spi.meta.NamespaceStmtCtx;
26 import org.opendaylight.yangtools.yang.parser.spi.meta.OverrideChildStatementSupport;
27 import org.opendaylight.yangtools.yang.parser.spi.meta.StatementFactory;
28 import org.opendaylight.yangtools.yang.parser.spi.meta.StatementSupport;
29 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext;
30 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext.Mutable;
31
32 final class StatementDefinitionContext<A, D extends DeclaredStatement<A>, E extends EffectiveStatement<A, D>> {
33     private final @NonNull StatementSupport<A, D, E> support;
34     private final Map<String, StatementDefinitionContext<?, ?, ?>> argumentSpecificSubDefinitions;
35
36     private Map<StatementDefinitionContext<?, ?, ?>, StatementDefinitionContext<?,?,?>> unknownStmtDefsOfYangStmts;
37
38     StatementDefinitionContext(final StatementSupport<A, D, E> support) {
39         this.support = requireNonNull(support);
40         this.argumentSpecificSubDefinitions = support.hasArgumentSpecificSupports() ? new HashMap<>() : null;
41     }
42
43     @NonNull StatementFactory<A, D, E> getFactory() {
44         return support;
45     }
46
47     A parseArgumentValue(final StmtContext<A, D, E> context, final String value) {
48         return support.parseArgumentValue(context, value);
49     }
50
51     A adaptArgumentValue(final StmtContext<A, D, E> context, final QNameModule targetModule) {
52         return support.adaptArgumentValue(context, targetModule);
53     }
54
55     @NonNull StatementDefinition getPublicView() {
56         return support.getPublicView();
57     }
58
59     Optional<StatementSupport<?, ?, ?>> getImplicitParentFor(final NamespaceStmtCtx parent,
60             final StatementDefinition stmtDef) {
61         return support instanceof ImplicitParentAwareStatementSupport implicit
62                 ? implicit.getImplicitParentFor(parent, stmtDef) : Optional.empty();
63     }
64
65     void onStatementAdded(final Mutable<A, D, E> stmt) {
66         support.onStatementAdded(stmt);
67     }
68
69     void onDeclarationFinished(final Mutable<A, D, E> statement, final ModelProcessingPhase phase) {
70         switch (phase) {
71             case SOURCE_PRE_LINKAGE:
72                 support.onPreLinkageDeclared(statement);
73                 break;
74             case SOURCE_LINKAGE:
75                 support.onLinkageDeclared(statement);
76                 break;
77             case STATEMENT_DEFINITION:
78                 support.onStatementDefinitionDeclared(statement);
79                 break;
80             case FULL_DECLARATION:
81                 support.onFullDefinitionDeclared(statement);
82                 break;
83             default:
84                 break;
85         }
86     }
87
88     @NonNull Optional<ArgumentDefinition> getArgumentDefinition() {
89         return support.getArgumentDefinition();
90     }
91
92     @NonNull QName getStatementName() {
93         return support.statementName();
94     }
95
96     @Override
97     public String toString() {
98         return MoreObjects.toStringHelper(this).add("statement", getStatementName()).toString();
99     }
100
101     @NonNull StatementDefinitionContext<?, ?, ?> getSubDefinitionSpecificForArgument(final String argument) {
102         if (!hasArgumentSpecificSubDefinitions()) {
103             return this;
104         }
105
106         StatementDefinitionContext<?, ?, ?> potential = argumentSpecificSubDefinitions.get(argument);
107         if (potential == null) {
108             final StatementSupport<?, ?, ?> argumentSpecificSupport = support.getSupportSpecificForArgument(argument);
109             potential = argumentSpecificSupport != null ? new StatementDefinitionContext<>(argumentSpecificSupport)
110                     : this;
111             argumentSpecificSubDefinitions.put(argument, potential);
112         }
113
114         return potential;
115     }
116
117     StatementSupport<A, D, E> support() {
118         return support;
119     }
120
121     boolean hasArgumentSpecificSubDefinitions() {
122         return support.hasArgumentSpecificSupports();
123     }
124
125     @NonNull StatementDefinitionContext<?, ?, ?> overrideDefinition(
126             final @NonNull StatementDefinitionContext<?, ?, ?> def) {
127         if (!(support instanceof OverrideChildStatementSupport)) {
128             return def;
129         }
130
131         if (unknownStmtDefsOfYangStmts != null) {
132             final StatementDefinitionContext<?, ?, ?> existing = unknownStmtDefsOfYangStmts.get(def);
133             if (existing != null) {
134                 return existing;
135             }
136         } else {
137             unknownStmtDefsOfYangStmts = new HashMap<>(4);
138         }
139
140         final StatementSupport<?, ?, ?> override =
141             ((OverrideChildStatementSupport) support).statementDefinitionOverrideOf(def.getPublicView());
142         final StatementDefinitionContext<?, ?, ?> ret;
143         if (override != null) {
144             ret = new StatementDefinitionContext<>(override);
145         } else {
146             ret = def;
147         }
148         unknownStmtDefsOfYangStmts.put(def, ret);
149         return ret;
150     }
151 }