e06411c2e9b41719512ae4c1eaee13c47ac07300
[yangtools.git] /
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.ImmutableList;
11 import com.google.common.collect.Iterables;
12 import java.util.ArrayList;
13 import java.util.Iterator;
14 import java.util.List;
15 import org.eclipse.jdt.annotation.NonNull;
16 import org.opendaylight.yangtools.yang.common.Uint64;
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.LengthEffectiveStatement;
21 import org.opendaylight.yangtools.yang.model.api.stmt.LengthStatement;
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.model.spi.stmt.DeclaredStatements;
25 import org.opendaylight.yangtools.yang.parser.rfc7950.stmt.ArgumentUtils;
26 import org.opendaylight.yangtools.yang.parser.spi.meta.AbstractStatementSupport;
27 import org.opendaylight.yangtools.yang.parser.spi.meta.EffectiveStmtCtx.Current;
28 import org.opendaylight.yangtools.yang.parser.spi.meta.InferenceException;
29 import org.opendaylight.yangtools.yang.parser.spi.meta.StmtContext;
30 import org.opendaylight.yangtools.yang.parser.spi.meta.SubstatementValidator;
31 import org.opendaylight.yangtools.yang.parser.spi.source.SourceException;
32
33 public final class LengthStatementSupport
34         extends AbstractStatementSupport<List<ValueRange>, LengthStatement, LengthEffectiveStatement> {
35     private static final SubstatementValidator SUBSTATEMENT_VALIDATOR = SubstatementValidator.builder(YangStmtMapping
36         .LENGTH)
37         .addOptional(YangStmtMapping.DESCRIPTION)
38         .addOptional(YangStmtMapping.ERROR_APP_TAG)
39         .addOptional(YangStmtMapping.ERROR_MESSAGE)
40         .addOptional(YangStmtMapping.REFERENCE)
41         .build();
42     private static final LengthStatementSupport INSTANCE = new LengthStatementSupport();
43
44     private LengthStatementSupport() {
45         super(YangStmtMapping.LENGTH, StatementPolicy.contextIndependent());
46     }
47
48     public static LengthStatementSupport getInstance() {
49         return INSTANCE;
50     }
51
52     @Override
53     public ImmutableList<ValueRange> parseArgumentValue(final StmtContext<?, ?, ?> ctx, final String value) {
54         final List<ValueRange> ranges = new ArrayList<>();
55
56         for (final String singleRange : ArgumentUtils.PIPE_SPLITTER.split(value)) {
57             final Iterator<String> boundaries = ArgumentUtils.TWO_DOTS_SPLITTER.split(singleRange).iterator();
58             final Number min = parseIntegerConstraintValue(ctx, boundaries.next());
59
60             final Number max;
61             if (boundaries.hasNext()) {
62                 max = parseIntegerConstraintValue(ctx, boundaries.next());
63
64                 // if min larger than max then error
65                 SourceException.throwIf(ArgumentUtils.compareNumbers(min, max) == 1, ctx,
66                     "Length constraint %s has descending order of boundaries; should be ascending.", singleRange);
67                 SourceException.throwIf(boundaries.hasNext(), ctx,
68                     "Wrong number of boundaries in length constraint %s.", singleRange);
69             } else {
70                 max = min;
71             }
72
73             // some of intervals overlapping
74             InferenceException.throwIf(
75                 ranges.size() > 1 && ArgumentUtils.compareNumbers(min, Iterables.getLast(ranges).upperBound()) != 1,
76                 ctx, "Some of the length ranges in %s are not disjoint", value);
77             ranges.add(ValueRange.of(min, max));
78         }
79
80         return ImmutableList.copyOf(ranges);
81     }
82
83     @Override
84     protected SubstatementValidator getSubstatementValidator() {
85         return SUBSTATEMENT_VALIDATOR;
86     }
87
88     @Override
89     protected LengthStatement createDeclared(final StmtContext<List<ValueRange>, LengthStatement, ?> ctx,
90             final ImmutableList<? extends DeclaredStatement<?>> substatements) {
91         return DeclaredStatements.createLength(ctx.getRawArgument(), ctx.getArgument(), substatements);
92     }
93
94     @Override
95     protected LengthStatement createEmptyDeclared(final StmtContext<List<ValueRange>, LengthStatement, ?> ctx) {
96         return DeclaredStatements.createLength(ctx.getRawArgument(), ctx.getArgument());
97     }
98
99     @Override
100     protected LengthEffectiveStatement createEffective(final Current<List<ValueRange>, LengthStatement> stmt,
101             final ImmutableList<? extends EffectiveStatement<?, ?>> substatements) {
102         return substatements.isEmpty() ? new EmptyLengthEffectiveStatement(stmt.declared())
103             : new RegularLengthEffectiveStatement(stmt.declared(), substatements);
104     }
105
106     private static @NonNull Number parseIntegerConstraintValue(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         // As per RFC6020/RFC7950 section 9.4.4:
115         //
116         //   An implementation is not required to support a length value larger than 18446744073709551615.
117         //
118         // We could support bigger precision at the cost of additional memory and/or potential ValueRange upper/lower
119         // bound inconsistency. We also take advantage of Uint64's interning facilities.
120         try {
121             return Uint64.valueOf(value).intern();
122         } catch (NumberFormatException e) {
123             throw new SourceException(ctx, e, "Value %s is not a valid unsigned integer", value);
124         } catch (IllegalArgumentException e) {
125             throw new SourceException(ctx, e, "Value %s exceeds maximum supported value %s", value, Uint64.MAX_VALUE);
126         }
127     }
128 }