/* * 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.case_; import static java.util.Objects.requireNonNull; import com.google.common.annotations.Beta; import com.google.common.collect.ImmutableList; import edu.umd.cs.findbugs.annotations.SuppressFBWarnings; 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.DeclaredStatement; import org.opendaylight.yangtools.yang.model.api.meta.EffectiveStatement; import org.opendaylight.yangtools.yang.model.api.meta.StatementSource; 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.parser.rfc7950.stmt.BaseImplicitStatementSupport; import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.EffectiveStatementMixins.EffectiveStatementWithFlags.FlagsBuilder; import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.SubstatementIndexingException; 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 BaseImplicitStatementSupport { private static final @NonNull CaseStatementSupport RFC6020_INSTANCE = new CaseStatementSupport( 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 @NonNull CaseStatementSupport RFC7950_INSTANCE = new CaseStatementSupport( 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 final SubstatementValidator validator; private CaseStatementSupport(final SubstatementValidator validator) { super(YangStmtMapping.CASE, StatementPolicy.legacyDeclaredCopy()); this.validator = requireNonNull(validator); } public static @NonNull CaseStatementSupport rfc6020Instance() { return RFC6020_INSTANCE; } public static @NonNull CaseStatementSupport rfc7950Instance() { return RFC7950_INSTANCE; } @Override protected SubstatementValidator getSubstatementValidator() { return validator; } @Override protected CaseStatement createDeclared(final StmtContext ctx, final ImmutableList> substatements) { final StatementSource source = ctx.source(); switch (source) { case CONTEXT: return new RegularUndeclaredCaseStatement(ctx.getArgument(), substatements); case DECLARATION: return new RegularCaseStatement(ctx.getArgument(), substatements); default: throw new IllegalStateException("Unhandled statement source " + source); } } @Override protected CaseStatement createEmptyDeclared(final StmtContext ctx) { final StatementSource source = ctx.source(); switch (source) { case CONTEXT: return new EmptyUndeclaredCaseStatement(ctx.getArgument()); case DECLARATION: return new EmptyCaseStatement(ctx.getArgument()); default: throw new IllegalStateException("Unhandled statement source " + source); } } @Override protected CaseEffectiveStatement createDeclaredEffective(final Current stmt, final ImmutableList> substatements) { try { return new DeclaredCaseEffectiveStatement(stmt.declared(), substatements, computeFlags(stmt, substatements), stmt.wrapSchemaPath(), findOriginal(stmt)); } catch (SubstatementIndexingException e) { throw new SourceException(e.getMessage(), stmt, e); } } @Override protected CaseEffectiveStatement createUndeclaredEffective(final Current stmt, final ImmutableList> substatements) { try { return new UndeclaredCaseEffectiveStatement(substatements, computeFlags(stmt, substatements), stmt.wrapSchemaPath(), findOriginal(stmt)); } catch (SubstatementIndexingException e) { throw new SourceException(e.getMessage(), stmt, e); } } private static @Nullable CaseSchemaNode findOriginal(final Current stmt) { return (CaseSchemaNode) stmt.original(); } private static int computeFlags(final Current stmt, final ImmutableList> 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 ImmutableList> substatements) { for (EffectiveStatement stmt : substatements) { if (stmt instanceof DataSchemaNode) { final Optional opt = ((DataSchemaNode) stmt).effectiveConfig(); if (opt.isPresent()) { return opt.orElseThrow(); } } } return null; } }