blob: e75a4838294a955de61edac610ac0c89a1095f68 [file] [log] [blame]
cparsons5d85e752018-06-26 13:47:28 -07001// Copyright 2018 The Bazel Authors. All rights reserved.
2//
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.
14
15package com.google.devtools.build.skydoc.fakebuildapi;
16
cparsons11c9f202018-07-11 10:30:02 -070017import com.google.common.collect.ImmutableMap;
cparsons5d85e752018-06-26 13:47:28 -070018import com.google.devtools.build.lib.cmdline.Label;
19import com.google.devtools.build.lib.events.Location;
20import com.google.devtools.build.lib.skylarkbuildapi.FileApi;
21import com.google.devtools.build.lib.skylarkbuildapi.FileTypeApi;
22import com.google.devtools.build.lib.skylarkbuildapi.ProviderApi;
23import com.google.devtools.build.lib.skylarkbuildapi.SkylarkAspectApi;
cparsons5d85e752018-06-26 13:47:28 -070024import com.google.devtools.build.lib.skylarkbuildapi.SkylarkRuleFunctionsApi;
25import com.google.devtools.build.lib.syntax.BaseFunction;
26import com.google.devtools.build.lib.syntax.Environment;
27import com.google.devtools.build.lib.syntax.EvalException;
28import com.google.devtools.build.lib.syntax.FuncallExpression;
cparsonsd790ce42018-06-27 15:29:02 -070029import com.google.devtools.build.lib.syntax.Runtime;
cparsons5d85e752018-06-26 13:47:28 -070030import com.google.devtools.build.lib.syntax.SkylarkDict;
31import com.google.devtools.build.lib.syntax.SkylarkList;
cparsons11c9f202018-07-11 10:30:02 -070032import com.google.devtools.build.skydoc.fakebuildapi.FakeDescriptor.Type;
cparsonsfc0e52f2018-07-09 11:06:57 -070033import com.google.devtools.build.skydoc.rendering.AttributeInfo;
cparsons5d85e752018-06-26 13:47:28 -070034import com.google.devtools.build.skydoc.rendering.RuleInfo;
cparsons11c9f202018-07-11 10:30:02 -070035import java.util.Comparator;
cparsons5d85e752018-06-26 13:47:28 -070036import java.util.List;
cparsonsfc0e52f2018-07-09 11:06:57 -070037import java.util.stream.Collectors;
cparsonsd790ce42018-06-27 15:29:02 -070038import javax.annotation.Nullable;
cparsons5d85e752018-06-26 13:47:28 -070039
40/**
41 * Fake implementation of {@link SkylarkRuleFunctionsApi}.
42 *
cparsonsd790ce42018-06-27 15:29:02 -070043 * <p>This fake hooks into the global {@code rule()} function, adding descriptors of calls of that
44 * function to a {@link List} given in the class constructor.</p>
cparsons5d85e752018-06-26 13:47:28 -070045 */
46public class FakeSkylarkRuleFunctionsApi implements SkylarkRuleFunctionsApi<FileApi> {
47
cparsons11c9f202018-07-11 10:30:02 -070048 private static final FakeDescriptor IMPLICIT_NAME_ATTRIBUTE_DESCRIPTOR =
49 new FakeDescriptor(Type.STRING, "A unique name for this rule.", true);
cparsons5d85e752018-06-26 13:47:28 -070050 private final List<RuleInfo> ruleInfoList;
51
52 /**
53 * Constructor.
54 *
55 * @param ruleInfoList the list of {@link RuleInfo} objects to which rule() invocation information
56 * will be added
57 */
58 public FakeSkylarkRuleFunctionsApi(List<RuleInfo> ruleInfoList) {
59 this.ruleInfoList = ruleInfoList;
60 }
61
62 @Override
63 public ProviderApi provider(String doc, Object fields, Location location) throws EvalException {
64 return null;
65 }
66
67 @Override
68 public BaseFunction rule(BaseFunction implementation, Boolean test, Object attrs,
69 Object implicitOutputs, Boolean executable, Boolean outputToGenfiles,
70 SkylarkList<?> fragments, SkylarkList<?> hostFragments, Boolean skylarkTestable,
71 SkylarkList<?> toolchains, String doc, SkylarkList<?> providesArg,
72 Boolean executionPlatformConstraintsAllowed, SkylarkList<?> execCompatibleWith,
73 FuncallExpression ast, Environment funcallEnv) throws EvalException {
cparsonsfc0e52f2018-07-09 11:06:57 -070074 List<AttributeInfo> attrInfos;
cparsons11c9f202018-07-11 10:30:02 -070075 ImmutableMap.Builder<String, FakeDescriptor> attrsMapBuilder = ImmutableMap.builder();
cparsonsd790ce42018-06-27 15:29:02 -070076 if (attrs != null && attrs != Runtime.NONE) {
cparsons5d85e752018-06-26 13:47:28 -070077 SkylarkDict<?, ?> attrsDict = (SkylarkDict<?, ?>) attrs;
cparsons11c9f202018-07-11 10:30:02 -070078 attrsMapBuilder.putAll(attrsDict.getContents(String.class, FakeDescriptor.class, "attrs"));
cparsons5d85e752018-06-26 13:47:28 -070079 }
80
cparsons11c9f202018-07-11 10:30:02 -070081 attrsMapBuilder.put("name", IMPLICIT_NAME_ATTRIBUTE_DESCRIPTOR);
82 attrInfos = attrsMapBuilder.build().entrySet().stream()
83 .map(entry -> new AttributeInfo(
84 entry.getKey(),
85 entry.getValue().getDocString(),
86 entry.getValue().getType().getDescription(),
87 entry.getValue().isMandatory()))
88 .collect(Collectors.toList());
89 attrInfos.sort(new AttributeNameComparator());
90
cparsonsd790ce42018-06-27 15:29:02 -070091 RuleDefinitionIdentifier functionIdentifier = new RuleDefinitionIdentifier();
cparsonsfc0e52f2018-07-09 11:06:57 -070092
93 ruleInfoList.add(new RuleInfo(functionIdentifier, ast.getLocation(), doc, attrInfos));
cparsonsd790ce42018-06-27 15:29:02 -070094 return functionIdentifier;
cparsons5d85e752018-06-26 13:47:28 -070095 }
96
97 @Override
98 public Label label(String labelString, Boolean relativeToCallerRepository, Location loc,
99 Environment env) throws EvalException {
100 return null;
101 }
102
103 @Override
104 public FileTypeApi<FileApi> fileType(SkylarkList<?> types, Location loc, Environment env)
105 throws EvalException {
106 return null;
107 }
108
109 @Override
110 public SkylarkAspectApi aspect(BaseFunction implementation, SkylarkList<?> attributeAspects,
111 Object attrs, SkylarkList<?> requiredAspectProvidersArg, SkylarkList<?> providesArg,
112 SkylarkList<?> fragments, SkylarkList<?> hostFragments, SkylarkList<?> toolchains, String doc,
113 FuncallExpression ast, Environment funcallEnv) throws EvalException {
114 return null;
115 }
cparsonsd790ce42018-06-27 15:29:02 -0700116
117 /**
118 * A fake {@link BaseFunction} implementation which serves as an identifier for a rule definition.
119 * A skylark invocation of 'rule()' should spawn a unique instance of this class and return it.
120 * Thus, skylark code such as 'foo = rule()' will result in 'foo' being assigned to a unique
121 * identifier, which can later be matched to a registered rule() invocation saved by the fake
122 * build API implementation.
123 */
124 private static class RuleDefinitionIdentifier extends BaseFunction {
125
cparsons93adb102018-07-11 11:44:17 -0700126 private static int idCounter = 0;
127
cparsonsd790ce42018-06-27 15:29:02 -0700128 public RuleDefinitionIdentifier() {
cparsons93adb102018-07-11 11:44:17 -0700129 super("RuleDefinitionIdentifier" + idCounter++);
cparsonsd790ce42018-06-27 15:29:02 -0700130 }
131
132 @Override
133 public boolean equals(@Nullable Object other) {
134 // Use exact object matching.
135 return this == other;
136 }
137 }
cparsons11c9f202018-07-11 10:30:02 -0700138
139 /**
140 * A comparator for {@link AttributeInfo} objects which sorts by attribute name alphabetically,
141 * except that any attribute named "name" is placed first.
142 */
143 private static class AttributeNameComparator implements Comparator<AttributeInfo> {
144
145 @Override
146 public int compare(AttributeInfo o1, AttributeInfo o2) {
147 if (o1.getName().equals("name")) {
148 return o2.getName().equals("name") ? 0 : -1;
149 } else if (o2.getName().equals("name")) {
150 return 1;
151 } else {
152 return o1.getName().compareTo(o2.getName());
153 }
154 }
155 }
cparsons5d85e752018-06-26 13:47:28 -0700156}