/* * Copyright (c) 2017 Pantheon Technologies, s.r.o. and others. All rights reserved. * * This program and the accompanying materials are made available under the * terms of the Eclipse Public License v1.0 which accompanies this distribution, * and is available at http://www.eclipse.org/legal/epl-v10.html */ package org.opendaylight.yangtools.yang.parser.rfc7950.stmt.meta; import com.google.common.annotations.Beta; import com.google.common.collect.ImmutableList; import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; import java.util.Collection; import java.util.Optional; import org.eclipse.jdt.annotation.NonNull; import org.eclipse.jdt.annotation.Nullable; import org.opendaylight.yangtools.yang.common.QName; import org.opendaylight.yangtools.yang.model.api.CaseSchemaNode; import org.opendaylight.yangtools.yang.model.api.DataSchemaNode; import org.opendaylight.yangtools.yang.model.api.Status; import org.opendaylight.yangtools.yang.model.api.YangStmtMapping; import org.opendaylight.yangtools.yang.model.api.meta.DeclarationReference; import org.opendaylight.yangtools.yang.model.api.meta.DeclaredStatement; import org.opendaylight.yangtools.yang.model.api.meta.EffectiveStatement; import org.opendaylight.yangtools.yang.model.api.meta.StatementOrigin; import org.opendaylight.yangtools.yang.model.api.stmt.CaseEffectiveStatement; import org.opendaylight.yangtools.yang.model.api.stmt.CaseStatement; import org.opendaylight.yangtools.yang.model.api.stmt.StatusEffectiveStatement; import org.opendaylight.yangtools.yang.model.ri.stmt.DeclaredStatementDecorators; import org.opendaylight.yangtools.yang.model.ri.stmt.DeclaredStatements; import org.opendaylight.yangtools.yang.model.ri.stmt.EffectiveStatements; import org.opendaylight.yangtools.yang.model.ri.stmt.ImplicitStatements; import org.opendaylight.yangtools.yang.model.spi.meta.EffectiveStatementMixins.EffectiveStatementWithFlags.FlagsBuilder; import org.opendaylight.yangtools.yang.model.spi.meta.SubstatementIndexingException; import org.opendaylight.yangtools.yang.parser.api.YangParserConfiguration; import org.opendaylight.yangtools.yang.parser.spi.meta.EffectiveStmtCtx.Current; import org.opendaylight.yangtools.yang.parser.spi.meta.EffectiveStmtCtx.Parent.EffectiveConfig; import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext; import org.opendaylight.yangtools.yang.parser.spi.meta.SubstatementValidator; import org.opendaylight.yangtools.yang.parser.spi.source.SourceException; @Beta public final class CaseStatementSupport extends AbstractImplicitStatementSupport { private static final SubstatementValidator RFC6020_VALIDATOR = SubstatementValidator.builder(YangStmtMapping.CASE) .addAny(YangStmtMapping.ANYXML) .addAny(YangStmtMapping.CHOICE) .addAny(YangStmtMapping.CONTAINER) .addOptional(YangStmtMapping.DESCRIPTION) .addAny(YangStmtMapping.IF_FEATURE) .addAny(YangStmtMapping.LEAF) .addAny(YangStmtMapping.LEAF_LIST) .addAny(YangStmtMapping.LIST) .addOptional(YangStmtMapping.REFERENCE) .addOptional(YangStmtMapping.STATUS) .addAny(YangStmtMapping.USES) .addOptional(YangStmtMapping.WHEN) .build(); private static final SubstatementValidator RFC7950_VALIDATOR = SubstatementValidator.builder(YangStmtMapping.CASE) .addAny(YangStmtMapping.ANYDATA) .addAny(YangStmtMapping.ANYXML) .addAny(YangStmtMapping.CHOICE) .addAny(YangStmtMapping.CONTAINER) .addOptional(YangStmtMapping.DESCRIPTION) .addAny(YangStmtMapping.IF_FEATURE) .addAny(YangStmtMapping.LEAF) .addAny(YangStmtMapping.LEAF_LIST) .addAny(YangStmtMapping.LIST) .addOptional(YangStmtMapping.REFERENCE) .addOptional(YangStmtMapping.STATUS) .addAny(YangStmtMapping.USES) .addOptional(YangStmtMapping.WHEN) .build(); private CaseStatementSupport(final YangParserConfiguration config, final SubstatementValidator validator) { super(YangStmtMapping.CASE, instantiatedPolicy(), config, validator); } public static @NonNull CaseStatementSupport rfc6020Instance(final YangParserConfiguration config) { return new CaseStatementSupport(config, RFC6020_VALIDATOR); } public static @NonNull CaseStatementSupport rfc7950Instance(final YangParserConfiguration config) { return new CaseStatementSupport(config, RFC7950_VALIDATOR); } @Override protected CaseStatement createDeclared(final StmtContext ctx, final ImmutableList> substatements) { final StatementOrigin origin = ctx.origin(); switch (origin) { case CONTEXT: return ImplicitStatements.createCase(ctx.getArgument(), substatements); case DECLARATION: return DeclaredStatements.createCase(ctx.getArgument(), substatements); default: throw new IllegalStateException("Unhandled statement origin " + origin); } } @Override protected CaseStatement attachDeclarationReference(final CaseStatement stmt, final DeclarationReference reference) { return DeclaredStatementDecorators.decorateCase(stmt, reference); } @Override protected CaseEffectiveStatement copyDeclaredEffective(final Current stmt, final CaseEffectiveStatement original) { return EffectiveStatements.copyCase(original, stmt.effectivePath(), computeFlags(stmt, original.effectiveSubstatements()), stmt.original(CaseSchemaNode.class)); } @Override protected CaseEffectiveStatement copyUndeclaredEffective(final Current stmt, final CaseEffectiveStatement original) { return EffectiveStatements.copyCase(original, stmt.effectivePath(), computeFlags(stmt, original.effectiveSubstatements()), stmt.original(CaseSchemaNode.class)); } @Override protected CaseEffectiveStatement createDeclaredEffective(final Current stmt, final ImmutableList> substatements) { try { return EffectiveStatements.createCase(stmt.declared(), stmt.effectivePath(), computeFlags(stmt, substatements), substatements, stmt.original(CaseSchemaNode.class)); } catch (SubstatementIndexingException e) { throw new SourceException(e.getMessage(), stmt, e); } } @Override protected CaseEffectiveStatement createUndeclaredEffective(final Current stmt, final ImmutableList> substatements) { try { return EffectiveStatements.createCase(stmt.effectivePath(), computeFlags(stmt, substatements), substatements, stmt.original(CaseSchemaNode.class)); } catch (SubstatementIndexingException e) { throw new SourceException(e.getMessage(), stmt, e); } } private static int computeFlags(final Current stmt, final Collection> substatements) { final Boolean config; final EffectiveConfig effective = stmt.effectiveConfig(); switch (effective) { case FALSE: config = Boolean.FALSE; break; case IGNORED: config = null; break; case TRUE: final Boolean sub = substatementEffectiveConfig(substatements); config = sub != null ? sub : Boolean.TRUE; break; case UNDETERMINED: config = substatementEffectiveConfig(substatements); break; default: throw new IllegalStateException("Unhandled effective config " + effective); } return new FlagsBuilder() .setHistory(stmt.history()) .setStatus(findFirstArgument(substatements, StatusEffectiveStatement.class, Status.CURRENT)) .setConfiguration(config) .toFlags(); } @SuppressFBWarnings(value = "NP_BOOLEAN_RETURN_NULL", justification = "Internal use tagged with @Nullable") private static @Nullable Boolean substatementEffectiveConfig( final Collection> substatements) { for (EffectiveStatement stmt : substatements) { if (stmt instanceof DataSchemaNode) { final Optional opt = ((DataSchemaNode) stmt).effectiveConfig(); if (opt.isPresent()) { return opt.orElseThrow(); } } } return null; } }