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