2 * Copyright (c) 2017 Pantheon Technologies, s.r.o. 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.parser.rfc7950.stmt.meta;
10 import static com.google.common.base.Preconditions.checkState;
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.DeclarationReference;
18 import org.opendaylight.yangtools.yang.model.api.meta.DeclaredStatement;
19 import org.opendaylight.yangtools.yang.model.api.meta.EffectiveStatement;
20 import org.opendaylight.yangtools.yang.model.api.stmt.DefaultEffectiveStatement;
21 import org.opendaylight.yangtools.yang.model.api.stmt.StatusEffectiveStatement;
22 import org.opendaylight.yangtools.yang.model.api.stmt.TypeEffectiveStatement;
23 import org.opendaylight.yangtools.yang.model.api.stmt.TypedefEffectiveStatement;
24 import org.opendaylight.yangtools.yang.model.api.stmt.TypedefStatement;
25 import org.opendaylight.yangtools.yang.model.ri.stmt.DeclaredStatementDecorators;
26 import org.opendaylight.yangtools.yang.model.ri.stmt.DeclaredStatements;
27 import org.opendaylight.yangtools.yang.model.ri.stmt.EffectiveStatements;
28 import org.opendaylight.yangtools.yang.model.spi.meta.EffectiveStatementMixins.EffectiveStatementWithFlags.FlagsBuilder;
29 import org.opendaylight.yangtools.yang.parser.api.YangParserConfiguration;
30 import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.EffectiveStmtUtils;
31 import org.opendaylight.yangtools.yang.parser.spi.TypeNamespace;
32 import org.opendaylight.yangtools.yang.parser.spi.meta.AbstractQNameStatementSupport;
33 import org.opendaylight.yangtools.yang.parser.spi.meta.BoundStmtCtx;
34 import org.opendaylight.yangtools.yang.parser.spi.meta.EffectiveStmtCtx.Current;
35 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelActionBuilder;
36 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelActionBuilder.InferenceAction;
37 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelActionBuilder.InferenceContext;
38 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelActionBuilder.Prerequisite;
39 import org.opendaylight.yangtools.yang.parser.spi.meta.ModelProcessingPhase;
40 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext;
41 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext.Mutable;
42 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContextUtils;
43 import org.opendaylight.yangtools.yang.parser.spi.meta.SubstatementValidator;
44 import org.opendaylight.yangtools.yang.parser.spi.source.SourceException;
46 public final class TypedefStatementSupport extends
47 AbstractQNameStatementSupport<TypedefStatement, TypedefEffectiveStatement> {
48 private static final SubstatementValidator SUBSTATEMENT_VALIDATOR = SubstatementValidator.builder(
49 YangStmtMapping.TYPEDEF)
50 .addOptional(YangStmtMapping.DEFAULT)
51 .addOptional(YangStmtMapping.DESCRIPTION)
52 .addOptional(YangStmtMapping.REFERENCE)
53 .addOptional(YangStmtMapping.STATUS)
54 .addMandatory(YangStmtMapping.TYPE)
55 .addOptional(YangStmtMapping.UNITS)
58 public TypedefStatementSupport(final YangParserConfiguration config) {
59 super(YangStmtMapping.TYPEDEF, StatementPolicy.exactReplica(), config, SUBSTATEMENT_VALIDATOR);
63 public QName parseArgumentValue(final StmtContext<?, ?, ?> ctx, final String value) {
64 return StmtContextUtils.parseIdentifier(ctx, value);
68 public void onFullDefinitionDeclared(final Mutable<QName, TypedefStatement, TypedefEffectiveStatement> stmt) {
69 super.onFullDefinitionDeclared(stmt);
71 final Mutable<?, ?, ?> parent = stmt.getParentContext();
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);
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.
84 final ModelActionBuilder action = stmt.newInferenceAction(ModelProcessingPhase.FULL_DECLARATION);
85 action.requiresCtx(grandParent.getRoot(), ModelProcessingPhase.FULL_DECLARATION);
86 action.apply(new InferenceAction() {
88 public void apply(final InferenceContext ctx) {
89 checkConflict(grandParent, stmt);
93 public void prerequisiteFailed(final Collection<? extends Prerequisite<?>> failed) {
102 protected TypedefStatement createDeclared(final BoundStmtCtx<QName> ctx,
103 final ImmutableList<DeclaredStatement<?>> substatements) {
104 return DeclaredStatements.createTypedef(ctx.getArgument(), substatements);
108 protected TypedefStatement attachDeclarationReference(final TypedefStatement stmt,
109 final DeclarationReference reference) {
110 return DeclaredStatementDecorators.decorateTypedef(stmt, reference);
114 protected TypedefEffectiveStatement createEffective(final Current<QName, TypedefStatement> stmt,
115 final ImmutableList<? extends EffectiveStatement<?, ?>> substatements) {
116 final TypedefStatement declared = stmt.declared();
117 checkState(!substatements.isEmpty(), "Refusing to create empty typedef for %s", stmt.declared());
119 final TypeEffectiveStatement<?> typeEffectiveStmt = findFirstStatement(substatements,
120 TypeEffectiveStatement.class);
121 final String dflt = findFirstArgument(substatements, DefaultEffectiveStatement.class, null);
122 SourceException.throwIf(
123 EffectiveStmtUtils.hasDefaultValueMarkedWithIfFeature(stmt.yangVersion(), typeEffectiveStmt, dflt), stmt,
124 "Typedef '%s' has default value '%s' marked with an if-feature statement.", stmt.argument(), dflt);
126 return EffectiveStatements.createTypedef(declared, computeFlags(substatements), substatements);
129 private static void checkConflict(final StmtContext<?, ?, ?> parent, final StmtContext<QName, ?, ?> stmt) {
130 final QName arg = stmt.getArgument();
131 final StmtContext<?, ?, ?> existing = parent.getFromNamespace(TypeNamespace.class, arg);
132 // RFC7950 sections 5.5 and 6.2.1: identifiers must not be shadowed
133 SourceException.throwIf(existing != null, stmt, "Duplicate name for typedef %s", arg);
136 private static int computeFlags(final ImmutableList<? extends EffectiveStatement<?, ?>> substatements) {
137 return new FlagsBuilder()
138 .setStatus(findFirstArgument(substatements, StatusEffectiveStatement.class, Status.CURRENT))