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