blob: 51a496dc7a7843157146a6d7af60d63d4f5a53f2 [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.
14
15package com.google.devtools.build.lib.packages;
16
17import com.google.common.annotations.VisibleForTesting;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010018import com.google.common.collect.ImmutableList;
19import com.google.common.collect.ImmutableMap;
20import com.google.common.collect.ImmutableSet;
21import com.google.common.collect.ImmutableSortedSet;
22import com.google.common.collect.Iterables;
23import com.google.common.collect.Lists;
24import com.google.common.collect.Maps;
25import com.google.common.collect.Sets;
Lukacs Berki6e91eb92015-09-21 09:12:37 +000026import com.google.devtools.build.lib.cmdline.Label;
Lukacs Berkia6434362015-09-15 13:56:14 +000027import com.google.devtools.build.lib.cmdline.LabelSyntaxException;
Kristina Chodorow73fa2032015-08-28 17:57:46 +000028import com.google.devtools.build.lib.cmdline.PackageIdentifier;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010029import com.google.devtools.build.lib.collect.CollectionUtils;
30import com.google.devtools.build.lib.collect.ImmutableSortedKeyMap;
31import com.google.devtools.build.lib.events.Event;
32import com.google.devtools.build.lib.events.EventHandler;
33import com.google.devtools.build.lib.events.Location;
34import com.google.devtools.build.lib.packages.AttributeMap.AcceptsLabelAttribute;
35import com.google.devtools.build.lib.packages.License.DistributionType;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010036import com.google.devtools.build.lib.packages.PackageFactory.Globber;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010037import com.google.devtools.build.lib.util.Pair;
Mark Schaller6df81792015-12-10 18:47:47 +000038import com.google.devtools.build.lib.util.Preconditions;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010039import com.google.devtools.build.lib.vfs.Canonicalizer;
40import com.google.devtools.build.lib.vfs.Path;
41import com.google.devtools.build.lib.vfs.PathFragment;
42
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010043import java.io.PrintStream;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010044import java.util.ArrayList;
45import java.util.Collection;
46import java.util.Collections;
47import java.util.HashMap;
48import java.util.List;
49import java.util.Map;
50import java.util.Set;
51
Han-Wen Nienhuys67e6f982016-01-15 15:54:12 +000052import javax.annotation.Nullable;
53
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010054/**
55 * A package, which is a container of {@link Rule}s, each of
56 * which contains a dictionary of named attributes.
57 *
58 * <p>Package instances are intended to be immutable and for all practical
59 * purposes can be treated as such. Note, however, that some member variables
60 * exposed via the public interface are not strictly immutable, so until their
61 * types are guaranteed immutable we're not applying the {@code @Immutable}
62 * annotation here.
63 */
Michajlo Matijkiw17f11eb2015-10-14 04:19:28 +000064public class Package {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010065
66 /**
67 * Common superclass for all name-conflict exceptions.
68 */
69 public static class NameConflictException extends Exception {
70 protected NameConflictException(String message) {
71 super(message);
72 }
73 }
74
75 /**
76 * The repository identifier for this package.
77 */
78 private final PackageIdentifier packageIdentifier;
79
80 /**
81 * The name of the package, e.g. "foo/bar".
82 */
83 protected final String name;
84
85 /**
86 * Like name, but in the form of a PathFragment.
87 */
88 private final PathFragment nameFragment;
89
90 /**
91 * The filename of this package's BUILD file.
92 */
93 protected Path filename;
94
95 /**
96 * The directory in which this package's BUILD file resides. All InputFile
97 * members of the packages are located relative to this directory.
98 */
99 private Path packageDirectory;
100
101 /**
Kristina Chodorow91876f02015-02-27 17:14:12 +0000102 * The name of the workspace this package is in. Used as a prefix for the runfiles directory.
103 * This can be set in the WORKSPACE file. This must be a valid target name.
104 */
Ulf Adamsd13207c2015-09-04 14:53:43 +0000105 protected String workspaceName;
Kristina Chodorow91876f02015-02-27 17:14:12 +0000106
107 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100108 * The root of the source tree in which this package was found. It is an invariant that
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000109 * {@code sourceRoot.getRelative(packageId.getPathFragment()).equals(packageDirectory)}.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100110 */
111 private Path sourceRoot;
112
113 /**
114 * The "Make" environment of this package, containing package-local
115 * definitions of "Make" variables.
116 */
117 private MakeEnvironment makeEnv;
118
119 /**
120 * The collection of all targets defined in this package, indexed by name.
121 */
122 protected Map<String, Target> targets;
123
124 /**
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000125 * Default visibility for rules that do not specify it.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100126 */
127 private RuleVisibility defaultVisibility;
128 private boolean defaultVisibilitySet;
129
130 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100131 * Default package-level 'testonly' value for rules that do not specify it.
132 */
133 private boolean defaultTestOnly = false;
134
135 /**
136 * Default package-level 'deprecation' value for rules that do not specify it.
137 */
138 private String defaultDeprecation;
139
140 /**
141 * Default header strictness checking for rules that do not specify it.
142 */
143 private String defaultHdrsCheck;
144
145 /**
146 * Default copts for cc_* rules. The rules' individual copts will append to
147 * this value.
148 */
149 private ImmutableList<String> defaultCopts;
150
151 /**
152 * The InputFile target corresponding to this package's BUILD file.
153 */
154 private InputFile buildFile;
155
156 /**
157 * True iff this package's BUILD files contained lexical or grammatical
158 * errors, or experienced errors during evaluation, or semantic errors during
159 * the construction of any rule.
160 *
161 * <p>Note: A package containing errors does not necessarily prevent a build;
162 * if all the rules needed for a given build were constructed prior to the
163 * first error, the build may proceed.
164 */
165 private boolean containsErrors;
166
167 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100168 * The set of labels subincluded by this package.
169 */
170 private Set<Label> subincludes;
171
172 /**
173 * The list of transitive closure of the Skylark file dependencies.
174 */
175 private ImmutableList<Label> skylarkFileDependencies;
176
177 /**
178 * The package's default "licenses" and "distribs" attributes, as specified
179 * in calls to licenses() and distribs() in the BUILD file.
180 */
181 // These sets contain the values specified by the most recent licenses() or
182 // distribs() declarations encountered during package parsing:
183 private License defaultLicense;
184 private Set<License.DistributionType> defaultDistributionSet;
185
186
187 /**
188 * The names of the package() attributes that declare default values for rule
189 * {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR} and {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
190 * values when not explicitly specified.
191 */
192 public static final String DEFAULT_COMPATIBLE_WITH_ATTRIBUTE = "default_compatible_with";
193 public static final String DEFAULT_RESTRICTED_TO_ATTRIBUTE = "default_restricted_to";
194
195 private Set<Label> defaultCompatibleWith = ImmutableSet.of();
196 private Set<Label> defaultRestrictedTo = ImmutableSet.of();
197
198 private ImmutableSet<String> features;
199
200 private ImmutableList<Event> events;
201
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100202 /**
203 * Package initialization, part 1 of 3: instantiates a new package with the
204 * given name.
205 *
206 * <p>As part of initialization, {@link Builder} constructs {@link InputFile}
207 * and {@link PackageGroup} instances that require a valid Package instance where
208 * {@link Package#getNameFragment()} is accessible. That's why these settings are
209 * applied here at the start.
210 *
211 * @precondition {@code name} must be a suffix of
212 * {@code filename.getParentDirectory())}.
213 */
Michajlo Matijkiw17f11eb2015-10-14 04:19:28 +0000214 private Package(PackageIdentifier packageId, String runfilesPrefix) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100215 this.packageIdentifier = packageId;
Ulf Adamsd13207c2015-09-04 14:53:43 +0000216 this.workspaceName = runfilesPrefix;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100217 this.nameFragment = Canonicalizer.fragments().intern(packageId.getPackageFragment());
218 this.name = nameFragment.getPathString();
219 }
220
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100221 /** Returns this packages' identifier. */
222 public PackageIdentifier getPackageIdentifier() {
223 return packageIdentifier;
224 }
225
226 /**
227 * Package initialization: part 2 of 3: sets this package's default header
228 * strictness checking.
229 *
230 * <p>This is needed to support C++-related rule classes
231 * which accesses {@link #getDefaultHdrsCheck} from the still-under-construction
232 * package.
233 */
234 protected void setDefaultHdrsCheck(String defaultHdrsCheck) {
235 this.defaultHdrsCheck = defaultHdrsCheck;
236 }
237
238 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100239 * Set the default 'testonly' value for this package.
240 */
241 protected void setDefaultTestOnly(boolean testOnly) {
242 defaultTestOnly = testOnly;
243 }
244
245 /**
246 * Set the default 'deprecation' value for this package.
247 */
248 protected void setDefaultDeprecation(String deprecation) {
249 defaultDeprecation = deprecation;
250 }
251
252 /**
253 * Sets the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
254 * attribute when not explicitly specified by the rule.
255 */
256 protected void setDefaultCompatibleWith(Set<Label> environments) {
257 defaultCompatibleWith = environments;
258 }
259
260 /**
261 * Sets the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
262 * attribute when not explicitly specified by the rule.
263 */
264 protected void setDefaultRestrictedTo(Set<Label> environments) {
265 defaultRestrictedTo = environments;
266 }
267
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000268 private static Path getSourceRoot(Path buildFile, PathFragment packageFragment) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100269 Path current = buildFile.getParentDirectory();
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000270 for (int i = 0, len = packageFragment.segmentCount(); i < len && current != null; i++) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100271 current = current.getParentDirectory();
272 }
273 return current;
274 }
275
276 /**
277 * Package initialization: part 3 of 3: applies all other settings and completes
278 * initialization of the package.
279 *
280 * <p>Only after this method is called can this package be considered "complete"
281 * and be shared publicly.
282 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000283 protected void finishInit(Builder builder) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100284 // If any error occurred during evaluation of this package, consider all
285 // rules in the package to be "in error" also (even if they were evaluated
286 // prior to the error). This behaviour is arguably stricter than need be,
287 // but stopping a build only for some errors but not others creates user
288 // confusion.
289 if (builder.containsErrors) {
290 for (Rule rule : builder.getTargets(Rule.class)) {
291 rule.setContainsErrors();
292 }
293 }
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000294 this.filename = builder.getFilename();
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100295 this.packageDirectory = filename.getParentDirectory();
296
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000297 this.sourceRoot = getSourceRoot(filename, packageIdentifier.getPathFragment());
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100298 if ((sourceRoot == null
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000299 || !sourceRoot.getRelative(packageIdentifier.getPathFragment()).equals(packageDirectory))
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100300 && !filename.getBaseName().equals("WORKSPACE")) {
301 throw new IllegalArgumentException(
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000302 "Invalid BUILD file name for package '" + packageIdentifier + "': " + filename);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100303 }
304
305 this.makeEnv = builder.makeEnv.build();
306 this.targets = ImmutableSortedKeyMap.copyOf(builder.targets);
307 this.defaultVisibility = builder.defaultVisibility;
308 this.defaultVisibilitySet = builder.defaultVisibilitySet;
309 if (builder.defaultCopts == null) {
310 this.defaultCopts = ImmutableList.of();
311 } else {
312 this.defaultCopts = ImmutableList.copyOf(builder.defaultCopts);
313 }
314 this.buildFile = builder.buildFile;
315 this.containsErrors = builder.containsErrors;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100316 this.subincludes = builder.subincludes.keySet();
317 this.skylarkFileDependencies = builder.skylarkFileDependencies;
318 this.defaultLicense = builder.defaultLicense;
319 this.defaultDistributionSet = builder.defaultDistributionSet;
320 this.features = ImmutableSortedSet.copyOf(builder.features);
321 this.events = ImmutableList.copyOf(builder.events);
322 }
323
324 /**
325 * Returns the list of subincluded labels on which the validity of this package depends.
326 */
327 public Set<Label> getSubincludeLabels() {
328 return subincludes;
329 }
330
331 /**
332 * Returns the list of transitive closure of the Skylark file dependencies of this package.
333 */
334 public ImmutableList<Label> getSkylarkFileDependencies() {
335 return skylarkFileDependencies;
336 }
337
338 /**
339 * Returns the filename of the BUILD file which defines this package. The
340 * parent directory of the BUILD file is the package directory.
341 */
342 public Path getFilename() {
343 return filename;
344 }
345
346 /**
347 * Returns the source root (a directory) beneath which this package's BUILD file was found.
348 *
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000349 * <p> Assumes invariant:
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000350 * {@code getSourceRoot().getRelative(packageId.getPathFragment()).equals(getPackageDirectory())}
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100351 */
352 public Path getSourceRoot() {
353 return sourceRoot;
354 }
355
356 /**
357 * Returns the directory containing the package's BUILD file.
358 */
359 public Path getPackageDirectory() {
360 return packageDirectory;
361 }
362
363 /**
364 * Returns the name of this package. If this build is using external repositories then this name
365 * may not be unique!
366 */
367 public String getName() {
368 return name;
369 }
370
371 /**
372 * Like {@link #getName}, but has type {@code PathFragment}.
373 */
374 public PathFragment getNameFragment() {
375 return nameFragment;
376 }
377
378 /**
379 * Returns the "Make" value from the package's make environment whose name
380 * is "varname", or null iff the variable is not defined in the environment.
381 */
382 public String lookupMakeVariable(String varname, String platform) {
383 return makeEnv.lookup(varname, platform);
384 }
385
386 /**
387 * Returns the make environment. This should only ever be used for serialization -- how the
388 * make variables are implemented is an implementation detail.
389 */
390 MakeEnvironment getMakeEnvironment() {
391 return makeEnv;
392 }
393
394 /**
Laurent Le Brune8ce8472015-03-12 14:20:01 +0000395 * Returns all make variables for a given platform.
396 */
397 public Map<String, String> getAllMakeVariables(String platform) {
398 ImmutableMap.Builder<String, String> map = ImmutableMap.builder();
399 for (String var : makeEnv.getBindings().keySet()) {
400 map.put(var, makeEnv.lookup(var, platform));
401 }
402 return map.build();
403 }
404
405 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100406 * Returns the label of this package's BUILD file.
407 *
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000408 * <p> Typically <code>getBuildFileLabel().getName().equals("BUILD")</code> --
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100409 * though not necessarily: data in a subdirectory of a test package may use a
410 * different filename to avoid inadvertently creating a new package.
411 */
Mark Schaller4fa83ac2015-07-10 16:59:37 +0000412 public Label getBuildFileLabel() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100413 return buildFile.getLabel();
414 }
415
416 /**
417 * Returns the InputFile target for this package's BUILD file.
418 */
419 public InputFile getBuildFile() {
420 return buildFile;
421 }
422
423 /**
424 * Returns true if errors were encountered during evaluation of this package.
425 * (The package may be incomplete and its contents should not be relied upon
426 * for critical operations. However, any Rules belonging to the package are
427 * guaranteed to be intact, unless their <code>containsErrors()</code> flag
428 * is set.)
429 */
430 public boolean containsErrors() {
431 return containsErrors;
432 }
433
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100434 public List<Event> getEvents() {
435 return events;
436 }
437
438 /**
439 * Returns an (immutable, unordered) view of all the targets belonging to this package.
440 */
441 public Collection<Target> getTargets() {
442 return getTargets(targets);
443 }
444
445 /**
446 * Common getTargets implementation, accessible by both {@link Package} and
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000447 * {@link Package.Builder}.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100448 */
449 private static Collection<Target> getTargets(Map<String, Target> targetMap) {
450 return Collections.unmodifiableCollection(targetMap.values());
451 }
452
453 /**
454 * Returns a (read-only, unordered) iterator of all the targets belonging
455 * to this package which are instances of the specified class.
456 */
457 public <T extends Target> Iterable<T> getTargets(Class<T> targetClass) {
458 return getTargets(targets, targetClass);
459 }
460
461 /**
462 * Common getTargets implementation, accessible by both {@link Package} and
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000463 * {@link Package.Builder}.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100464 */
465 private static <T extends Target> Iterable<T> getTargets(Map<String, Target> targetMap,
466 Class<T> targetClass) {
467 return Iterables.filter(targetMap.values(), targetClass);
468 }
469
470 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100471 * Returns the rule that corresponds to a particular BUILD target name. Useful
472 * for walking through the dependency graph of a target.
473 * Fails if the target is not a Rule.
474 */
Ulf Adamsb313a972015-05-04 12:05:57 +0000475 @VisibleForTesting // Should be package-private
476 public Rule getRule(String targetName) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100477 return (Rule) targets.get(targetName);
478 }
479
480 /**
Kristina Chodorow91876f02015-02-27 17:14:12 +0000481 * Returns this package's workspace name.
482 *
483 * <p>Package-private to encourage callers to get their workspace name from a rule, not a
484 * package.</p>
485 */
486 String getWorkspaceName() {
487 return workspaceName;
488 }
489
490 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100491 * Returns the features specified in the <code>package()</code> declaration.
492 */
493 public ImmutableSet<String> getFeatures() {
494 return features;
495 }
496
497 /**
498 * Returns the target (a member of this package) whose name is "targetName".
499 * First rules are searched, then output files, then input files. The target
500 * name must be valid, as defined by {@code LabelValidator#validateTargetName}.
501 *
502 * @throws NoSuchTargetException if the specified target was not found.
503 */
504 public Target getTarget(String targetName) throws NoSuchTargetException {
505 Target target = targets.get(targetName);
506 if (target != null) {
507 return target;
508 }
509
510 // No such target.
511
512 // If there's a file on the disk that's not mentioned in the BUILD file,
513 // produce a more informative error. NOTE! this code path is only executed
514 // on failure, which is (relatively) very rare. In the common case no
515 // stat(2) is executed.
516 Path filename = getPackageDirectory().getRelative(targetName);
517 String suffix;
518 if (!new PathFragment(targetName).isNormalized()) {
519 // Don't check for file existence in this case because the error message
520 // would be confusing and wrong. If the targetName is "foo/bar/.", and
521 // there is a directory "foo/bar", it doesn't mean that "//pkg:foo/bar/."
522 // is a valid label.
523 suffix = "";
524 } else if (filename.isDirectory()) {
525 suffix = "; however, a source directory of this name exists. (Perhaps add "
526 + "'exports_files([\"" + targetName + "\"])' to " + name + "/BUILD, or define a "
527 + "filegroup?)";
528 } else if (filename.exists()) {
529 suffix = "; however, a source file of this name exists. (Perhaps add "
530 + "'exports_files([\"" + targetName + "\"])' to " + name + "/BUILD?)";
531 } else {
532 suffix = "";
533 }
534
535 try {
536 throw new NoSuchTargetException(createLabel(targetName), "target '" + targetName
537 + "' not declared in package '" + name + "'" + suffix + " defined by "
538 + this.filename);
Lukacs Berkia6434362015-09-15 13:56:14 +0000539 } catch (LabelSyntaxException e) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100540 throw new IllegalArgumentException(targetName);
541 }
542 }
543
544 /**
545 * Creates a label for a target inside this package.
546 *
Lukacs Berkia6434362015-09-15 13:56:14 +0000547 * @throws LabelSyntaxException if the {@code targetName} is invalid
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100548 */
Lukacs Berkia6434362015-09-15 13:56:14 +0000549 public Label createLabel(String targetName) throws LabelSyntaxException {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100550 return Label.create(packageIdentifier, targetName);
551 }
552
553 /**
554 * Returns the default visibility for this package.
555 */
556 public RuleVisibility getDefaultVisibility() {
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000557 return defaultVisibility;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100558 }
559
560 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100561 * Returns the default testonly value.
562 */
563 public Boolean getDefaultTestOnly() {
564 return defaultTestOnly;
565 }
566
567 /**
Greg Estrenc48214d2015-03-17 22:43:19 +0000568 * Returns the default deprecation value.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100569 */
570 public String getDefaultDeprecation() {
571 return defaultDeprecation;
572 }
573
574 /**
575 * Gets the default header checking mode.
576 */
577 public String getDefaultHdrsCheck() {
Googler198a54f2016-01-18 11:02:10 +0000578 return defaultHdrsCheck != null ? defaultHdrsCheck : "strict";
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100579 }
580
581 /**
582 * Returns the default copts value, to which rules should append their
583 * specific copts.
584 */
585 public ImmutableList<String> getDefaultCopts() {
586 return defaultCopts;
587 }
588
589 /**
590 * Returns whether the default header checking mode has been set or it is the
591 * default value.
592 */
593 public boolean isDefaultHdrsCheckSet() {
594 return defaultHdrsCheck != null;
595 }
596
597 public boolean isDefaultVisibilitySet() {
598 return defaultVisibilitySet;
599 }
600
601 /**
602 * Gets the parsed license object for the default license
603 * declared by this package.
604 */
605 public License getDefaultLicense() {
606 return defaultLicense;
607 }
608
609 /**
610 * Returns the parsed set of distributions declared as the default for this
611 * package.
612 */
613 public Set<License.DistributionType> getDefaultDistribs() {
614 return defaultDistributionSet;
615 }
616
617 /**
618 * Returns the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
619 * attribute when not explicitly specified by the rule.
620 */
621 public Set<Label> getDefaultCompatibleWith() {
622 return defaultCompatibleWith;
623 }
624
625 /**
626 * Returns the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
627 * attribute when not explicitly specified by the rule.
628 */
629 public Set<Label> getDefaultRestrictedTo() {
630 return defaultRestrictedTo;
631 }
632
633 @Override
634 public String toString() {
Han-Wen Nienhuys58070da2015-04-20 11:37:58 +0000635 return "Package(" + name + ")="
636 + (targets != null ? getTargets(Rule.class) : "initializing...");
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100637 }
638
639 /**
640 * Dumps the package for debugging. Do not depend on the exact format/contents of this debugging
641 * output.
642 */
643 public void dump(PrintStream out) {
644 out.println(" Package " + getName() + " (" + getFilename() + ")");
645
646 // Rules:
647 out.println(" Rules");
648 for (Rule rule : getTargets(Rule.class)) {
649 out.println(" " + rule.getTargetKind() + " " + rule.getLabel());
650 for (Attribute attr : rule.getAttributes()) {
651 for (Object possibleValue : AggregatingAttributeMapper.of(rule)
652 .visitAttribute(attr.getName(), attr.getType())) {
653 out.println(" " + attr.getName() + " = " + possibleValue);
654 }
655 }
656 }
657
658 // Files:
659 out.println(" Files");
660 for (FileTarget file : getTargets(FileTarget.class)) {
661 out.print(" " + file.getTargetKind() + " " + file.getLabel());
662 if (file instanceof OutputFile) {
663 out.println(" (generated by " + ((OutputFile) file).getGeneratingRule().getLabel() + ")");
664 } else {
665 out.println();
666 }
667 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100668 }
669
670 /**
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000671 * Builder class for {@link Package} that does its own globbing.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100672 *
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000673 * <p>Despite its name, this is the normal builder used when parsing BUILD files.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100674 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000675 public static class LegacyBuilder extends Builder {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100676 private Globber globber = null;
677
Ulf Adamsd13207c2015-09-04 14:53:43 +0000678 LegacyBuilder(PackageIdentifier packageId, String runfilesPrefix) {
679 super(packageId, runfilesPrefix);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100680 }
681
682 /**
Damien Martin-Guillerez8ca065c22015-11-27 16:22:59 +0000683 * Derive a LegacyBuilder from a normal Builder.
684 */
685 LegacyBuilder(Builder builder) {
686 super(builder.pkg);
687 if (builder.getFilename() != null) {
688 setFilename(builder.getFilename());
689 }
690 }
691
692 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100693 * Sets the globber used for this package's glob expansions.
694 */
695 LegacyBuilder setGlobber(Globber globber) {
696 this.globber = globber;
697 return this;
698 }
699
700 /**
701 * Removes a target from the {@link Package} under construction. Intended to be used only by
Michajlo Matijkiw91407182015-05-29 14:27:09 +0000702 * {@link com.google.devtools.build.lib.skyframe.PackageFunction} to remove targets whose
703 * labels cross subpackage boundaries.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100704 */
705 public void removeTarget(Target target) {
706 if (target.getPackage() == pkg) {
707 this.targets.remove(target.getName());
708 }
709 }
710
711 /**
712 * Returns the glob patterns requested by {@link PackageFactory} during evaluation of this
Michajlo Matijkiw91407182015-05-29 14:27:09 +0000713 * package's BUILD file. Intended to be used only by
714 * {@link com.google.devtools.build.lib.skyframe.PackageFunction} to mark the appropriate
715 * Skyframe dependencies after the fact.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100716 */
717 public Set<Pair<String, Boolean>> getGlobPatterns() {
718 return globber.getGlobPatterns();
719 }
720 }
721
Damien Martin-Guillerez8ca065c22015-11-27 16:22:59 +0000722 public static LegacyBuilder newExternalPackageBuilder(Path workspacePath, String runfilesPrefix) {
Lukacs Berkie19ee272015-12-10 11:34:29 +0000723 LegacyBuilder b = new LegacyBuilder(Label.EXTERNAL_PACKAGE_IDENTIFIER, runfilesPrefix);
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +0000724 b.setFilename(workspacePath);
725 b.setMakeEnv(new MakeEnvironment.Builder());
726 return b;
727 }
728
729 public static class Builder {
Ulf Adamsd13207c2015-09-04 14:53:43 +0000730 protected static Package newPackage(PackageIdentifier packageId, String runfilesPrefix) {
731 return new Package(packageId, runfilesPrefix);
Ulf Adams4226be22015-09-03 17:00:54 +0000732 }
733
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100734 /**
735 * The output instance for this builder. Needs to be instantiated and
736 * available with name info throughout initialization. All other settings
Michajlo Matijkiw17f11eb2015-10-14 04:19:28 +0000737 * are applied during {@link #build}. See {@link Package#Package}
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100738 * and {@link Package#finishInit} for details.
739 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000740 protected Package pkg;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100741
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000742 private Path filename = null;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100743 private Label buildFileLabel = null;
744 private InputFile buildFile = null;
745 private MakeEnvironment.Builder makeEnv = null;
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000746 private RuleVisibility defaultVisibility = ConstantRuleVisibility.PRIVATE;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100747 private boolean defaultVisibilitySet;
748 private List<String> defaultCopts = null;
749 private List<String> features = new ArrayList<>();
750 private List<Event> events = Lists.newArrayList();
751 private boolean containsErrors = false;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100752
753 private License defaultLicense = License.NO_LICENSE;
754 private Set<License.DistributionType> defaultDistributionSet = License.DEFAULT_DISTRIB;
755
756 protected Map<String, Target> targets = new HashMap<>();
757 protected Map<Label, EnvironmentGroup> environmentGroups = new HashMap<>();
758
759 protected Map<Label, Path> subincludes = null;
Kristina Chodorow91876f02015-02-27 17:14:12 +0000760 protected ImmutableList<Label> skylarkFileDependencies = ImmutableList.of();
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100761
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +0000762 protected ExternalPackageBuilder externalPackageData = new ExternalPackageBuilder();
763
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100764 /**
765 * True iff the "package" function has already been called in this package.
766 */
767 private boolean packageFunctionUsed;
768
769 /**
770 * The collection of the prefixes of every output file. Maps every prefix
771 * to an output file whose prefix it is.
772 *
773 * <p>This is needed to make the output file prefix conflict check be
774 * reasonably fast. However, since it can potentially take a lot of memory and
775 * is useless after the package has been loaded, it isn't passed to the
776 * package itself.
777 */
778 private Map<String, OutputFile> outputFilePrefixes = new HashMap<>();
779
780 private boolean alreadyBuilt = false;
781
782 private EventHandler builderEventHandler = new EventHandler() {
783 @Override
784 public void handle(Event event) {
785 addEvent(event);
786 }
787 };
788
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000789 protected Builder(Package pkg) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100790 this.pkg = pkg;
791 if (pkg.getName().startsWith("javatests/")) {
792 setDefaultTestonly(true);
793 }
794 }
795
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +0000796 public Builder(PackageIdentifier id, String runfilesPrefix) {
Ulf Adamsd13207c2015-09-04 14:53:43 +0000797 this(newPackage(id, runfilesPrefix));
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100798 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100799
800 protected PackageIdentifier getPackageIdentifier() {
801 return pkg.getPackageIdentifier();
802 }
803
Damien Martin-Guillerez8ca065c22015-11-27 16:22:59 +0000804 /** Determine if we are in the WORKSPACE file or not */
805 public boolean isWorkspace() {
Lukacs Berkie19ee272015-12-10 11:34:29 +0000806 return pkg.getPackageIdentifier().equals(Label.EXTERNAL_PACKAGE_IDENTIFIER);
Damien Martin-Guillerez8ca065c22015-11-27 16:22:59 +0000807 }
808
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100809 /**
810 * Sets the name of this package's BUILD file.
811 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000812 Builder setFilename(Path filename) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100813 this.filename = filename;
814 try {
815 buildFileLabel = createLabel(filename.getBaseName());
816 addInputFile(buildFileLabel, Location.fromFile(filename));
Lukacs Berkia6434362015-09-15 13:56:14 +0000817 } catch (LabelSyntaxException e) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100818 // This can't actually happen.
819 throw new AssertionError("Package BUILD file has an illegal name: " + filename);
820 }
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000821 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100822 }
823
824 public Label getBuildFileLabel() {
825 return buildFileLabel;
826 }
827
828 Path getFilename() {
829 return filename;
830 }
831
Janak Ramakrishnan0a4c6e42015-09-17 00:37:58 +0000832 public List<Event> getEvents() {
833 return events;
834 }
835
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100836 /**
837 * Sets this package's Make environment.
838 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000839 Builder setMakeEnv(MakeEnvironment.Builder makeEnv) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100840 this.makeEnv = makeEnv;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000841 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100842 }
843
Googler94296442015-03-25 09:47:41 +0000844 MakeEnvironment.Builder getMakeEnvironment() {
845 return makeEnv;
846 }
847
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100848 /**
849 * Sets the default visibility for this package. Called at most once per
850 * package from PackageFactory.
851 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000852 Builder setDefaultVisibility(RuleVisibility visibility) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100853 this.defaultVisibility = visibility;
854 this.defaultVisibilitySet = true;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000855 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100856 }
857
858 /**
859 * Sets whether the default visibility is set in the BUILD file.
860 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000861 Builder setDefaultVisibilitySet(boolean defaultVisibilitySet) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100862 this.defaultVisibilitySet = defaultVisibilitySet;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000863 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100864 }
865
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100866 /** Sets the default value of 'testonly'. Rule-level 'testonly' will override this. */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000867 Builder setDefaultTestonly(boolean defaultTestonly) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100868 pkg.setDefaultTestOnly(defaultTestonly);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000869 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100870 }
871
872 /**
873 * Sets the default value of 'deprecation'. Rule-level 'deprecation' will append to this.
874 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000875 Builder setDefaultDeprecation(String defaultDeprecation) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100876 pkg.setDefaultDeprecation(defaultDeprecation);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000877 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100878 }
879
880 /**
Kristina Chodorow91876f02015-02-27 17:14:12 +0000881 * Uses the workspace name from {@code //external} to set this package's workspace name.
882 */
Han-Wen Nienhuys2cffe032015-09-21 15:28:40 +0000883 @VisibleForTesting
884 public Builder setWorkspaceName(String workspaceName) {
Kristina Chodorow91876f02015-02-27 17:14:12 +0000885 pkg.workspaceName = workspaceName;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000886 return this;
Kristina Chodorow91876f02015-02-27 17:14:12 +0000887 }
888
889 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100890 * Returns whether the "package" function has been called yet
891 */
892 public boolean isPackageFunctionUsed() {
893 return packageFunctionUsed;
894 }
895
896 public void setPackageFunctionUsed() {
897 packageFunctionUsed = true;
898 }
899
900 /**
901 * Sets the default header checking mode.
902 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000903 public Builder setDefaultHdrsCheck(String hdrsCheck) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100904 // Note that this setting is propagated directly to the package because
905 // other code needs the ability to read this info directly from the
906 // under-construction package. See {@link Package#setDefaultHdrsCheck}.
907 pkg.setDefaultHdrsCheck(hdrsCheck);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000908 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100909 }
910
911 /**
912 * Sets the default value of copts. Rule-level copts will append to this.
913 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000914 public Builder setDefaultCopts(List<String> defaultCopts) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100915 this.defaultCopts = defaultCopts;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000916 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100917 }
918
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000919 public Builder addFeatures(Iterable<String> features) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100920 Iterables.addAll(this.features, features);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000921 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100922 }
923
924 /**
925 * Declares that errors were encountering while loading this package.
926 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000927 public Builder setContainsErrors() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100928 containsErrors = true;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000929 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100930 }
931
932 public boolean containsErrors() {
933 return containsErrors;
934 }
935
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000936 public Builder addEvents(Iterable<Event> events) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100937 for (Event event : events) {
938 addEvent(event);
939 }
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000940 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100941 }
942
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000943 public Builder addEvent(Event event) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100944 this.events.add(event);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000945 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100946 }
947
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000948 Builder setSkylarkFileDependencies(ImmutableList<Label> skylarkFileDependencies) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100949 this.skylarkFileDependencies = skylarkFileDependencies;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000950 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100951 }
952
953 /**
954 * Sets the default license for this package.
955 */
956 void setDefaultLicense(License license) {
957 this.defaultLicense = license;
958 }
959
960 License getDefaultLicense() {
961 return defaultLicense;
962 }
963
964 /**
965 * Initializes the default set of distributions for targets in this package.
966 *
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000967 * <p> TODO(bazel-team): (2011) consider moving the license & distribs info into Metadata--maybe
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100968 * even in the Build language.
969 */
970 void setDefaultDistribs(Set<DistributionType> dists) {
971 this.defaultDistributionSet = dists;
972 }
973
974 Set<DistributionType> getDefaultDistribs() {
975 return defaultDistributionSet;
976 }
977
978 /**
979 * Sets the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
980 * attribute when not explicitly specified by the rule. Records a package error if
981 * any labels are duplicated.
982 */
983 void setDefaultCompatibleWith(List<Label> environments, String attrName, Location location) {
984 if (!checkForDuplicateLabels(environments, "package " + pkg.getName(), attrName, location,
985 builderEventHandler)) {
986 setContainsErrors();
987 }
988 pkg.setDefaultCompatibleWith(ImmutableSet.copyOf(environments));
989 }
990
991 /**
992 * Sets the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
993 * attribute when not explicitly specified by the rule. Records a package error if
994 * any labels are duplicated.
995 */
996 void setDefaultRestrictedTo(List<Label> environments, String attrName, Location location) {
997 if (!checkForDuplicateLabels(environments, "package " + pkg.getName(), attrName, location,
998 builderEventHandler)) {
999 setContainsErrors();
1000 }
1001
1002 pkg.setDefaultRestrictedTo(ImmutableSet.copyOf(environments));
1003 }
1004
1005 /**
Mark Schalleree624452016-01-13 18:41:24 +00001006 * Creates a new {@link Rule} {@code r} where {@code r.getPackage()} is the {@link Package}
1007 * associated with this {@link Builder}.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001008 *
Mark Schalleree624452016-01-13 18:41:24 +00001009 * <p>The created {@link Rule} will have no attribute values, no output files, and therefore
1010 * will be in an invalid state.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001011 */
Mark Schalleree624452016-01-13 18:41:24 +00001012 Rule createRule(
1013 Label label,
1014 RuleClass ruleClass,
1015 Location location,
Michajlo Matijkiw1a759072015-10-22 13:52:00 +00001016 AttributeContainer attributeContainer) {
1017 return new Rule(pkg, label, ruleClass, location, attributeContainer);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001018 }
1019
1020 /**
1021 * Called by the parser when a "mocksubinclude" is encountered, to record the
1022 * mappings from labels to absolute paths upon which that the validity of
1023 * this package depends.
1024 */
1025 void addSubinclude(Label label, Path resolvedPath) {
1026 if (subincludes == null) {
1027 // This is a TreeMap because the order needs to be deterministic.
1028 subincludes = Maps.newTreeMap();
1029 }
1030
1031 Path oldResolvedPath = subincludes.put(label, resolvedPath);
1032 if (oldResolvedPath != null && !oldResolvedPath.equals(resolvedPath)){
1033 // The same label should have been resolved to the same path
1034 throw new IllegalStateException("Ambiguous subinclude path");
1035 }
1036 }
1037
1038 public Set<Label> getSubincludeLabels() {
1039 return subincludes == null ? Sets.<Label>newHashSet() : subincludes.keySet();
1040 }
1041
1042 public Map<Label, Path> getSubincludes() {
1043 return subincludes == null ? Maps.<Label, Path>newHashMap() : subincludes;
1044 }
1045
1046 public Collection<Target> getTargets() {
1047 return Package.getTargets(targets);
1048 }
1049
Han-Wen Nienhuys67e6f982016-01-15 15:54:12 +00001050 @Nullable
1051 public Target getTarget(String name) {
1052 return targets.get(name);
1053 }
1054
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001055 /**
1056 * Returns an (immutable, unordered) view of all the targets belonging to
1057 * this package which are instances of the specified class.
1058 */
1059 <T extends Target> Iterable<T> getTargets(Class<T> targetClass) {
1060 return Package.getTargets(targets, targetClass);
1061 }
1062
1063 /**
1064 * An input file name conflicts with an existing package member.
1065 */
1066 static class GeneratedLabelConflict extends NameConflictException {
1067 private GeneratedLabelConflict(String message) {
1068 super(message);
1069 }
1070 }
1071
1072 /**
1073 * Creates an input file target in this package with the specified name.
1074 *
1075 * @param targetName name of the input file. This must be a valid target
1076 * name as defined by {@link
1077 * com.google.devtools.build.lib.cmdline.LabelValidator#validateTargetName}.
1078 * @return the newly-created InputFile, or the old one if it already existed.
1079 * @throws GeneratedLabelConflict if the name was already taken by a Rule or
1080 * an OutputFile target.
1081 * @throws IllegalArgumentException if the name is not a valid label
1082 */
1083 InputFile createInputFile(String targetName, Location location)
1084 throws GeneratedLabelConflict {
1085 Target existing = targets.get(targetName);
1086 if (existing == null) {
1087 try {
1088 return addInputFile(createLabel(targetName), location);
Lukacs Berkia6434362015-09-15 13:56:14 +00001089 } catch (LabelSyntaxException e) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001090 throw new IllegalArgumentException("FileTarget in package " + pkg.getName()
1091 + " has illegal name: " + targetName);
1092 }
1093 } else if (existing instanceof InputFile) {
1094 return (InputFile) existing; // idempotent
1095 } else {
1096 throw new GeneratedLabelConflict("generated label '//" + pkg.getName() + ":"
1097 + targetName + "' conflicts with existing "
1098 + existing.getTargetKind());
1099 }
1100 }
1101
1102 /**
1103 * Sets the visibility and license for an input file. The input file must already exist as
1104 * a member of this package.
1105 * @throws IllegalArgumentException if the input file doesn't exist in this
1106 * package's target map.
1107 */
1108 void setVisibilityAndLicense(InputFile inputFile, RuleVisibility visibility, License license) {
1109 String filename = inputFile.getName();
1110 Target cacheInstance = targets.get(filename);
Ulf Adams07dba942015-03-05 14:47:37 +00001111 if (!(cacheInstance instanceof InputFile)) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001112 throw new IllegalArgumentException("Can't set visibility for nonexistent FileTarget "
1113 + filename + " in package " + pkg.getName() + ".");
1114 }
1115 if (!((InputFile) cacheInstance).isVisibilitySpecified()
1116 || cacheInstance.getVisibility() != visibility
1117 || cacheInstance.getLicense() != license) {
1118 targets.put(filename, new InputFile(
1119 pkg, cacheInstance.getLabel(), cacheInstance.getLocation(), visibility, license));
1120 }
1121 }
1122
1123 /**
1124 * Creates a label for a target inside this package.
1125 *
Lukacs Berkia6434362015-09-15 13:56:14 +00001126 * @throws LabelSyntaxException if the {@code targetName} is invalid
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001127 */
Lukacs Berkia6434362015-09-15 13:56:14 +00001128 Label createLabel(String targetName) throws LabelSyntaxException {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001129 return Label.create(pkg.getPackageIdentifier(), targetName);
1130 }
1131
1132 /**
1133 * Adds a package group to the package.
1134 */
1135 void addPackageGroup(String name, Collection<String> packages, Collection<Label> includes,
1136 EventHandler eventHandler, Location location)
Lukacs Berkia6434362015-09-15 13:56:14 +00001137 throws NameConflictException, LabelSyntaxException {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001138 PackageGroup group =
1139 new PackageGroup(createLabel(name), pkg, packages, includes, eventHandler, location);
1140 Target existing = targets.get(group.getName());
1141 if (existing != null) {
1142 throw nameConflict(group, existing);
1143 }
1144
1145 targets.put(group.getName(), group);
1146
1147 if (group.containsErrors()) {
1148 setContainsErrors();
1149 }
1150 }
1151
1152 /**
1153 * Checks if any labels in the given list appear multiple times and reports an appropriate
1154 * error message if so. Returns true if no duplicates were found, false otherwise.
1155 *
Laurent Le Brun41079bd2015-06-19 09:37:25 +00001156 * <p> TODO(bazel-team): apply this to all build functions (maybe automatically?), possibly
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001157 * integrate with RuleClass.checkForDuplicateLabels.
1158 */
1159 private static boolean checkForDuplicateLabels(Collection<Label> labels, String owner,
1160 String attrName, Location location, EventHandler eventHandler) {
1161 Set<Label> dupes = CollectionUtils.duplicatedElementsOf(labels);
1162 for (Label dupe : dupes) {
1163 eventHandler.handle(Event.error(location, String.format(
1164 "label '%s' is duplicated in the '%s' list of '%s'", dupe, attrName, owner)));
1165 }
1166 return dupes.isEmpty();
1167 }
1168
1169 /**
1170 * Adds an environment group to the package.
1171 */
1172 void addEnvironmentGroup(String name, List<Label> environments, List<Label> defaults,
1173 EventHandler eventHandler, Location location)
Lukacs Berkia6434362015-09-15 13:56:14 +00001174 throws NameConflictException, LabelSyntaxException {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001175
1176 if (!checkForDuplicateLabels(environments, name, "environments", location, eventHandler)
1177 || !checkForDuplicateLabels(defaults, name, "defaults", location, eventHandler)) {
1178 setContainsErrors();
1179 return;
1180 }
1181
1182 EnvironmentGroup group = new EnvironmentGroup(createLabel(name), pkg, environments,
1183 defaults, location);
1184 Target existing = targets.get(group.getName());
1185 if (existing != null) {
1186 throw nameConflict(group, existing);
1187 }
1188
1189 targets.put(group.getName(), group);
1190 Collection<Event> membershipErrors = group.validateMembership();
1191 if (!membershipErrors.isEmpty()) {
1192 for (Event error : membershipErrors) {
1193 eventHandler.handle(error);
1194 }
1195 setContainsErrors();
1196 return;
1197 }
1198
1199 // For each declared environment, make sure it doesn't also belong to some other group.
1200 for (Label environment : group.getEnvironments()) {
1201 EnvironmentGroup otherGroup = environmentGroups.get(environment);
1202 if (otherGroup != null) {
1203 eventHandler.handle(Event.error(location, "environment " + environment + " belongs to"
1204 + " both " + group.getLabel() + " and " + otherGroup.getLabel()));
1205 setContainsErrors();
1206 } else {
1207 environmentGroups.put(environment, group);
1208 }
1209 }
1210 }
1211
1212 void addRule(Rule rule) throws NameConflictException {
1213 checkForConflicts(rule);
1214 // Now, modify the package:
1215 for (OutputFile outputFile : rule.getOutputFiles()) {
1216 targets.put(outputFile.getName(), outputFile);
1217 PathFragment outputFileFragment = new PathFragment(outputFile.getName());
1218 for (int i = 1; i < outputFileFragment.segmentCount(); i++) {
1219 String prefix = outputFileFragment.subFragment(0, i).toString();
1220 if (!outputFilePrefixes.containsKey(prefix)) {
1221 outputFilePrefixes.put(prefix, outputFile);
1222 }
1223 }
1224 }
1225 targets.put(rule.getName(), rule);
1226 if (rule.containsErrors()) {
1227 this.setContainsErrors();
1228 }
1229 }
1230
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001231 private Builder beforeBuild() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001232 Preconditions.checkNotNull(pkg);
1233 Preconditions.checkNotNull(filename);
1234 Preconditions.checkNotNull(buildFileLabel);
1235 Preconditions.checkNotNull(makeEnv);
1236 // Freeze subincludes.
1237 subincludes = (subincludes == null)
1238 ? Collections.<Label, Path>emptyMap()
1239 : Collections.unmodifiableMap(subincludes);
1240
1241 // We create the original BUILD InputFile when the package filename is set; however, the
1242 // visibility may be overridden with an exports_files directive, so we need to obtain the
1243 // current instance here.
1244 buildFile = (InputFile) Preconditions.checkNotNull(targets.get(buildFileLabel.getName()));
1245
1246 List<Rule> rules = Lists.newArrayList(getTargets(Rule.class));
1247
1248 // All labels mentioned in a rule that refer to an unknown target in the
1249 // current package are assumed to be InputFiles, so let's create them:
1250 for (final Rule rule : rules) {
1251 AggregatingAttributeMapper.of(rule).visitLabels(new AcceptsLabelAttribute() {
1252 @Override
1253 public void acceptLabelAttribute(Label label, Attribute attribute) {
1254 createInputFileMaybe(label, rule.getAttributeLocation(attribute.getName()));
1255 }
1256 });
1257 }
1258
1259 // "test_suite" rules have the idiosyncratic semantics of implicitly
1260 // depending on all tests in the package, iff tests=[] and suites=[].
1261 // Note, we implement this here when the Package is fully constructed,
1262 // since clearly this information isn't available at Rule construction
1263 // time, as forward references are permitted.
1264 List<Label> allTests = new ArrayList<>();
1265 for (Rule rule : rules) {
Greg Estrenc48214d2015-03-17 22:43:19 +00001266 if (TargetUtils.isTestRule(rule) && !TargetUtils.hasManualTag(rule)) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001267 allTests.add(rule.getLabel());
1268 }
1269 }
Laurent Le Brund6ff51c2015-09-03 23:45:05 +00001270 Collections.sort(allTests);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001271 for (Rule rule : rules) {
1272 AttributeMap attributes = NonconfigurableAttributeMapper.of(rule);
1273 if (rule.getRuleClass().equals("test_suite")
Lukacs Berkiffa73ad2015-09-18 11:40:12 +00001274 && attributes.get("tests", BuildType.LABEL_LIST).isEmpty()
1275 && (!attributes.has("suites", BuildType.LABEL_LIST)
1276 || attributes.get("suites", BuildType.LABEL_LIST).isEmpty())) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001277 rule.setAttributeValueByName("$implicit_tests", allTests);
1278 }
1279 }
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001280 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001281 }
1282
Michajlo Matijkiw91407182015-05-29 14:27:09 +00001283 /** Intended for use by {@link com.google.devtools.build.lib.skyframe.PackageFunction} only. */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001284 public Builder buildPartial() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001285 if (alreadyBuilt) {
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001286 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001287 }
1288 return beforeBuild();
1289 }
1290
Michajlo Matijkiw91407182015-05-29 14:27:09 +00001291 /** Intended for use by {@link com.google.devtools.build.lib.skyframe.PackageFunction} only. */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001292 public Package finishBuild() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001293 if (alreadyBuilt) {
1294 return pkg;
1295 }
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +00001296
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001297 // Freeze targets and distributions.
1298 targets = ImmutableMap.copyOf(targets);
1299 defaultDistributionSet =
1300 Collections.unmodifiableSet(defaultDistributionSet);
1301
Greg Estrend0f10dc2015-03-17 22:19:37 +00001302 // Now all targets have been loaded, so we validate the group's member environments.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001303 for (EnvironmentGroup envGroup : ImmutableSet.copyOf(environmentGroups.values())) {
Greg Estrend0f10dc2015-03-17 22:19:37 +00001304 Collection<Event> errors = envGroup.processMemberEnvironments(targets);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001305 if (!errors.isEmpty()) {
1306 addEvents(errors);
1307 setContainsErrors();
1308 }
1309 }
1310
1311 // Build the package.
1312 pkg.finishInit(this);
1313 alreadyBuilt = true;
1314 return pkg;
1315 }
1316
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +00001317 protected ExternalPackageBuilder externalPackageData() {
1318 return externalPackageData;
1319 }
1320
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001321 public Package build() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001322 if (alreadyBuilt) {
1323 return pkg;
1324 }
1325 beforeBuild();
1326 return finishBuild();
1327 }
1328
1329 /**
1330 * If "label" refers to a non-existent target in the current package, create
1331 * an InputFile target.
1332 */
1333 void createInputFileMaybe(Label label, Location location) {
Damien Martin-Guillerez49978052016-01-21 19:55:07 +00001334 if (label != null && label.getPackageIdentifier().equals(pkg.getPackageIdentifier())) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001335 if (!targets.containsKey(label.getName())) {
1336 addInputFile(label, location);
1337 }
1338 }
1339 }
1340
1341 private InputFile addInputFile(Label label, Location location) {
1342 InputFile inputFile = new InputFile(pkg, label, location);
1343 Target prev = targets.put(label.getName(), inputFile);
1344 Preconditions.checkState(prev == null);
1345 return inputFile;
1346 }
1347
1348 /**
1349 * Precondition check for addRule. We must maintain these invariants of the
1350 * package:
1351 * - Each name refers to at most one target.
1352 * - No rule with errors is inserted into the package.
1353 * - The generating rule of every output file in the package must itself be
1354 * in the package.
1355 */
1356 private void checkForConflicts(Rule rule) throws NameConflictException {
1357 String name = rule.getName();
1358 Target existing = targets.get(name);
1359 if (existing != null) {
1360 throw nameConflict(rule, existing);
1361 }
1362 Map<String, OutputFile> outputFiles = new HashMap<>();
1363
1364 for (OutputFile outputFile : rule.getOutputFiles()) {
1365 String outputFileName = outputFile.getName();
1366 if (outputFiles.put(outputFileName, outputFile) != null) { // dups within a single rule:
1367 throw duplicateOutputFile(outputFile, outputFile);
1368 }
1369 existing = targets.get(outputFileName);
1370 if (existing != null) {
1371 throw duplicateOutputFile(outputFile, existing);
1372 }
1373
1374 // Check if this output file is the prefix of an already existing one
1375 if (outputFilePrefixes.containsKey(outputFileName)) {
1376 throw conflictingOutputFile(outputFile, outputFilePrefixes.get(outputFileName));
1377 }
1378
1379 // Check if a prefix of this output file matches an already existing one
1380 PathFragment outputFileFragment = new PathFragment(outputFileName);
1381 for (int i = 1; i < outputFileFragment.segmentCount(); i++) {
1382 String prefix = outputFileFragment.subFragment(0, i).toString();
1383 if (outputFiles.containsKey(prefix)) {
1384 throw conflictingOutputFile(outputFile, outputFiles.get(prefix));
1385 }
1386 if (targets.containsKey(prefix)
1387 && targets.get(prefix) instanceof OutputFile) {
1388 throw conflictingOutputFile(outputFile, (OutputFile) targets.get(prefix));
1389 }
1390
1391 if (!outputFilePrefixes.containsKey(prefix)) {
1392 outputFilePrefixes.put(prefix, outputFile);
1393 }
1394 }
1395 }
1396
1397 checkForInputOutputConflicts(rule, outputFiles.keySet());
1398 }
1399
1400 /**
1401 * A utility method that checks for conflicts between
1402 * input file names and output file names for a rule from a build
1403 * file.
1404 * @param rule the rule whose inputs and outputs are
1405 * to be checked for conflicts.
1406 * @param outputFiles a set containing the names of output
1407 * files to be generated by the rule.
1408 * @throws NameConflictException if a conflict is found.
1409 */
1410 private void checkForInputOutputConflicts(Rule rule, Set<String> outputFiles)
1411 throws NameConflictException {
1412 PathFragment packageFragment = rule.getLabel().getPackageFragment();
1413 for (Label inputLabel : rule.getLabels()) {
1414 if (packageFragment.equals(inputLabel.getPackageFragment())
1415 && outputFiles.contains(inputLabel.getName())) {
1416 throw inputOutputNameConflict(rule, inputLabel.getName());
1417 }
1418 }
1419 }
1420
1421 /** An output file conflicts with another output file or the BUILD file. */
1422 private NameConflictException duplicateOutputFile(OutputFile duplicate, Target existing) {
1423 return new NameConflictException(duplicate.getTargetKind() + " '" + duplicate.getName()
1424 + "' in rule '" + duplicate.getGeneratingRule().getName() + "' "
1425 + conflictsWith(existing));
1426 }
1427
1428 /** The package contains two targets with the same name. */
1429 private NameConflictException nameConflict(Target duplicate, Target existing) {
1430 return new NameConflictException(duplicate.getTargetKind() + " '" + duplicate.getName()
1431 + "' in package '" + duplicate.getLabel().getPackageName() + "' "
1432 + conflictsWith(existing));
1433 }
1434
1435 /** A a rule has a input/output name conflict. */
1436 private NameConflictException inputOutputNameConflict(Rule rule, String conflictingName) {
1437 return new NameConflictException("rule '" + rule.getName() + "' has file '"
1438 + conflictingName + "' as both an input and an output");
1439 }
1440
1441 private static NameConflictException conflictingOutputFile(
1442 OutputFile added, OutputFile existing) {
1443 if (added.getGeneratingRule() == existing.getGeneratingRule()) {
1444 return new NameConflictException(String.format(
1445 "rule '%s' has conflicting output files '%s' and '%s'", added.getGeneratingRule()
1446 .getName(), added.getName(), existing.getName()));
1447 } else {
1448 return new NameConflictException(String.format(
1449 "output file '%s' of rule '%s' conflicts with output file '%s' of rule '%s'", added
1450 .getName(), added.getGeneratingRule().getName(), existing.getName(), existing
1451 .getGeneratingRule().getName()));
1452 }
1453 }
1454
1455 /**
1456 * Utility function for generating exception messages.
1457 */
1458 private static String conflictsWith(Target target) {
1459 String message = "conflicts with existing ";
1460 if (target instanceof OutputFile) {
1461 return message + "generated file from rule '"
1462 + ((OutputFile) target).getGeneratingRule().getName()
1463 + "'";
1464 } else {
1465 return message + target.getTargetKind();
1466 }
1467 }
1468 }
1469}