b2db853e0bed9b3ba4a80338deb4c2ca48f585ed
[yangtools.git] / yang / yang-parser-rfc7950 / src / main / java / org / opendaylight / yangtools / yang / parser / rfc7950 / stmt / typedef / TypedefStatementSupport.java
1 /*
2  * Copyright (c) 2017 Pantheon Technologies, s.r.o. 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.typedef;
9
10 import static com.google.common.base.Preconditions.checkState;
11
12 import com.google.common.collect.ImmutableList;
13 import java.util.Collection;
14 import org.opendaylight.yangtools.yang.common.QName;
15 import org.opendaylight.yangtools.yang.model.api.Status;
16 import org.opendaylight.yangtools.yang.model.api.YangStmtMapping;
17 import org.opendaylight.yangtools.yang.model.api.meta.DeclaredStatement;
18 import org.opendaylight.yangtools.yang.model.api.meta.EffectiveStatement;
19 import org.opendaylight.yangtools.yang.model.api.stmt.DefaultEffectiveStatement;
20 import org.opendaylight.yangtools.yang.model.api.stmt.StatusEffectiveStatement;
21 import org.opendaylight.yangtools.yang.model.api.stmt.TypeEffectiveStatement;
22 import org.opendaylight.yangtools.yang.model.api.stmt.TypedefEffectiveStatement;
23 import org.opendaylight.yangtools.yang.model.api.stmt.TypedefStatement;
24 import org.opendaylight.yangtools.yang.model.ri.stmt.DeclaredStatements;
25 import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.EffectiveStatementMixins.EffectiveStatementWithFlags.FlagsBuilder;
26 import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.EffectiveStmtUtils;
27 import org.opendaylight.yangtools.yang.parser.spi.TypeNamespace;
28 import org.opendaylight.yangtools.yang.parser.spi.meta.AbstractQNameStatementSupport;
29 import org.opendaylight.yangtools.yang.parser.spi.meta.EffectiveStmtCtx.Current;
30 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelActionBuilder;
31 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelActionBuilder.InferenceAction;
32 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelActionBuilder.InferenceContext;
33 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelActionBuilder.Prerequisite;
34 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelProcessingPhase;
35 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext;
36 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext.Mutable;
37 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContextUtils;
38 import org.opendaylight.yangtools.yang.parser.spi.meta.SubstatementValidator;
39 import org.opendaylight.yangtools.yang.parser.spi.source.SourceException;
40
41 public final class TypedefStatementSupport extends
42         AbstractQNameStatementSupport<TypedefStatement, TypedefEffectiveStatement> {
43     private static final SubstatementValidator SUBSTATEMENT_VALIDATOR = SubstatementValidator.builder(
44         YangStmtMapping.TYPEDEF)
45         .addOptional(YangStmtMapping.DEFAULT)
46         .addOptional(YangStmtMapping.DESCRIPTION)
47         .addOptional(YangStmtMapping.REFERENCE)
48         .addOptional(YangStmtMapping.STATUS)
49         .addMandatory(YangStmtMapping.TYPE)
50         .addOptional(YangStmtMapping.UNITS)
51         .build();
52     private static final TypedefStatementSupport INSTANCE = new TypedefStatementSupport();
53
54     private TypedefStatementSupport() {
55         super(YangStmtMapping.TYPEDEF, StatementPolicy.exactReplica());
56     }
57
58     public static TypedefStatementSupport getInstance() {
59         return INSTANCE;
60     }
61
62     @Override
63     public QName parseArgumentValue(final StmtContext<?, ?, ?> ctx, final String value) {
64         return StmtContextUtils.parseIdentifier(ctx, value);
65     }
66
67     @Override
68     public void onFullDefinitionDeclared(final Mutable<QName, TypedefStatement, TypedefEffectiveStatement> stmt) {
69         super.onFullDefinitionDeclared(stmt);
70
71         final Mutable<?, ?, ?> parent = stmt.getParentContext();
72         if (parent != null) {
73             // Shadowing check: make sure we do not trample on pre-existing definitions. This catches sibling
74             // declarations and parent declarations which have already been declared.
75             checkConflict(parent, stmt);
76             parent.addContext(TypeNamespace.class, stmt.getArgument(), stmt);
77
78             final StmtContext<?, ?, ?> grandParent = parent.getParentContext();
79             if (grandParent != null) {
80                 // Shadowing check: make sure grandparent does not see a conflicting definition. This is required to
81                 // ensure that a typedef in child scope does not shadow a typedef in parent scope which occurs later in
82                 // the text. For that check we need the full declaration of our model.
83
84                 final ModelActionBuilder action = stmt.newInferenceAction(ModelProcessingPhase.FULL_DECLARATION);
85                 action.requiresCtx(grandParent.getRoot(), ModelProcessingPhase.FULL_DECLARATION);
86                 action.apply(new InferenceAction() {
87                     @Override
88                     public void apply(final InferenceContext ctx) {
89                         checkConflict(grandParent, stmt);
90                     }
91
92                     @Override
93                     public void prerequisiteFailed(final Collection<? extends Prerequisite<?>> failed) {
94                         // No-op
95                     }
96                 });
97             }
98         }
99     }
100
101     @Override
102     protected SubstatementValidator getSubstatementValidator() {
103         return SUBSTATEMENT_VALIDATOR;
104     }
105
106     @Override
107     protected TypedefStatement createDeclared(final StmtContext<QName, TypedefStatement, ?> ctx,
108             final ImmutableList<? extends DeclaredStatement<?>> substatements) {
109         return DeclaredStatements.createTypedef(ctx.getArgument(), substatements);
110     }
111
112     @Override
113     protected TypedefStatement createEmptyDeclared(final StmtContext<QName, TypedefStatement, ?> ctx) {
114         return DeclaredStatements.createTypedef(ctx.getArgument());
115     }
116
117     @Override
118     protected TypedefEffectiveStatement createEffective(final Current<QName, TypedefStatement> stmt,
119             final ImmutableList<? extends EffectiveStatement<?, ?>> substatements) {
120         final TypedefStatement declared = stmt.declared();
121         checkState(!substatements.isEmpty(), "Refusing to create empty typedef for %s", stmt.declared());
122
123         final TypeEffectiveStatement<?> typeEffectiveStmt = findFirstStatement(substatements,
124             TypeEffectiveStatement.class);
125         final String dflt = findFirstArgument(substatements, DefaultEffectiveStatement.class, null);
126         SourceException.throwIf(
127             EffectiveStmtUtils.hasDefaultValueMarkedWithIfFeature(stmt.yangVersion(), typeEffectiveStmt, dflt), stmt,
128             "Typedef '%s' has default value '%s' marked with an if-feature statement.", stmt.argument(), dflt);
129
130         return new TypedefEffectiveStatementImpl(declared, stmt.effectivePath(), computeFlags(substatements),
131             substatements);
132     }
133
134     private static void checkConflict(final StmtContext<?, ?, ?> parent, final StmtContext<QName, ?, ?> stmt) {
135         final QName arg = stmt.getArgument();
136         final StmtContext<?, ?, ?> existing = parent.getFromNamespace(TypeNamespace.class, arg);
137         // RFC7950 sections 5.5 and 6.2.1: identifiers must not be shadowed
138         SourceException.throwIf(existing != null, stmt, "Duplicate name for typedef %s", arg);
139     }
140
141     private static int computeFlags(final ImmutableList<? extends EffectiveStatement<?, ?>> substatements) {
142         return new FlagsBuilder()
143                 .setStatus(findFirstArgument(substatements, StatusEffectiveStatement.class, Status.CURRENT))
144                 .toFlags();
145     }
146 }