blob: 8cfe77943c617ada17b029ae168ca9b92400e086 [file] [log] [blame]
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +05301/*
2 * Copyright 2016-present Open Networking Laboratory
3 *
4 * Licensed under the Apache License, Version 2.0 (the "License"); you may not
5 * use this file except in compliance with the License. You may obtain a copy of
6 * 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, WITHOUT
12 * WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the
13 * License for the specific language governing permissions and limitations under
14 * the License.
15 */
16
17package org.onosproject.yangutils.linker.impl;
18
Vidyashree Rama1b499062016-07-12 20:52:48 +053019import java.util.Collections;
Gaurav Agrawal95b416c2016-06-07 14:00:26 +053020import java.util.HashSet;
Vidyashree Rama1b499062016-07-12 20:52:48 +053021import java.util.Iterator;
22import java.util.LinkedList;
23import java.util.List;
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053024import java.util.Set;
Gaurav Agrawal95b416c2016-06-07 14:00:26 +053025import org.onosproject.yangutils.datamodel.ResolvableType;
Vidyashree Rama1b499062016-07-12 20:52:48 +053026import org.onosproject.yangutils.datamodel.YangImport;
27import org.onosproject.yangutils.datamodel.YangInclude;
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053028import org.onosproject.yangutils.datamodel.YangNode;
Gaurav Agrawal95b416c2016-06-07 14:00:26 +053029import org.onosproject.yangutils.datamodel.YangReferenceResolver;
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053030import org.onosproject.yangutils.datamodel.YangSubModule;
31import org.onosproject.yangutils.datamodel.exceptions.DataModelException;
32import org.onosproject.yangutils.linker.YangLinker;
33import org.onosproject.yangutils.linker.exceptions.LinkerException;
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053034
35import static org.onosproject.yangutils.utils.UtilConstants.NEW_LINE;
36
37/**
38 * Representation of entity which provides linking service of YANG files.
39 */
VinodKumarS-Huawei2ee9e7e2016-06-01 14:30:22 +053040public class YangLinkerManager
41 implements YangLinker {
Gaurav Agrawal95b416c2016-06-07 14:00:26 +053042
43 /*
44 * Set of all the YANG nodes, corresponding to the YANG files parsed by
45 * parser.
46 */
47 Set<YangNode> yangNodeSet = new HashSet<>();
48
49 /**
50 * Returns set of YANG node.
51 *
52 * @return set of YANG node
53 */
54 public Set<YangNode> getYangNodeSet() {
55 return yangNodeSet;
56 }
57
58 /**
59 * Creates YANG nodes set.
60 *
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053061 * @param yangNodeSet YANG node information set
Gaurav Agrawal95b416c2016-06-07 14:00:26 +053062 */
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053063 public void createYangNodeSet(Set<YangNode> yangNodeSet) {
64 getYangNodeSet().addAll(yangNodeSet);
Gaurav Agrawal95b416c2016-06-07 14:00:26 +053065 }
66
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053067 @Override
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053068 public void resolveDependencies(Set<YangNode> yangNodeSet) {
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053069
Gaurav Agrawal95b416c2016-06-07 14:00:26 +053070 // Create YANG node set.
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053071 createYangNodeSet(yangNodeSet);
Gaurav Agrawal95b416c2016-06-07 14:00:26 +053072
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053073 // Carry out linking of sub module with module.
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053074 linkSubModulesToParentModule(yangNodeSet);
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053075
76 // Add references to import list.
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053077 addRefToYangFilesImportList(yangNodeSet);
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053078
79 // Add reference to include list.
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053080 addRefToYangFilesIncludeList(yangNodeSet);
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053081
Vidyashree Rama1b499062016-07-12 20:52:48 +053082 // Update the priority for all the files.
83 updateFilePriority(yangNodeSet);
84
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053085 // TODO check for circular import/include.
86
87 // Carry out inter-file linking.
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053088 processInterFileLinking(yangNodeSet);
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053089 }
90
91 /**
92 * Resolves sub-module linking by linking sub module with parent module.
93 *
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053094 * @param yangNodeSet set of YANG files info
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +053095 * @throws LinkerException fails to link sub-module to parent module
96 */
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053097 public void linkSubModulesToParentModule(Set<YangNode> yangNodeSet)
VinodKumarS-Huawei2ee9e7e2016-06-01 14:30:22 +053098 throws LinkerException {
Gaurav Agrawal8a5af142016-06-15 13:58:01 +053099 for (YangNode yangNode : yangNodeSet) {
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530100 if (yangNode instanceof YangSubModule) {
101 try {
Gaurav Agrawal95b416c2016-06-07 14:00:26 +0530102 ((YangSubModule) yangNode).linkWithModule(getYangNodeSet());
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530103 } catch (DataModelException e) {
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530104 String errorInfo = "YANG file error: " + yangNode.getName() + " at line: "
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530105 + e.getLineNumber() + " at position: " + e.getCharPositionInLine() + NEW_LINE
106 + e.getMessage();
107 throw new LinkerException(errorInfo);
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530108 // TODO add file path in exception message in util manager.
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530109 }
110 }
111 }
112 }
113
114 /**
115 * Adds imported node information to the import list.
116 *
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530117 * @param yangNodeSet set of YANG files info
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530118 * @throws LinkerException fails to find imported module
119 */
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530120 public void addRefToYangFilesImportList(Set<YangNode> yangNodeSet) throws LinkerException {
121 for (YangNode yangNode : yangNodeSet) {
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530122 if (yangNode instanceof YangReferenceResolver) {
Gaurav Agrawal95b416c2016-06-07 14:00:26 +0530123 try {
124 ((YangReferenceResolver) yangNode).addReferencesToImportList(getYangNodeSet());
125 } catch (DataModelException e) {
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530126 String errorInfo = "Error in file: " + yangNode.getName() + " at line: "
Gaurav Agrawal95b416c2016-06-07 14:00:26 +0530127 + e.getLineNumber() + " at position: " + e.getCharPositionInLine() + NEW_LINE
128 + e.getMessage();
129 throw new LinkerException(errorInfo);
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530130 // TODO add file path in exception message in util manager.
Gaurav Agrawal95b416c2016-06-07 14:00:26 +0530131 }
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530132 }
133 }
134 }
135
136 /**
137 * Adds included node information to the include list.
138 *
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530139 * @param yangNodeSet set of YANG files info
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530140 * @throws LinkerException fails to find included sub-module
141 */
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530142 public void addRefToYangFilesIncludeList(Set<YangNode> yangNodeSet) throws LinkerException {
143 for (YangNode yangNode : yangNodeSet) {
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530144 if (yangNode instanceof YangReferenceResolver) {
Gaurav Agrawal95b416c2016-06-07 14:00:26 +0530145 try {
146 ((YangReferenceResolver) yangNode).addReferencesToIncludeList(getYangNodeSet());
147 } catch (DataModelException e) {
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530148 String errorInfo = "Error in file: " + yangNode.getName() + " at line: "
Gaurav Agrawal95b416c2016-06-07 14:00:26 +0530149 + e.getLineNumber() + " at position: " + e.getCharPositionInLine() + NEW_LINE
150 + e.getMessage();
151 throw new LinkerException(errorInfo);
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530152 // TODO add file path in exception message in util manager.
Gaurav Agrawal95b416c2016-06-07 14:00:26 +0530153 }
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530154 }
155 }
156 }
157
158 /**
159 * Processes inter file linking for type and uses.
160 *
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530161 * @param yangNodeSet set of YANG files info
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530162 * @throws LinkerException a violation in linker execution
163 */
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530164 public void processInterFileLinking(Set<YangNode> yangNodeSet)
VinodKumarS-Huawei2ee9e7e2016-06-01 14:30:22 +0530165 throws LinkerException {
Vidyashree Rama1b499062016-07-12 20:52:48 +0530166 List<YangNode> yangNodeSortedList = new LinkedList<>();
167 yangNodeSortedList.addAll(yangNodeSet);
168 Collections.sort(yangNodeSortedList);
169 for (YangNode yangNode : yangNodeSortedList) {
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530170 try {
Vidyashree Ramadeac28b2016-06-20 15:12:43 +0530171 ((YangReferenceResolver) yangNode)
172 .resolveInterFileLinking(ResolvableType.YANG_IF_FEATURE);
Shankara-Huaweidf7b9ca2016-07-14 11:35:34 +0530173 ((YangReferenceResolver) yangNode)
174 .resolveInterFileLinking(ResolvableType.YANG_USES);
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530175 ((YangReferenceResolver) yangNode)
Bharat saraswalb1170bd2016-07-14 13:26:18 +0530176 .resolveInterFileLinking(ResolvableType.YANG_AUGMENT);
177 ((YangReferenceResolver) yangNode)
VinodKumarS-Huawei2ee9e7e2016-06-01 14:30:22 +0530178 .resolveInterFileLinking(ResolvableType.YANG_DERIVED_DATA_TYPE);
janani be18b5342016-07-13 21:06:41 +0530179 ((YangReferenceResolver) yangNode)
180 .resolveInterFileLinking(ResolvableType.YANG_LEAFREF);
Shankara-Huaweidf7b9ca2016-07-14 11:35:34 +0530181 ((YangReferenceResolver) yangNode)
182 .resolveInterFileLinking(ResolvableType.YANG_BASE);
183 ((YangReferenceResolver) yangNode)
184 .resolveInterFileLinking(ResolvableType.YANG_IDENTITYREF);
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530185 } catch (DataModelException e) {
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530186 String errorInfo = "Error in file: " + yangNode.getName() + " at line: "
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530187 + e.getLineNumber() + " at position: " + e.getCharPositionInLine() + NEW_LINE + e.getMessage();
188 throw new LinkerException(errorInfo);
Gaurav Agrawal8a5af142016-06-15 13:58:01 +0530189 // TODO add file path in exception message in util manager.
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530190 }
191 }
192 }
Vidyashree Rama1b499062016-07-12 20:52:48 +0530193
194 /**
195 * Updates the priority for all the input files.
196 *
197 * @param yangNodeSet set of YANG files info
198 */
199 public void updateFilePriority(Set<YangNode> yangNodeSet) {
200 for (YangNode yangNode : yangNodeSet) {
201 updateFilePriorityOfNode(yangNode);
202 }
203 }
204
205 /**
206 * Updates priority of the node.
207 *
208 * @param yangNode YANG node information
209 */
210 public void updateFilePriorityOfNode(YangNode yangNode) {
211 int curNodePriority = yangNode.getPriority();
212 if (yangNode instanceof YangReferenceResolver) {
213 List<YangImport> yangImportList = ((YangReferenceResolver) yangNode).getImportList();
214 if (yangImportList != null && !yangImportList.isEmpty()) {
215 Iterator<YangImport> importInfoIterator = yangImportList.iterator();
216 // Run through the imported list to update priority.
217 while (importInfoIterator.hasNext()) {
218 YangImport yangImport = importInfoIterator.next();
219 YangNode importedNode = yangImport.getImportedNode();
220 if (curNodePriority >= importedNode.getPriority()) {
221 importedNode.setPriority(curNodePriority + 1);
222 updateFilePriorityOfNode(importedNode);
223 }
224 }
225 }
226
227 List<YangInclude> yangIncludeList = ((YangReferenceResolver) yangNode).getIncludeList();
228 if (yangIncludeList != null && !yangIncludeList.isEmpty()) {
229 Iterator<YangInclude> includeInfoIterator = yangIncludeList.iterator();
230 // Run through the imported list to update priority.
231 while (includeInfoIterator.hasNext()) {
232 YangInclude yangInclude = includeInfoIterator.next();
233 YangNode includedNode = yangInclude.getIncludedNode();
234 if (curNodePriority >= includedNode.getPriority()) {
235 includedNode.setPriority(curNodePriority + 1);
236 updateFilePriorityOfNode(includedNode);
237 }
238 }
239 }
240 }
241 }
Gaurav Agrawal0d43bb52016-05-17 18:06:38 +0530242}