Migrate rawStatementArgument()/getStatementSourceReference() callers
[yangtools.git] / yang / yang-parser-rfc7950 / src / main / java / org / opendaylight / yangtools / yang / parser / rfc7950 / stmt / range / RangeStatementSupport.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.range;
9
10 import com.google.common.collect.ImmutableList;
11 import com.google.common.collect.Iterables;
12 import java.math.BigDecimal;
13 import java.math.BigInteger;
14 import java.util.ArrayList;
15 import java.util.Iterator;
16 import java.util.List;
17 import org.opendaylight.yangtools.yang.model.api.YangStmtMapping;
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.RangeEffectiveStatement;
21 import org.opendaylight.yangtools.yang.model.api.stmt.RangeStatement;
22 import org.opendaylight.yangtools.yang.model.api.stmt.UnresolvedNumber;
23 import org.opendaylight.yangtools.yang.model.api.stmt.ValueRange;
24 import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.ArgumentUtils;
25 import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.BaseStatementSupport;
26 import org.opendaylight.yangtools.yang.parser.spi.meta.EffectiveStmtCtx.Current;
27 import org.opendaylight.yangtools.yang.parser.spi.meta.InferenceException;
28 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext;
29 import org.opendaylight.yangtools.yang.parser.spi.meta.SubstatementValidator;
30 import org.opendaylight.yangtools.yang.parser.spi.source.SourceException;
31
32 public final class RangeStatementSupport
33         extends BaseStatementSupport<List<ValueRange>, RangeStatement, RangeEffectiveStatement> {
34     private static final SubstatementValidator SUBSTATEMENT_VALIDATOR = SubstatementValidator.builder(YangStmtMapping
35         .RANGE)
36         .addOptional(YangStmtMapping.DESCRIPTION)
37         .addOptional(YangStmtMapping.ERROR_APP_TAG)
38         .addOptional(YangStmtMapping.ERROR_MESSAGE)
39         .addOptional(YangStmtMapping.REFERENCE)
40         .build();
41     private static final RangeStatementSupport INSTANCE = new RangeStatementSupport();
42
43     private RangeStatementSupport() {
44         super(YangStmtMapping.RANGE);
45     }
46
47     public static RangeStatementSupport getInstance() {
48         return INSTANCE;
49     }
50
51     @Override
52     public ImmutableList<ValueRange> parseArgumentValue(final StmtContext<?, ?, ?> ctx, final String rangeArgument) {
53         final List<ValueRange> ranges = new ArrayList<>();
54
55         for (final String singleRange : ArgumentUtils.PIPE_SPLITTER.split(rangeArgument)) {
56             final Iterator<String> boundaries = ArgumentUtils.TWO_DOTS_SPLITTER.split(singleRange).iterator();
57             final Number min = parseDecimalConstraintValue(ctx, boundaries.next());
58
59             final Number max;
60             if (boundaries.hasNext()) {
61                 max = parseDecimalConstraintValue(ctx, boundaries.next());
62
63                 // if min larger than max then error
64                 SourceException.throwIf(ArgumentUtils.compareNumbers(min, max) == 1, ctx.sourceReference(),
65                     "Range constraint %s has descending order of boundaries; should be ascending", singleRange);
66                 SourceException.throwIf(boundaries.hasNext(), ctx.sourceReference(),
67                     "Wrong number of boundaries in range constraint %s", singleRange);
68             } else {
69                 max = min;
70             }
71
72             // some of intervals overlapping
73             InferenceException.throwIf(ranges.size() > 1
74                 && ArgumentUtils.compareNumbers(min, Iterables.getLast(ranges).upperBound()) != 1,
75                 ctx.sourceReference(),  "Some of the value ranges in %s are not disjoint", rangeArgument);
76             ranges.add(ValueRange.of(min, max));
77         }
78
79         return ImmutableList.copyOf(ranges);
80     }
81
82     @Override
83     protected RangeStatement createDeclared(final StmtContext<List<ValueRange>, RangeStatement, ?> ctx,
84             final ImmutableList<? extends DeclaredStatement<?>> substatements) {
85         return new RegularRangeStatement(ctx.coerceRawStatementArgument(), ctx.coerceStatementArgument(),
86             substatements);
87     }
88
89     @Override
90     protected RangeStatement createEmptyDeclared(final StmtContext<List<ValueRange>, RangeStatement, ?> ctx) {
91         return new EmptyRangeStatement(ctx.coerceRawStatementArgument(), ctx.coerceStatementArgument());
92     }
93
94     @Override
95     protected RangeEffectiveStatement createEffective(final Current<List<ValueRange>, RangeStatement> stmt,
96             final ImmutableList<? extends EffectiveStatement<?, ?>> substatements) {
97         return substatements.isEmpty() ? new EmptyRangeEffectiveStatement(stmt.declared())
98             : new RegularRangeEffectiveStatement(stmt.declared(), substatements);
99     }
100
101     @Override
102     protected SubstatementValidator getSubstatementValidator() {
103         return SUBSTATEMENT_VALIDATOR;
104     }
105
106     private static Number parseDecimalConstraintValue(final StmtContext<?, ?, ?> ctx, final String value) {
107         if ("max".equals(value)) {
108             return UnresolvedNumber.max();
109         }
110         if ("min".equals(value)) {
111             return UnresolvedNumber.min();
112         }
113
114         try {
115             return value.indexOf('.') != -1 ? new BigDecimal(value) : new BigInteger(value);
116         } catch (final NumberFormatException e) {
117             throw new SourceException(ctx.sourceReference(), e, "Value %s is not a valid decimal number", value);
118         }
119     }
120 }