blob: 1c3d82b393d1eef02a027515b0d5b8a69f068461 [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
cparsonsb68bf022018-10-17 10:33:40 -070017import com.google.common.collect.ImmutableList;
cparsons11c9f202018-07-11 10:30:02 -070018import com.google.common.collect.ImmutableMap;
cparsons5d85e752018-06-26 13:47:28 -070019import com.google.devtools.build.lib.cmdline.Label;
cparsons17710542018-07-13 15:23:46 -070020import com.google.devtools.build.lib.cmdline.LabelSyntaxException;
cparsons5d85e752018-06-26 13:47:28 -070021import com.google.devtools.build.lib.events.Location;
22import com.google.devtools.build.lib.skylarkbuildapi.FileApi;
23import com.google.devtools.build.lib.skylarkbuildapi.FileTypeApi;
24import com.google.devtools.build.lib.skylarkbuildapi.ProviderApi;
25import com.google.devtools.build.lib.skylarkbuildapi.SkylarkAspectApi;
cparsons5d85e752018-06-26 13:47:28 -070026import com.google.devtools.build.lib.skylarkbuildapi.SkylarkRuleFunctionsApi;
27import com.google.devtools.build.lib.syntax.BaseFunction;
28import com.google.devtools.build.lib.syntax.Environment;
29import com.google.devtools.build.lib.syntax.EvalException;
30import com.google.devtools.build.lib.syntax.FuncallExpression;
cparsonsd790ce42018-06-27 15:29:02 -070031import com.google.devtools.build.lib.syntax.Runtime;
cparsons5d85e752018-06-26 13:47:28 -070032import com.google.devtools.build.lib.syntax.SkylarkDict;
33import com.google.devtools.build.lib.syntax.SkylarkList;
cparsonsb68bf022018-10-17 10:33:40 -070034import com.google.devtools.build.lib.syntax.SkylarkType;
cparsons11c9f202018-07-11 10:30:02 -070035import com.google.devtools.build.skydoc.fakebuildapi.FakeDescriptor.Type;
cparsonsfc0e52f2018-07-09 11:06:57 -070036import com.google.devtools.build.skydoc.rendering.AttributeInfo;
cparsonsb68bf022018-10-17 10:33:40 -070037import com.google.devtools.build.skydoc.rendering.ProviderFieldInfo;
38import com.google.devtools.build.skydoc.rendering.ProviderInfo;
cparsons5d85e752018-06-26 13:47:28 -070039import com.google.devtools.build.skydoc.rendering.RuleInfo;
cparsons11c9f202018-07-11 10:30:02 -070040import java.util.Comparator;
cparsons5d85e752018-06-26 13:47:28 -070041import java.util.List;
cparsonsb68bf022018-10-17 10:33:40 -070042import java.util.Map;
cparsonsfc0e52f2018-07-09 11:06:57 -070043import java.util.stream.Collectors;
cparsonsd790ce42018-06-27 15:29:02 -070044import javax.annotation.Nullable;
cparsons5d85e752018-06-26 13:47:28 -070045
46/**
47 * Fake implementation of {@link SkylarkRuleFunctionsApi}.
48 *
cparsonsd790ce42018-06-27 15:29:02 -070049 * <p>This fake hooks into the global {@code rule()} function, adding descriptors of calls of that
50 * function to a {@link List} given in the class constructor.</p>
cparsons5d85e752018-06-26 13:47:28 -070051 */
52public class FakeSkylarkRuleFunctionsApi implements SkylarkRuleFunctionsApi<FileApi> {
53
cparsons11c9f202018-07-11 10:30:02 -070054 private static final FakeDescriptor IMPLICIT_NAME_ATTRIBUTE_DESCRIPTOR =
brandjon09f7cbc2018-10-04 12:45:07 -070055 new FakeDescriptor(Type.STRING, "A unique name for this target.", true);
cparsons5d85e752018-06-26 13:47:28 -070056 private final List<RuleInfo> ruleInfoList;
cparsonsb68bf022018-10-17 10:33:40 -070057 private final List<ProviderInfo> providerInfoList;
cparsons5d85e752018-06-26 13:47:28 -070058
59 /**
60 * Constructor.
61 *
62 * @param ruleInfoList the list of {@link RuleInfo} objects to which rule() invocation information
63 * will be added
cparsonsb68bf022018-10-17 10:33:40 -070064 * @param providerInfoList the list of {@link ProviderInfo} objects to which provider()
65 * invocation information will be added
cparsons5d85e752018-06-26 13:47:28 -070066 */
cparsonsb68bf022018-10-17 10:33:40 -070067 public FakeSkylarkRuleFunctionsApi(List<RuleInfo> ruleInfoList,
68 List<ProviderInfo> providerInfoList) {
cparsons5d85e752018-06-26 13:47:28 -070069 this.ruleInfoList = ruleInfoList;
cparsonsb68bf022018-10-17 10:33:40 -070070 this.providerInfoList = providerInfoList;
cparsons5d85e752018-06-26 13:47:28 -070071 }
72
73 @Override
74 public ProviderApi provider(String doc, Object fields, Location location) throws EvalException {
cparsonsb68bf022018-10-17 10:33:40 -070075 FakeProviderApi fakeProvider = new FakeProviderApi();
76 // Field documentation will be output preserving the order in which the fields are listed.
77 ImmutableList.Builder<ProviderFieldInfo> providerFieldInfos = ImmutableList.builder();
78 if (fields instanceof SkylarkList) {
79 @SuppressWarnings("unchecked")
80 SkylarkList<String> fieldNames = (SkylarkList<String>)
81 SkylarkType.cast(
82 fields,
83 SkylarkList.class, String.class, location,
84 "Expected list of strings or dictionary of string -> string for 'fields'");
85 for (String fieldName : fieldNames) {
86 providerFieldInfos.add(new ProviderFieldInfo(fieldName));
87 }
88 } else if (fields instanceof SkylarkDict) {
89 Map<String, String> dict = SkylarkType.castMap(
90 fields,
91 String.class, String.class,
92 "Expected list of strings or dictionary of string -> string for 'fields'");
93 for (Map.Entry<String, String> fieldEntry : dict.entrySet()) {
94 providerFieldInfos.add(new ProviderFieldInfo(fieldEntry.getKey(), fieldEntry.getValue()));
95 }
96 } else {
97 // fields is NONE, so there is no field information to add.
98 }
99 providerInfoList.add(new ProviderInfo(fakeProvider, doc, providerFieldInfos.build()));
100 return fakeProvider;
cparsons5d85e752018-06-26 13:47:28 -0700101 }
102
103 @Override
104 public BaseFunction rule(BaseFunction implementation, Boolean test, Object attrs,
105 Object implicitOutputs, Boolean executable, Boolean outputToGenfiles,
106 SkylarkList<?> fragments, SkylarkList<?> hostFragments, Boolean skylarkTestable,
107 SkylarkList<?> toolchains, String doc, SkylarkList<?> providesArg,
108 Boolean executionPlatformConstraintsAllowed, SkylarkList<?> execCompatibleWith,
109 FuncallExpression ast, Environment funcallEnv) throws EvalException {
cparsonsfc0e52f2018-07-09 11:06:57 -0700110 List<AttributeInfo> attrInfos;
cparsons11c9f202018-07-11 10:30:02 -0700111 ImmutableMap.Builder<String, FakeDescriptor> attrsMapBuilder = ImmutableMap.builder();
cparsonsd790ce42018-06-27 15:29:02 -0700112 if (attrs != null && attrs != Runtime.NONE) {
cparsons5d85e752018-06-26 13:47:28 -0700113 SkylarkDict<?, ?> attrsDict = (SkylarkDict<?, ?>) attrs;
cparsons11c9f202018-07-11 10:30:02 -0700114 attrsMapBuilder.putAll(attrsDict.getContents(String.class, FakeDescriptor.class, "attrs"));
cparsons5d85e752018-06-26 13:47:28 -0700115 }
116
cparsons11c9f202018-07-11 10:30:02 -0700117 attrsMapBuilder.put("name", IMPLICIT_NAME_ATTRIBUTE_DESCRIPTOR);
118 attrInfos = attrsMapBuilder.build().entrySet().stream()
cparsons9e58e222018-08-28 13:01:34 -0700119 .filter(entry -> !entry.getKey().startsWith("_"))
cparsons11c9f202018-07-11 10:30:02 -0700120 .map(entry -> new AttributeInfo(
121 entry.getKey(),
122 entry.getValue().getDocString(),
123 entry.getValue().getType().getDescription(),
124 entry.getValue().isMandatory()))
125 .collect(Collectors.toList());
126 attrInfos.sort(new AttributeNameComparator());
127
cparsonsd790ce42018-06-27 15:29:02 -0700128 RuleDefinitionIdentifier functionIdentifier = new RuleDefinitionIdentifier();
cparsonsfc0e52f2018-07-09 11:06:57 -0700129
130 ruleInfoList.add(new RuleInfo(functionIdentifier, ast.getLocation(), doc, attrInfos));
cparsonsd790ce42018-06-27 15:29:02 -0700131 return functionIdentifier;
cparsons5d85e752018-06-26 13:47:28 -0700132 }
133
134 @Override
135 public Label label(String labelString, Boolean relativeToCallerRepository, Location loc,
136 Environment env) throws EvalException {
cparsons17710542018-07-13 15:23:46 -0700137 try {
138 return Label.parseAbsolute(
139 labelString,
140 /* defaultToMain= */ false,
141 /* repositoryMapping= */ ImmutableMap.of());
142 } catch (LabelSyntaxException e) {
143 throw new EvalException(loc, "Illegal absolute label syntax: " + labelString);
144 }
cparsons5d85e752018-06-26 13:47:28 -0700145 }
146
147 @Override
148 public FileTypeApi<FileApi> fileType(SkylarkList<?> types, Location loc, Environment env)
149 throws EvalException {
150 return null;
151 }
152
153 @Override
154 public SkylarkAspectApi aspect(BaseFunction implementation, SkylarkList<?> attributeAspects,
155 Object attrs, SkylarkList<?> requiredAspectProvidersArg, SkylarkList<?> providesArg,
156 SkylarkList<?> fragments, SkylarkList<?> hostFragments, SkylarkList<?> toolchains, String doc,
157 FuncallExpression ast, Environment funcallEnv) throws EvalException {
cparsons6922b5f2018-08-28 08:32:09 -0700158 return new FakeSkylarkAspect();
cparsons5d85e752018-06-26 13:47:28 -0700159 }
cparsonsd790ce42018-06-27 15:29:02 -0700160
161 /**
162 * A fake {@link BaseFunction} implementation which serves as an identifier for a rule definition.
163 * A skylark invocation of 'rule()' should spawn a unique instance of this class and return it.
164 * Thus, skylark code such as 'foo = rule()' will result in 'foo' being assigned to a unique
165 * identifier, which can later be matched to a registered rule() invocation saved by the fake
166 * build API implementation.
167 */
168 private static class RuleDefinitionIdentifier extends BaseFunction {
169
cparsons93adb102018-07-11 11:44:17 -0700170 private static int idCounter = 0;
171
cparsonsd790ce42018-06-27 15:29:02 -0700172 public RuleDefinitionIdentifier() {
cparsons93adb102018-07-11 11:44:17 -0700173 super("RuleDefinitionIdentifier" + idCounter++);
cparsonsd790ce42018-06-27 15:29:02 -0700174 }
175
176 @Override
177 public boolean equals(@Nullable Object other) {
178 // Use exact object matching.
179 return this == other;
180 }
181 }
cparsons11c9f202018-07-11 10:30:02 -0700182
183 /**
184 * A comparator for {@link AttributeInfo} objects which sorts by attribute name alphabetically,
185 * except that any attribute named "name" is placed first.
186 */
187 private static class AttributeNameComparator implements Comparator<AttributeInfo> {
188
189 @Override
190 public int compare(AttributeInfo o1, AttributeInfo o2) {
191 if (o1.getName().equals("name")) {
192 return o2.getName().equals("name") ? 0 : -1;
193 } else if (o2.getName().equals("name")) {
194 return 1;
195 } else {
196 return o1.getName().compareTo(o2.getName());
197 }
198 }
199 }
cparsons5d85e752018-06-26 13:47:28 -0700200}