blob: c78aa1ec2e94a900335658c1a8f952bcedf88275 [file] [log] [blame]
Vidyashree Ramaa2f73982016-04-12 23:33:33 +05301/*
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +05302 * Copyright 2016-present Open Networking Laboratory
Vidyashree Ramaa2f73982016-04-12 23:33:33 +05303 *
4 * Licensed under the Apache License, Version 2.0 (the "License");
5 * you may not use this file except in compliance with the License.
6 * You may obtain a copy of the License at
7 *
8 * http://www.apache.org/licenses/LICENSE-2.0
9 *
10 * Unless required by applicable law or agreed to in writing, software
11 * distributed under the License is distributed on an "AS IS" BASIS,
12 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
13 * See the License for the specific language governing permissions and
14 * limitations under the License.
15 */
16
17package org.onosproject.yangutils.parser.impl.listeners;
18
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053019import org.onosproject.yangutils.datamodel.YangDerivedInfo;
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +053020import org.onosproject.yangutils.datamodel.YangRangeRestriction;
21import org.onosproject.yangutils.datamodel.YangStringRestriction;
22import org.onosproject.yangutils.datamodel.YangType;
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053023import org.onosproject.yangutils.parser.Parsable;
24import org.onosproject.yangutils.parser.antlrgencode.GeneratedYangParser;
25import org.onosproject.yangutils.parser.exceptions.ParserException;
26import org.onosproject.yangutils.parser.impl.TreeWalkListener;
27import org.onosproject.yangutils.utils.YangConstructType;
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053028
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053029import static org.onosproject.yangutils.datamodel.YangDataTypes.BINARY;
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +053030import static org.onosproject.yangutils.datamodel.YangDataTypes.DERIVED;
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053031import static org.onosproject.yangutils.datamodel.YangDataTypes.STRING;
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053032import static org.onosproject.yangutils.parser.impl.parserutils.ListenerErrorLocation.ENTRY;
Vidyashree Rama1db15562016-05-17 16:16:15 +053033import static org.onosproject.yangutils.parser.impl.parserutils.ListenerErrorLocation.EXIT;
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053034import static org.onosproject.yangutils.parser.impl.parserutils.ListenerErrorMessageConstruction.constructListenerErrorMessage;
35import static org.onosproject.yangutils.parser.impl.parserutils.ListenerErrorType.INVALID_HOLDER;
Vidyashree Rama1db15562016-05-17 16:16:15 +053036import static org.onosproject.yangutils.parser.impl.parserutils.ListenerErrorType.MISSING_CURRENT_HOLDER;
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053037import static org.onosproject.yangutils.parser.impl.parserutils.ListenerErrorType.MISSING_HOLDER;
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053038import static org.onosproject.yangutils.parser.impl.parserutils.ListenerValidation.checkStackIsNotEmpty;
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +053039import static org.onosproject.yangutils.utils.RestrictionResolver.processLengthRestriction;
40import static org.onosproject.yangutils.utils.YangConstructType.LENGTH_DATA;
41import static org.onosproject.yangutils.utils.YangConstructType.TYPE_DATA;
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053042
43/*
44 * Reference: RFC6020 and YANG ANTLR Grammar
45 *
46 * ABNF grammar as per RFC6020
47 * length-stmt = length-keyword sep length-arg-str optsep
48 * (";" /
49 * "{" stmtsep
50 * ;; these stmts can appear in any order
51 * [error-message-stmt stmtsep]
52 * [error-app-tag-stmt stmtsep]
53 * [description-stmt stmtsep]
54 * [reference-stmt stmtsep]
55 * "}")
56 *
57 *
58 * ANTLR grammar rule
59 * lengthStatement : LENGTH_KEYWORD length
60 * (STMTEND | LEFT_CURLY_BRACE commonStatements RIGHT_CURLY_BRACE);
61 */
62
63/**
64 * Represents listener based call back function corresponding to the "length"
65 * rule defined in ANTLR grammar file for corresponding ABNF rule in RFC 6020.
66 */
67public final class LengthRestrictionListener {
68
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053069 /**
70 * Creates a new length restriction listener.
71 */
72 private LengthRestrictionListener() {
73 }
74
75 /**
76 * It is called when parser receives an input matching the grammar
77 * rule (length), performs validation and updates the data model
78 * tree.
79 *
80 * @param listener listener's object
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +053081 * @param ctx context object of the grammar rule
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053082 */
83 public static void processLengthRestrictionEntry(TreeWalkListener listener,
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +053084 GeneratedYangParser.LengthStatementContext ctx) {
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053085
86 // Check for stack to be non empty.
87 checkStackIsNotEmpty(listener, MISSING_HOLDER, LENGTH_DATA, ctx.length().getText(), ENTRY);
88
89 Parsable tmpData = listener.getParsedDataStack().peek();
90 if (tmpData.getYangConstructType() == TYPE_DATA) {
91 YangType type = (YangType) tmpData;
Vidyashree Rama1db15562016-05-17 16:16:15 +053092 setLengthRestriction(listener, type, ctx);
Vidyashree Ramaa2f73982016-04-12 23:33:33 +053093 } else {
94 throw new ParserException(constructListenerErrorMessage(INVALID_HOLDER, LENGTH_DATA,
95 ctx.length().getText(), ENTRY));
96 }
97 }
98
99 /**
100 * Sets the length restriction to type.
101 *
Vidyashree Rama1db15562016-05-17 16:16:15 +0530102 * @param listener listener's object
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530103 * @param type Yang type for which length restriction to be set
104 * @param ctx context object of the grammar rule
Vidyashree Ramaa2f73982016-04-12 23:33:33 +0530105 */
Vidyashree Rama1db15562016-05-17 16:16:15 +0530106 private static void setLengthRestriction(TreeWalkListener listener, YangType type,
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +0530107 GeneratedYangParser.LengthStatementContext ctx) {
Vidyashree Ramaa2f73982016-04-12 23:33:33 +0530108
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +0530109 if (type.getDataType() == DERIVED) {
110 ((YangDerivedInfo<YangRangeRestriction>) type.getDataTypeExtendedInfo())
111 .setLengthRestrictionString(ctx.length().getText());
112 ((YangDerivedInfo<YangRangeRestriction>) type.getDataTypeExtendedInfo())
113 .setLineNumber(ctx.getStart().getLine());
114 ((YangDerivedInfo<YangRangeRestriction>) type.getDataTypeExtendedInfo())
115 .setCharPosition(ctx.getStart().getCharPositionInLine());
116 return;
117 }
Vidyashree Ramaa2f73982016-04-12 23:33:33 +0530118
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530119 if (type.getDataType() != STRING && type.getDataType() != BINARY) {
Vidyashree Ramaa2f73982016-04-12 23:33:33 +0530120 ParserException parserException = new ParserException("YANG file error : " +
121 YangConstructType.getYangConstructType(LENGTH_DATA) + " name " + ctx.length().getText() +
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530122 " can be used to restrict the built-in type string/binary or types derived from string/binary.");
Vidyashree Ramaa2f73982016-04-12 23:33:33 +0530123 parserException.setLine(ctx.getStart().getLine());
124 parserException.setCharPosition(ctx.getStart().getCharPositionInLine());
125 throw parserException;
126 }
127
Gaurav Agrawalcfa1c412016-05-03 00:41:48 +0530128 YangRangeRestriction lengthRestriction = processLengthRestriction(null, ctx.getStart().getLine(),
129 ctx.getStart().getCharPositionInLine(), false, ctx.length().getText());
130
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530131 if (type.getDataType() == STRING) {
132 YangStringRestriction stringRestriction = (YangStringRestriction) type.getDataTypeExtendedInfo();
133 if (stringRestriction == null) {
134 stringRestriction = new YangStringRestriction();
135 type.setDataTypeExtendedInfo(stringRestriction);
136 }
Vidyashree Ramaa2f73982016-04-12 23:33:33 +0530137
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530138 stringRestriction.setLengthRestriction(lengthRestriction);
139 } else {
140 type.setDataTypeExtendedInfo(lengthRestriction);
Vidyashree Ramaa2f73982016-04-12 23:33:33 +0530141 }
142
Vidyashree Rama1db15562016-05-17 16:16:15 +0530143 listener.getParsedDataStack().push(lengthRestriction);
144 }
145
146 /**
147 * Performs validation and updates the data model tree.
148 * It is called when parser exits from grammar rule (length).
149 *
150 * @param listener listener's object
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530151 * @param ctx context object of the grammar rule
Vidyashree Rama1db15562016-05-17 16:16:15 +0530152 */
153 public static void processLengthRestrictionExit(TreeWalkListener listener,
154 GeneratedYangParser.LengthStatementContext ctx) {
155
156 // Check for stack to be non empty.
157 checkStackIsNotEmpty(listener, MISSING_HOLDER, LENGTH_DATA, ctx.length().getText(), EXIT);
158
159 Parsable tmpData = listener.getParsedDataStack().peek();
160 if (tmpData instanceof YangRangeRestriction) {
161 listener.getParsedDataStack().pop();
162 } else if (tmpData instanceof YangType
163 && ((YangType) tmpData).getDataType() == DERIVED) {
164 // TODO : need to handle in linker
165 } else {
166 throw new ParserException(constructListenerErrorMessage(MISSING_CURRENT_HOLDER, LENGTH_DATA,
167 ctx.length().getText(), EXIT));
168 }
Vidyashree Ramaa2f73982016-04-12 23:33:33 +0530169 }
170}