blob: aec9eb1d91c2c9f031138ad9105a25d52709a69f [file] [log] [blame]
Damien Martin-Guillerezf88f4d82015-09-25 13:56:55 +00001// Copyright 2014 The Bazel Authors. All rights reserved.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01002//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14package com.google.devtools.build.docgen;
15
16import com.google.common.base.Preconditions;
17import com.google.common.collect.ImmutableMap;
18import com.google.common.collect.ImmutableSet;
Luis Fernando Pino Duque2b0b5cc2016-04-26 09:31:27 +000019import com.google.devtools.build.lib.analysis.ConfiguredRuleClassProvider;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010020import com.google.devtools.build.lib.analysis.RuleDefinition;
Lukacs Berki6e91eb92015-09-21 09:12:37 +000021import com.google.devtools.build.lib.cmdline.Label;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010022import com.google.devtools.build.lib.packages.Attribute;
Lukacs Berkiffa73ad2015-09-18 11:40:12 +000023import com.google.devtools.build.lib.packages.BuildType;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010024import com.google.devtools.build.lib.packages.TriState;
Lukacs Berkiffa73ad2015-09-18 11:40:12 +000025import com.google.devtools.build.lib.syntax.Type;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010026import java.util.HashMap;
27import java.util.LinkedList;
28import java.util.Map;
29import java.util.Set;
30
31/**
32 * A class storing a rule attribute documentation along with some meta information.
33 * The class provides functionality to compute the ancestry level of this attribute's
34 * generator rule definition class compared to other rule definition classes.
35 *
36 * <p>Warning, two RuleDocumentationAttribute objects are equal based on only the attributeName.
37 */
David Chen31fce142015-08-18 08:39:51 +000038public class RuleDocumentationAttribute implements Comparable<RuleDocumentationAttribute> {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010039
Googlercffe7352017-03-02 16:01:13 +000040 private static final ImmutableMap<Type<?>, String> TYPE_DESC =
41 ImmutableMap.<Type<?>, String>builder()
42 .put(Type.BOOLEAN, "Boolean")
43 .put(Type.INTEGER, "Integer")
44 .put(Type.INTEGER_LIST, "List of integers")
45 .put(Type.STRING, "String")
46 .put(Type.STRING_LIST, "List of strings")
47 .put(BuildType.TRISTATE, "Integer")
48 .put(BuildType.LABEL, "<a href=\"../build-ref.html#labels\">Label</a>")
49 .put(BuildType.LABEL_LIST, "List of <a href=\"../build-ref.html#labels\">labels</a>")
50 .put(
51 BuildType.LABEL_DICT_UNARY,
52 "Dictionary mapping strings to <a href=\"../build-ref.html#labels\">labels</a>")
53 .put(BuildType.NODEP_LABEL, "<a href=\"../build-ref.html#name\">Name</a>")
54 .put(BuildType.NODEP_LABEL_LIST, "List of <a href=\"../build-ref.html#name\">names</a>")
55 .put(BuildType.OUTPUT, "<a href=\"../build-ref.html#filename\">Filename</a>")
56 .put(
57 BuildType.OUTPUT_LIST, "List of <a href=\"../build-ref.html#filename\">filenames</a>")
58 .build();
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010059
60 private final Class<? extends RuleDefinition> definitionClass;
61 private final String attributeName;
62 private final String htmlDocumentation;
63 private final String commonType;
David Chenac13e222016-02-24 22:17:42 +000064 // Used to expand rule link references in the attribute documentation.
65 private RuleLinkExpander linkExpander;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010066 private int startLineCnt;
David Chenac13e222016-02-24 22:17:42 +000067 private String fileName;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010068 private Set<String> flags;
David Chen31fce142015-08-18 08:39:51 +000069 private Attribute attribute;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010070
71 /**
72 * Creates common RuleDocumentationAttribute such as deps or data.
73 * These attribute docs have no definitionClass or htmlDocumentation (it's in the BE header).
74 */
75 static RuleDocumentationAttribute create(
76 String attributeName, String commonType, String htmlDocumentation) {
77 RuleDocumentationAttribute docAttribute = new RuleDocumentationAttribute(
David Chenac13e222016-02-24 22:17:42 +000078 null, attributeName, htmlDocumentation, 0, "", ImmutableSet.<String>of(), commonType);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010079 return docAttribute;
80 }
81
82 /**
83 * Creates a RuleDocumentationAttribute with all the necessary fields for explicitly
84 * defined rule attributes.
85 */
86 static RuleDocumentationAttribute create(Class<? extends RuleDefinition> definitionClass,
David Chenac13e222016-02-24 22:17:42 +000087 String attributeName, String htmlDocumentation, int startLineCnt, String fileName,
88 Set<String> flags) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010089 return new RuleDocumentationAttribute(definitionClass, attributeName, htmlDocumentation,
David Chenac13e222016-02-24 22:17:42 +000090 startLineCnt, fileName, flags, null);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010091 }
92
93 private RuleDocumentationAttribute(Class<? extends RuleDefinition> definitionClass,
David Chenac13e222016-02-24 22:17:42 +000094 String attributeName, String htmlDocumentation, int startLineCnt, String fileName,
95 Set<String> flags, String commonType) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010096 Preconditions.checkNotNull(attributeName, "AttributeName must not be null.");
97 this.definitionClass = definitionClass;
98 this.attributeName = attributeName;
99 this.htmlDocumentation = htmlDocumentation;
100 this.startLineCnt = startLineCnt;
101 this.flags = flags;
102 this.commonType = commonType;
103 }
104
105 /**
David Chen31fce142015-08-18 08:39:51 +0000106 * Sets the Attribute object that this documents.
107 */
108 void setAttribute(Attribute attribute) {
109 this.attribute = attribute;
110 }
111
112 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100113 * Returns the name of the rule attribute.
114 */
David Chen31fce142015-08-18 08:39:51 +0000115 public String getAttributeName() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100116 return attributeName;
117 }
118
119 /**
David Chen31fce142015-08-18 08:39:51 +0000120 * Returns whether this attribute is marked as deprecated.
121 */
122 public boolean isDeprecated() {
123 return hasFlag(DocgenConsts.FLAG_DEPRECATED);
124 }
125
126 /**
David Chenac13e222016-02-24 22:17:42 +0000127 * Sets the {@link RuleLinkExpander} to be used to expand links in the HTML documentation.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100128 */
David Chenac13e222016-02-24 22:17:42 +0000129 public void setRuleLinkExpander(RuleLinkExpander linkExpander) {
130 this.linkExpander = linkExpander;
131 }
132
133 /**
134 * Returns the html documentation of the rule attribute.
135 */
136 public String getHtmlDocumentation() throws BuildEncyclopediaDocException {
137 String expandedHtmlDoc = htmlDocumentation;
138 if (linkExpander != null) {
139 try {
140 expandedHtmlDoc = linkExpander.expand(expandedHtmlDoc);
141 } catch (IllegalArgumentException e) {
142 throw new BuildEncyclopediaDocException(fileName, startLineCnt, e.getMessage());
143 }
144 }
145 return expandedHtmlDoc;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100146 }
147
David Chen31fce142015-08-18 08:39:51 +0000148 private String getDefaultValue() {
149 if (attribute == null) {
150 return "";
151 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100152 String prefix = "; default is ";
153 Object value = attribute.getDefaultValueForTesting();
154 if (value instanceof Boolean) {
155 return prefix + ((Boolean) value ? "1" : "0");
156 } else if (value instanceof Integer) {
157 return prefix + String.valueOf(value);
158 } else if (value instanceof String && !(((String) value).isEmpty())) {
159 return prefix + "\"" + value + "\"";
160 } else if (value instanceof TriState) {
161 switch((TriState) value) {
162 case AUTO:
163 return prefix + "-1";
164 case NO:
165 return prefix + "0";
166 case YES:
167 return prefix + "1";
168 }
169 } else if (value instanceof Label) {
170 return prefix + "<code>" + value + "</code>";
171 }
172 return "";
173 }
174
175 /**
David Chen31fce142015-08-18 08:39:51 +0000176 * Returns a string containing the synopsis for this attribute.
177 */
178 public String getSynopsis() {
179 if (attribute == null) {
180 return "";
181 }
182 StringBuilder sb = new StringBuilder()
183 .append(TYPE_DESC.get(attribute.getType()))
184 .append("; " + (attribute.isMandatory() ? "required" : "optional"))
gregce27a91eb2017-04-04 22:30:25 +0000185 .append(!attribute.isConfigurable()
186 ? String.format("; <a href=\"%s#configurable-attributes\">nonconfigurable</a>",
187 RuleDocumentation.COMMON_DEFINITIONS_PAGE)
188 : "")
David Chen31fce142015-08-18 08:39:51 +0000189 .append(getDefaultValue());
190 return sb.toString();
191 }
192
193 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100194 * Returns the number of first line of the attribute documentation in its declaration file.
195 */
196 int getStartLineCnt() {
197 return startLineCnt;
198 }
199
200 /**
201 * Returns true if the attribute doc is of a common attribute type.
202 */
David Chen31fce142015-08-18 08:39:51 +0000203 public boolean isCommonType() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100204 return commonType != null;
205 }
206
207 /**
208 * Returns the common attribute type if this attribute doc is of a common type
209 * otherwise actualRule.
210 */
211 String getGeneratedInRule(String actualRule) {
212 return isCommonType() ? commonType : actualRule;
213 }
214
215 /**
216 * Returns true if this attribute documentation has the parameter flag.
217 */
218 boolean hasFlag(String flag) {
219 return flags.contains(flag);
220 }
221
222 /**
223 * Returns the length of a shortest path from usingClass to the definitionClass of this
224 * RuleDocumentationAttribute in the rule definition ancestry graph. Returns -1
225 * if definitionClass is not the ancestor (transitively) of usingClass.
226 */
Luis Fernando Pino Duque2b0b5cc2016-04-26 09:31:27 +0000227 int getDefinitionClassAncestryLevel(Class<? extends RuleDefinition> usingClass,
228 ConfiguredRuleClassProvider ruleClassProvider) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100229 if (usingClass.equals(definitionClass)) {
230 return 0;
231 }
232 // Storing nodes (rule class definitions) with the length of the shortest path from usingClass
233 Map<Class<? extends RuleDefinition>, Integer> visited = new HashMap<>();
234 LinkedList<Class<? extends RuleDefinition>> toVisit = new LinkedList<>();
235 visited.put(usingClass, 0);
236 toVisit.add(usingClass);
237 // Searching the shortest path from usingClass to this.definitionClass using BFS
238 do {
239 Class<? extends RuleDefinition> ancestor = toVisit.removeFirst();
Luis Fernando Pino Duque2b0b5cc2016-04-26 09:31:27 +0000240 visitAncestor(ancestor, visited, toVisit, ruleClassProvider);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100241 if (ancestor.equals(definitionClass)) {
242 return visited.get(ancestor);
243 }
244 } while (!toVisit.isEmpty());
245 return -1;
246 }
247
248 private void visitAncestor(
249 Class<? extends RuleDefinition> usingClass,
250 Map<Class<? extends RuleDefinition>, Integer> visited,
Luis Fernando Pino Duque2b0b5cc2016-04-26 09:31:27 +0000251 LinkedList<Class<? extends RuleDefinition>> toVisit,
252 ConfiguredRuleClassProvider ruleClassProvider) {
253 RuleDefinition instance = getRuleDefinition(usingClass, ruleClassProvider);
Googler58505032015-03-19 16:12:34 +0000254 for (Class<? extends RuleDefinition> ancestor : instance.getMetadata().ancestors()) {
255 if (!visited.containsKey(ancestor)) {
256 toVisit.addLast(ancestor);
257 visited.put(ancestor, visited.get(usingClass) + 1);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100258 }
259 }
260 }
261
Luis Fernando Pino Duque2b0b5cc2016-04-26 09:31:27 +0000262 private RuleDefinition getRuleDefinition(Class<? extends RuleDefinition> usingClass,
263 ConfiguredRuleClassProvider ruleClassProvider) {
264 if (ruleClassProvider == null) {
265 try {
Ulf Adams653a16a2016-08-11 12:39:46 +0000266 return usingClass.getConstructor().newInstance();
267 } catch (ReflectiveOperationException | IllegalArgumentException e) {
Luis Fernando Pino Duque2b0b5cc2016-04-26 09:31:27 +0000268 throw new IllegalStateException(e);
269 }
270 }
271 return ruleClassProvider.getRuleClassDefinition(usingClass.getName());
272 }
273
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100274 private int getAttributeOrderingPriority(RuleDocumentationAttribute attribute) {
275 if (DocgenConsts.ATTRIBUTE_ORDERING.containsKey(attribute.attributeName)) {
276 return DocgenConsts.ATTRIBUTE_ORDERING.get(attribute.attributeName);
277 } else {
278 return 0;
279 }
280 }
281
282 @Override
283 public int compareTo(RuleDocumentationAttribute o) {
284 int thisPriority = getAttributeOrderingPriority(this);
285 int otherPriority = getAttributeOrderingPriority(o);
286 if (thisPriority > otherPriority) {
287 return 1;
288 } else if (thisPriority < otherPriority) {
289 return -1;
290 } else {
291 return this.attributeName.compareTo(o.attributeName);
292 }
293 }
294
295 @Override
296 public boolean equals(Object obj) {
297 if (this == obj) {
298 return true;
299 }
300 if (!(obj instanceof RuleDocumentationAttribute)) {
301 return false;
302 }
303 return attributeName.equals(((RuleDocumentationAttribute) obj).attributeName);
304 }
305
306 @Override
307 public int hashCode() {
308 return attributeName.hashCode();
309 }
310}