51fa88be222d8e82f351572ac47603e0f98ebd2a
[yangtools.git] / yang / yang-parser-rfc7950 / src / main / java / org / opendaylight / yangtools / yang / parser / rfc7950 / stmt / meta / KeyStatementSupport.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.meta;
9
10 import com.google.common.base.CharMatcher;
11 import com.google.common.base.Splitter;
12 import com.google.common.collect.ImmutableList;
13 import com.google.common.collect.ImmutableSet;
14 import com.google.common.collect.ImmutableSet.Builder;
15 import java.util.Set;
16 import org.opendaylight.yangtools.yang.common.QName;
17 import org.opendaylight.yangtools.yang.common.QNameModule;
18 import org.opendaylight.yangtools.yang.model.api.YangStmtMapping;
19 import org.opendaylight.yangtools.yang.model.api.meta.DeclarationReference;
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.stmt.KeyEffectiveStatement;
23 import org.opendaylight.yangtools.yang.model.api.stmt.KeyStatement;
24 import org.opendaylight.yangtools.yang.model.ri.stmt.DeclaredStatementDecorators;
25 import org.opendaylight.yangtools.yang.model.ri.stmt.DeclaredStatements;
26 import org.opendaylight.yangtools.yang.model.ri.stmt.EffectiveStatements;
27 import org.opendaylight.yangtools.yang.parser.antlr.YangStatementLexer;
28 import org.opendaylight.yangtools.yang.parser.api.YangParserConfiguration;
29 import org.opendaylight.yangtools.yang.parser.spi.meta.AbstractStatementSupport;
30 import org.opendaylight.yangtools.yang.parser.spi.meta.EffectiveStmtCtx.Current;
31 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext;
32 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContextUtils;
33 import org.opendaylight.yangtools.yang.parser.spi.meta.SubstatementValidator;
34 import org.opendaylight.yangtools.yang.parser.spi.source.SourceException;
35
36 public final class KeyStatementSupport
37         extends AbstractStatementSupport<Set<QName>, KeyStatement, KeyEffectiveStatement> {
38     /**
39      * This is equivalent to {@link YangStatementLexer#SEP}'s definition. Currently equivalent to the non-repeating
40      * part of:
41      *
42      * <p>
43      * {@code SEP: [ \n\r\t]+ -> type(SEP);}.
44      */
45     private static final CharMatcher SEP = CharMatcher.anyOf(" \n\r\t").precomputed();
46
47     /**
48      * Splitter corresponding to {@code key-arg} ABNF as defined
49      * in <a href="https://tools.ietf.org/html/rfc6020#section-12">RFC6020, section 12</a>:
50      *
51      * <p>
52      * {@code key-arg             = node-identifier *(sep node-identifier)}
53      *
54      * <p>
55      * We also account for {@link #SEP} not handling repetition by ignoring empty strings.
56      */
57     private static final Splitter KEY_ARG_SPLITTER = Splitter.on(SEP).omitEmptyStrings();
58
59     private static final SubstatementValidator SUBSTATEMENT_VALIDATOR =
60         SubstatementValidator.builder(YangStmtMapping.KEY).build();
61
62     public KeyStatementSupport(final YangParserConfiguration config) {
63         super(YangStmtMapping.KEY, StatementPolicy.copyDeclared(
64             // Identity comparison is sufficient because adaptArgumentValue() is careful about reuse.
65             (copy, current, substatements) -> copy.getArgument() == current.getArgument()),
66             config, SUBSTATEMENT_VALIDATOR);
67     }
68
69     @Override
70     public ImmutableSet<QName> parseArgumentValue(final StmtContext<?, ?, ?> ctx, final String value) {
71         final Builder<QName> builder = ImmutableSet.builder();
72         int tokens = 0;
73         for (String keyToken : KEY_ARG_SPLITTER.split(value)) {
74             builder.add(StmtContextUtils.parseNodeIdentifier(ctx, keyToken));
75             tokens++;
76         }
77
78         // Throws NPE on nulls, retains first inserted value, cannot be modified
79         final ImmutableSet<QName> ret = builder.build();
80         SourceException.throwIf(ret.size() != tokens, ctx, "Key argument '%s' contains duplicates", value);
81         return ret;
82     }
83
84     @Override
85     public Set<QName> adaptArgumentValue(final StmtContext<Set<QName>, KeyStatement, KeyEffectiveStatement> ctx,
86             final QNameModule targetModule) {
87         final Builder<QName> builder = ImmutableSet.builder();
88         boolean replaced = false;
89         for (final QName qname : ctx.getArgument()) {
90             if (!targetModule.equals(qname.getModule())) {
91                 final QName newQname = qname.bindTo(targetModule).intern();
92                 builder.add(newQname);
93                 replaced = true;
94             } else {
95                 builder.add(qname);
96             }
97         }
98
99         // This makes sure we reuse the collection when a grouping is instantiated in the same module.
100         return replaced ? builder.build() : ctx.argument();
101     }
102
103     @Override
104     protected KeyStatement createDeclared(final StmtContext<Set<QName>, KeyStatement, ?> ctx,
105             final ImmutableList<? extends DeclaredStatement<?>> substatements) {
106         return DeclaredStatements.createKey(ctx.getRawArgument(), ctx.getArgument(), substatements);
107     }
108
109     @Override
110     protected KeyStatement attachDeclarationReference(final KeyStatement stmt, final DeclarationReference reference) {
111         return DeclaredStatementDecorators.decorateKey(stmt, reference);
112     }
113
114     @Override
115     protected KeyEffectiveStatement createEffective(final Current<Set<QName>, KeyStatement> stmt,
116             final ImmutableList<? extends EffectiveStatement<?, ?>> substatements) {
117         return EffectiveStatements.createKey(stmt.declared(), stmt.getArgument(), substatements);
118     }
119 }