blob: 8a666b0ecfe1471d25fd45e179e9f6310afe564b [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;
Mark Schaller6df81792015-12-10 18:47:47 +000036import com.google.devtools.build.lib.util.Preconditions;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010037import com.google.devtools.build.lib.vfs.Canonicalizer;
38import com.google.devtools.build.lib.vfs.Path;
39import com.google.devtools.build.lib.vfs.PathFragment;
40
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010041import java.io.PrintStream;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010042import java.util.ArrayList;
43import java.util.Collection;
44import java.util.Collections;
45import java.util.HashMap;
46import java.util.List;
47import java.util.Map;
48import java.util.Set;
49
Han-Wen Nienhuys67e6f982016-01-15 15:54:12 +000050import javax.annotation.Nullable;
51
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010052/**
53 * A package, which is a container of {@link Rule}s, each of
54 * which contains a dictionary of named attributes.
55 *
56 * <p>Package instances are intended to be immutable and for all practical
57 * purposes can be treated as such. Note, however, that some member variables
58 * exposed via the public interface are not strictly immutable, so until their
59 * types are guaranteed immutable we're not applying the {@code @Immutable}
60 * annotation here.
61 */
Michajlo Matijkiw17f11eb2015-10-14 04:19:28 +000062public class Package {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010063
64 /**
65 * Common superclass for all name-conflict exceptions.
66 */
67 public static class NameConflictException extends Exception {
68 protected NameConflictException(String message) {
69 super(message);
70 }
71 }
72
73 /**
74 * The repository identifier for this package.
75 */
76 private final PackageIdentifier packageIdentifier;
77
78 /**
79 * The name of the package, e.g. "foo/bar".
80 */
81 protected final String name;
82
83 /**
84 * Like name, but in the form of a PathFragment.
85 */
86 private final PathFragment nameFragment;
87
88 /**
89 * The filename of this package's BUILD file.
90 */
91 protected Path filename;
92
93 /**
94 * The directory in which this package's BUILD file resides. All InputFile
95 * members of the packages are located relative to this directory.
96 */
97 private Path packageDirectory;
98
99 /**
Kristina Chodorow91876f02015-02-27 17:14:12 +0000100 * The name of the workspace this package is in. Used as a prefix for the runfiles directory.
101 * This can be set in the WORKSPACE file. This must be a valid target name.
102 */
Ulf Adamsd13207c2015-09-04 14:53:43 +0000103 protected String workspaceName;
Kristina Chodorow91876f02015-02-27 17:14:12 +0000104
105 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100106 * The root of the source tree in which this package was found. It is an invariant that
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000107 * {@code sourceRoot.getRelative(packageId.getPathFragment()).equals(packageDirectory)}.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100108 */
109 private Path sourceRoot;
110
111 /**
112 * The "Make" environment of this package, containing package-local
113 * definitions of "Make" variables.
114 */
115 private MakeEnvironment makeEnv;
116
117 /**
118 * The collection of all targets defined in this package, indexed by name.
119 */
120 protected Map<String, Target> targets;
121
122 /**
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000123 * Default visibility for rules that do not specify it.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100124 */
125 private RuleVisibility defaultVisibility;
126 private boolean defaultVisibilitySet;
127
128 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100129 * Default package-level 'testonly' value for rules that do not specify it.
130 */
131 private boolean defaultTestOnly = false;
132
133 /**
134 * Default package-level 'deprecation' value for rules that do not specify it.
135 */
136 private String defaultDeprecation;
137
138 /**
139 * Default header strictness checking for rules that do not specify it.
140 */
141 private String defaultHdrsCheck;
142
143 /**
144 * Default copts for cc_* rules. The rules' individual copts will append to
145 * this value.
146 */
147 private ImmutableList<String> defaultCopts;
148
149 /**
150 * The InputFile target corresponding to this package's BUILD file.
151 */
152 private InputFile buildFile;
153
154 /**
155 * True iff this package's BUILD files contained lexical or grammatical
156 * errors, or experienced errors during evaluation, or semantic errors during
157 * the construction of any rule.
158 *
159 * <p>Note: A package containing errors does not necessarily prevent a build;
160 * if all the rules needed for a given build were constructed prior to the
161 * first error, the build may proceed.
162 */
163 private boolean containsErrors;
164
165 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100166 * The set of labels subincluded by this package.
167 */
168 private Set<Label> subincludes;
169
170 /**
171 * The list of transitive closure of the Skylark file dependencies.
172 */
173 private ImmutableList<Label> skylarkFileDependencies;
174
175 /**
176 * The package's default "licenses" and "distribs" attributes, as specified
177 * in calls to licenses() and distribs() in the BUILD file.
178 */
179 // These sets contain the values specified by the most recent licenses() or
180 // distribs() declarations encountered during package parsing:
181 private License defaultLicense;
182 private Set<License.DistributionType> defaultDistributionSet;
183
184
185 /**
186 * The names of the package() attributes that declare default values for rule
187 * {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR} and {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
188 * values when not explicitly specified.
189 */
190 public static final String DEFAULT_COMPATIBLE_WITH_ATTRIBUTE = "default_compatible_with";
191 public static final String DEFAULT_RESTRICTED_TO_ATTRIBUTE = "default_restricted_to";
192
193 private Set<Label> defaultCompatibleWith = ImmutableSet.of();
194 private Set<Label> defaultRestrictedTo = ImmutableSet.of();
195
196 private ImmutableSet<String> features;
197
198 private ImmutableList<Event> events;
199
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100200 /**
201 * Package initialization, part 1 of 3: instantiates a new package with the
202 * given name.
203 *
204 * <p>As part of initialization, {@link Builder} constructs {@link InputFile}
205 * and {@link PackageGroup} instances that require a valid Package instance where
206 * {@link Package#getNameFragment()} is accessible. That's why these settings are
207 * applied here at the start.
208 *
209 * @precondition {@code name} must be a suffix of
210 * {@code filename.getParentDirectory())}.
211 */
Nathan Harmata42fb5602016-05-25 20:32:08 +0000212 protected Package(PackageIdentifier packageId, String runfilesPrefix) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100213 this.packageIdentifier = packageId;
Ulf Adamsd13207c2015-09-04 14:53:43 +0000214 this.workspaceName = runfilesPrefix;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100215 this.nameFragment = Canonicalizer.fragments().intern(packageId.getPackageFragment());
216 this.name = nameFragment.getPathString();
217 }
218
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100219 /** Returns this packages' identifier. */
220 public PackageIdentifier getPackageIdentifier() {
221 return packageIdentifier;
222 }
223
224 /**
225 * Package initialization: part 2 of 3: sets this package's default header
226 * strictness checking.
227 *
228 * <p>This is needed to support C++-related rule classes
229 * which accesses {@link #getDefaultHdrsCheck} from the still-under-construction
230 * package.
231 */
232 protected void setDefaultHdrsCheck(String defaultHdrsCheck) {
233 this.defaultHdrsCheck = defaultHdrsCheck;
234 }
235
236 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100237 * Set the default 'testonly' value for this package.
238 */
239 protected void setDefaultTestOnly(boolean testOnly) {
240 defaultTestOnly = testOnly;
241 }
242
243 /**
244 * Set the default 'deprecation' value for this package.
245 */
246 protected void setDefaultDeprecation(String deprecation) {
247 defaultDeprecation = deprecation;
248 }
249
250 /**
251 * Sets the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
252 * attribute when not explicitly specified by the rule.
253 */
254 protected void setDefaultCompatibleWith(Set<Label> environments) {
255 defaultCompatibleWith = environments;
256 }
257
258 /**
259 * Sets the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
260 * attribute when not explicitly specified by the rule.
261 */
262 protected void setDefaultRestrictedTo(Set<Label> environments) {
263 defaultRestrictedTo = environments;
264 }
265
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000266 private static Path getSourceRoot(Path buildFile, PathFragment packageFragment) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100267 Path current = buildFile.getParentDirectory();
Kristina Chodorowb6fbab72016-01-28 14:38:31 +0000268 for (int i = 0, len = packageFragment.segmentCount();
269 i < len && !packageFragment.equals(PathFragment.EMPTY_FRAGMENT); i++) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100270 current = current.getParentDirectory();
271 }
272 return current;
273 }
274
275 /**
276 * Package initialization: part 3 of 3: applies all other settings and completes
277 * initialization of the package.
278 *
279 * <p>Only after this method is called can this package be considered "complete"
280 * and be shared publicly.
281 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000282 protected void finishInit(Builder builder) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100283 // If any error occurred during evaluation of this package, consider all
284 // rules in the package to be "in error" also (even if they were evaluated
285 // prior to the error). This behaviour is arguably stricter than need be,
286 // but stopping a build only for some errors but not others creates user
287 // confusion.
288 if (builder.containsErrors) {
289 for (Rule rule : builder.getTargets(Rule.class)) {
290 rule.setContainsErrors();
291 }
292 }
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000293 this.filename = builder.getFilename();
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100294 this.packageDirectory = filename.getParentDirectory();
295
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000296 this.sourceRoot = getSourceRoot(filename, packageIdentifier.getPathFragment());
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100297 if ((sourceRoot == null
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000298 || !sourceRoot.getRelative(packageIdentifier.getPathFragment()).equals(packageDirectory))
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100299 && !filename.getBaseName().equals("WORKSPACE")) {
300 throw new IllegalArgumentException(
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000301 "Invalid BUILD file name for package '" + packageIdentifier + "': " + filename);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100302 }
303
304 this.makeEnv = builder.makeEnv.build();
305 this.targets = ImmutableSortedKeyMap.copyOf(builder.targets);
306 this.defaultVisibility = builder.defaultVisibility;
307 this.defaultVisibilitySet = builder.defaultVisibilitySet;
308 if (builder.defaultCopts == null) {
309 this.defaultCopts = ImmutableList.of();
310 } else {
311 this.defaultCopts = ImmutableList.copyOf(builder.defaultCopts);
312 }
313 this.buildFile = builder.buildFile;
314 this.containsErrors = builder.containsErrors;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100315 this.subincludes = builder.subincludes.keySet();
316 this.skylarkFileDependencies = builder.skylarkFileDependencies;
317 this.defaultLicense = builder.defaultLicense;
318 this.defaultDistributionSet = builder.defaultDistributionSet;
319 this.features = ImmutableSortedSet.copyOf(builder.features);
320 this.events = ImmutableList.copyOf(builder.events);
321 }
322
323 /**
324 * Returns the list of subincluded labels on which the validity of this package depends.
325 */
326 public Set<Label> getSubincludeLabels() {
327 return subincludes;
328 }
329
330 /**
331 * Returns the list of transitive closure of the Skylark file dependencies of this package.
332 */
333 public ImmutableList<Label> getSkylarkFileDependencies() {
334 return skylarkFileDependencies;
335 }
336
337 /**
338 * Returns the filename of the BUILD file which defines this package. The
339 * parent directory of the BUILD file is the package directory.
340 */
341 public Path getFilename() {
342 return filename;
343 }
344
345 /**
346 * Returns the source root (a directory) beneath which this package's BUILD file was found.
347 *
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000348 * <p> Assumes invariant:
Kristina Chodorowdc30c072015-04-22 13:48:47 +0000349 * {@code getSourceRoot().getRelative(packageId.getPathFragment()).equals(getPackageDirectory())}
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100350 */
351 public Path getSourceRoot() {
352 return sourceRoot;
353 }
354
355 /**
356 * Returns the directory containing the package's BUILD file.
357 */
358 public Path getPackageDirectory() {
359 return packageDirectory;
360 }
361
362 /**
363 * Returns the name of this package. If this build is using external repositories then this name
364 * may not be unique!
365 */
366 public String getName() {
367 return name;
368 }
369
370 /**
371 * Like {@link #getName}, but has type {@code PathFragment}.
372 */
373 public PathFragment getNameFragment() {
374 return nameFragment;
375 }
376
377 /**
378 * Returns the "Make" value from the package's make environment whose name
379 * is "varname", or null iff the variable is not defined in the environment.
380 */
381 public String lookupMakeVariable(String varname, String platform) {
382 return makeEnv.lookup(varname, platform);
383 }
384
385 /**
386 * Returns the make environment. This should only ever be used for serialization -- how the
387 * make variables are implemented is an implementation detail.
388 */
389 MakeEnvironment getMakeEnvironment() {
390 return makeEnv;
391 }
392
393 /**
Laurent Le Brune8ce8472015-03-12 14:20:01 +0000394 * Returns all make variables for a given platform.
395 */
396 public Map<String, String> getAllMakeVariables(String platform) {
397 ImmutableMap.Builder<String, String> map = ImmutableMap.builder();
398 for (String var : makeEnv.getBindings().keySet()) {
399 map.put(var, makeEnv.lookup(var, platform));
400 }
401 return map.build();
402 }
403
404 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100405 * Returns the label of this package's BUILD file.
406 *
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000407 * <p> Typically <code>getBuildFileLabel().getName().equals("BUILD")</code> --
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100408 * though not necessarily: data in a subdirectory of a test package may use a
409 * different filename to avoid inadvertently creating a new package.
410 */
Mark Schaller4fa83ac2015-07-10 16:59:37 +0000411 public Label getBuildFileLabel() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100412 return buildFile.getLabel();
413 }
414
415 /**
416 * Returns the InputFile target for this package's BUILD file.
417 */
418 public InputFile getBuildFile() {
419 return buildFile;
420 }
421
422 /**
423 * Returns true if errors were encountered during evaluation of this package.
424 * (The package may be incomplete and its contents should not be relied upon
425 * for critical operations. However, any Rules belonging to the package are
426 * guaranteed to be intact, unless their <code>containsErrors()</code> flag
427 * is set.)
428 */
429 public boolean containsErrors() {
430 return containsErrors;
431 }
432
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100433 public List<Event> getEvents() {
434 return events;
435 }
436
437 /**
438 * Returns an (immutable, unordered) view of all the targets belonging to this package.
439 */
440 public Collection<Target> getTargets() {
441 return getTargets(targets);
442 }
443
444 /**
445 * Common getTargets implementation, accessible by both {@link Package} and
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000446 * {@link Package.Builder}.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100447 */
448 private static Collection<Target> getTargets(Map<String, Target> targetMap) {
449 return Collections.unmodifiableCollection(targetMap.values());
450 }
451
452 /**
453 * Returns a (read-only, unordered) iterator of all the targets belonging
454 * to this package which are instances of the specified class.
455 */
456 public <T extends Target> Iterable<T> getTargets(Class<T> targetClass) {
457 return getTargets(targets, targetClass);
458 }
459
460 /**
461 * Common getTargets implementation, accessible by both {@link Package} and
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000462 * {@link Package.Builder}.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100463 */
464 private static <T extends Target> Iterable<T> getTargets(Map<String, Target> targetMap,
465 Class<T> targetClass) {
466 return Iterables.filter(targetMap.values(), targetClass);
467 }
468
469 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100470 * Returns the rule that corresponds to a particular BUILD target name. Useful
471 * for walking through the dependency graph of a target.
472 * Fails if the target is not a Rule.
473 */
Ulf Adamsb313a972015-05-04 12:05:57 +0000474 @VisibleForTesting // Should be package-private
475 public Rule getRule(String targetName) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100476 return (Rule) targets.get(targetName);
477 }
478
479 /**
Kristina Chodorow91876f02015-02-27 17:14:12 +0000480 * Returns this package's workspace name.
481 *
482 * <p>Package-private to encourage callers to get their workspace name from a rule, not a
483 * package.</p>
484 */
Kristina Chodorow734e7f72016-01-29 15:04:31 +0000485 public String getWorkspaceName() {
Kristina Chodorow91876f02015-02-27 17:14:12 +0000486 return workspaceName;
487 }
488
489 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100490 * Returns the features specified in the <code>package()</code> declaration.
491 */
492 public ImmutableSet<String> getFeatures() {
493 return features;
494 }
495
496 /**
497 * Returns the target (a member of this package) whose name is "targetName".
498 * First rules are searched, then output files, then input files. The target
499 * name must be valid, as defined by {@code LabelValidator#validateTargetName}.
500 *
501 * @throws NoSuchTargetException if the specified target was not found.
502 */
503 public Target getTarget(String targetName) throws NoSuchTargetException {
504 Target target = targets.get(targetName);
505 if (target != null) {
506 return target;
507 }
508
509 // No such target.
510
511 // If there's a file on the disk that's not mentioned in the BUILD file,
512 // produce a more informative error. NOTE! this code path is only executed
513 // on failure, which is (relatively) very rare. In the common case no
514 // stat(2) is executed.
515 Path filename = getPackageDirectory().getRelative(targetName);
516 String suffix;
517 if (!new PathFragment(targetName).isNormalized()) {
518 // Don't check for file existence in this case because the error message
519 // would be confusing and wrong. If the targetName is "foo/bar/.", and
520 // there is a directory "foo/bar", it doesn't mean that "//pkg:foo/bar/."
521 // is a valid label.
522 suffix = "";
523 } else if (filename.isDirectory()) {
524 suffix = "; however, a source directory of this name exists. (Perhaps add "
525 + "'exports_files([\"" + targetName + "\"])' to " + name + "/BUILD, or define a "
526 + "filegroup?)";
527 } else if (filename.exists()) {
528 suffix = "; however, a source file of this name exists. (Perhaps add "
529 + "'exports_files([\"" + targetName + "\"])' to " + name + "/BUILD?)";
530 } else {
531 suffix = "";
532 }
533
Nathan Harmata42fb5602016-05-25 20:32:08 +0000534 throw makeNoSuchTargetException(targetName, suffix);
535 }
536
537 protected NoSuchTargetException makeNoSuchTargetException(String targetName, String suffix) {
538 Label label;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100539 try {
Nathan Harmata42fb5602016-05-25 20:32:08 +0000540 label = createLabel(targetName);
Lukacs Berkia6434362015-09-15 13:56:14 +0000541 } catch (LabelSyntaxException e) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100542 throw new IllegalArgumentException(targetName);
543 }
Nathan Harmata42fb5602016-05-25 20:32:08 +0000544 String msg = String.format(
545 "target '%s' not declared in package '%s'%s defined by %s",
546 targetName,
547 name,
548 suffix,
549 filename);
550 return new NoSuchTargetException(label, msg);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100551 }
552
553 /**
554 * Creates a label for a target inside this package.
555 *
Lukacs Berkia6434362015-09-15 13:56:14 +0000556 * @throws LabelSyntaxException if the {@code targetName} is invalid
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100557 */
Lukacs Berkia6434362015-09-15 13:56:14 +0000558 public Label createLabel(String targetName) throws LabelSyntaxException {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100559 return Label.create(packageIdentifier, targetName);
560 }
561
562 /**
563 * Returns the default visibility for this package.
564 */
565 public RuleVisibility getDefaultVisibility() {
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000566 return defaultVisibility;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100567 }
568
569 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100570 * Returns the default testonly value.
571 */
572 public Boolean getDefaultTestOnly() {
573 return defaultTestOnly;
574 }
575
576 /**
Greg Estrenc48214d2015-03-17 22:43:19 +0000577 * Returns the default deprecation value.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100578 */
579 public String getDefaultDeprecation() {
580 return defaultDeprecation;
581 }
582
583 /**
584 * Gets the default header checking mode.
585 */
586 public String getDefaultHdrsCheck() {
Googler198a54f2016-01-18 11:02:10 +0000587 return defaultHdrsCheck != null ? defaultHdrsCheck : "strict";
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100588 }
589
590 /**
591 * Returns the default copts value, to which rules should append their
592 * specific copts.
593 */
594 public ImmutableList<String> getDefaultCopts() {
595 return defaultCopts;
596 }
597
598 /**
599 * Returns whether the default header checking mode has been set or it is the
600 * default value.
601 */
602 public boolean isDefaultHdrsCheckSet() {
603 return defaultHdrsCheck != null;
604 }
605
606 public boolean isDefaultVisibilitySet() {
607 return defaultVisibilitySet;
608 }
609
610 /**
611 * Gets the parsed license object for the default license
612 * declared by this package.
613 */
614 public License getDefaultLicense() {
615 return defaultLicense;
616 }
617
618 /**
619 * Returns the parsed set of distributions declared as the default for this
620 * package.
621 */
622 public Set<License.DistributionType> getDefaultDistribs() {
623 return defaultDistributionSet;
624 }
625
626 /**
627 * Returns the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
628 * attribute when not explicitly specified by the rule.
629 */
630 public Set<Label> getDefaultCompatibleWith() {
631 return defaultCompatibleWith;
632 }
633
634 /**
635 * Returns the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
636 * attribute when not explicitly specified by the rule.
637 */
638 public Set<Label> getDefaultRestrictedTo() {
639 return defaultRestrictedTo;
640 }
641
642 @Override
643 public String toString() {
Han-Wen Nienhuys58070da2015-04-20 11:37:58 +0000644 return "Package(" + name + ")="
645 + (targets != null ? getTargets(Rule.class) : "initializing...");
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100646 }
647
648 /**
649 * Dumps the package for debugging. Do not depend on the exact format/contents of this debugging
650 * output.
651 */
652 public void dump(PrintStream out) {
653 out.println(" Package " + getName() + " (" + getFilename() + ")");
654
655 // Rules:
656 out.println(" Rules");
657 for (Rule rule : getTargets(Rule.class)) {
658 out.println(" " + rule.getTargetKind() + " " + rule.getLabel());
659 for (Attribute attr : rule.getAttributes()) {
660 for (Object possibleValue : AggregatingAttributeMapper.of(rule)
661 .visitAttribute(attr.getName(), attr.getType())) {
662 out.println(" " + attr.getName() + " = " + possibleValue);
663 }
664 }
665 }
666
667 // Files:
668 out.println(" Files");
669 for (FileTarget file : getTargets(FileTarget.class)) {
670 out.print(" " + file.getTargetKind() + " " + file.getLabel());
671 if (file instanceof OutputFile) {
672 out.println(" (generated by " + ((OutputFile) file).getGeneratingRule().getLabel() + ")");
673 } else {
674 out.println();
675 }
676 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100677 }
678
Nathan Harmata42fb5602016-05-25 20:32:08 +0000679 public static Builder newExternalPackageBuilder(Builder.Helper helper, Path workspacePath,
680 String runfilesPrefix) {
681 Builder b = new Builder(helper.createFreshPackage(
682 Label.EXTERNAL_PACKAGE_IDENTIFIER, runfilesPrefix));
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +0000683 b.setFilename(workspacePath);
684 b.setMakeEnv(new MakeEnvironment.Builder());
685 return b;
686 }
687
Nathan Harmata42fb5602016-05-25 20:32:08 +0000688 /** A builder for {@link Package} objects. Only intended to be used by {@link PackageFactory}. */
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +0000689 public static class Builder {
Nathan Harmata42fb5602016-05-25 20:32:08 +0000690 public static interface Helper {
691 /**
692 * Returns a fresh {@link Package} instance that a {@link Builder} will internally mutate
693 * during package loading.
694 */
695 Package createFreshPackage(PackageIdentifier packageId, String runfilesPrefix);
696 }
697
698 /** {@link Helper} that simply calls the {@link Package} constructor. */
699 public static class DefaultHelper implements Helper {
700 public static final DefaultHelper INSTANCE = new DefaultHelper();
701
702 private DefaultHelper() {
703 }
704
705 @Override
706 public Package createFreshPackage(PackageIdentifier packageId, String runfilesPrefix) {
707 return new Package(packageId, runfilesPrefix);
708 }
Ulf Adams4226be22015-09-03 17:00:54 +0000709 }
710
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100711 /**
712 * The output instance for this builder. Needs to be instantiated and
713 * available with name info throughout initialization. All other settings
Michajlo Matijkiw17f11eb2015-10-14 04:19:28 +0000714 * are applied during {@link #build}. See {@link Package#Package}
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100715 * and {@link Package#finishInit} for details.
716 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000717 protected Package pkg;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100718
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000719 private Path filename = null;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100720 private Label buildFileLabel = null;
721 private InputFile buildFile = null;
722 private MakeEnvironment.Builder makeEnv = null;
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000723 private RuleVisibility defaultVisibility = ConstantRuleVisibility.PRIVATE;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100724 private boolean defaultVisibilitySet;
725 private List<String> defaultCopts = null;
726 private List<String> features = new ArrayList<>();
727 private List<Event> events = Lists.newArrayList();
728 private boolean containsErrors = false;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100729
730 private License defaultLicense = License.NO_LICENSE;
731 private Set<License.DistributionType> defaultDistributionSet = License.DEFAULT_DISTRIB;
732
733 protected Map<String, Target> targets = new HashMap<>();
734 protected Map<Label, EnvironmentGroup> environmentGroups = new HashMap<>();
735
736 protected Map<Label, Path> subincludes = null;
Kristina Chodorow91876f02015-02-27 17:14:12 +0000737 protected ImmutableList<Label> skylarkFileDependencies = ImmutableList.of();
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100738
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +0000739 protected ExternalPackageBuilder externalPackageData = new ExternalPackageBuilder();
740
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100741 /**
742 * True iff the "package" function has already been called in this package.
743 */
744 private boolean packageFunctionUsed;
745
746 /**
747 * The collection of the prefixes of every output file. Maps every prefix
748 * to an output file whose prefix it is.
749 *
750 * <p>This is needed to make the output file prefix conflict check be
751 * reasonably fast. However, since it can potentially take a lot of memory and
752 * is useless after the package has been loaded, it isn't passed to the
753 * package itself.
754 */
755 private Map<String, OutputFile> outputFilePrefixes = new HashMap<>();
756
757 private boolean alreadyBuilt = false;
758
759 private EventHandler builderEventHandler = new EventHandler() {
760 @Override
761 public void handle(Event event) {
762 addEvent(event);
763 }
764 };
765
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000766 protected Builder(Package pkg) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100767 this.pkg = pkg;
768 if (pkg.getName().startsWith("javatests/")) {
769 setDefaultTestonly(true);
770 }
771 }
772
Nathan Harmata42fb5602016-05-25 20:32:08 +0000773 public Builder(Helper helper, PackageIdentifier id, String runfilesPrefix) {
774 this(helper.createFreshPackage(id, runfilesPrefix));
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100775 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100776
777 protected PackageIdentifier getPackageIdentifier() {
778 return pkg.getPackageIdentifier();
779 }
780
Damien Martin-Guillerez8ca065c22015-11-27 16:22:59 +0000781 /** Determine if we are in the WORKSPACE file or not */
782 public boolean isWorkspace() {
Lukacs Berkie19ee272015-12-10 11:34:29 +0000783 return pkg.getPackageIdentifier().equals(Label.EXTERNAL_PACKAGE_IDENTIFIER);
Damien Martin-Guillerez8ca065c22015-11-27 16:22:59 +0000784 }
785
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100786 /**
787 * Sets the name of this package's BUILD file.
788 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000789 Builder setFilename(Path filename) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100790 this.filename = filename;
791 try {
792 buildFileLabel = createLabel(filename.getBaseName());
793 addInputFile(buildFileLabel, Location.fromFile(filename));
Lukacs Berkia6434362015-09-15 13:56:14 +0000794 } catch (LabelSyntaxException e) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100795 // This can't actually happen.
796 throw new AssertionError("Package BUILD file has an illegal name: " + filename);
797 }
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000798 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100799 }
800
801 public Label getBuildFileLabel() {
802 return buildFileLabel;
803 }
804
805 Path getFilename() {
806 return filename;
807 }
808
Janak Ramakrishnan0a4c6e42015-09-17 00:37:58 +0000809 public List<Event> getEvents() {
810 return events;
811 }
812
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100813 /**
814 * Sets this package's Make environment.
815 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000816 Builder setMakeEnv(MakeEnvironment.Builder makeEnv) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100817 this.makeEnv = makeEnv;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000818 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100819 }
820
Googler94296442015-03-25 09:47:41 +0000821 MakeEnvironment.Builder getMakeEnvironment() {
822 return makeEnv;
823 }
824
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100825 /**
826 * Sets the default visibility for this package. Called at most once per
827 * package from PackageFactory.
828 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000829 Builder setDefaultVisibility(RuleVisibility visibility) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100830 this.defaultVisibility = visibility;
831 this.defaultVisibilitySet = true;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000832 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100833 }
834
835 /**
836 * Sets whether the default visibility is set in the BUILD file.
837 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000838 Builder setDefaultVisibilitySet(boolean defaultVisibilitySet) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100839 this.defaultVisibilitySet = defaultVisibilitySet;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000840 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100841 }
842
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100843 /** Sets the default value of 'testonly'. Rule-level 'testonly' will override this. */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000844 Builder setDefaultTestonly(boolean defaultTestonly) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100845 pkg.setDefaultTestOnly(defaultTestonly);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000846 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100847 }
848
849 /**
850 * Sets the default value of 'deprecation'. Rule-level 'deprecation' will append to this.
851 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000852 Builder setDefaultDeprecation(String defaultDeprecation) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100853 pkg.setDefaultDeprecation(defaultDeprecation);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000854 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100855 }
856
857 /**
Kristina Chodorow91876f02015-02-27 17:14:12 +0000858 * Uses the workspace name from {@code //external} to set this package's workspace name.
859 */
Han-Wen Nienhuys2cffe032015-09-21 15:28:40 +0000860 @VisibleForTesting
861 public Builder setWorkspaceName(String workspaceName) {
Kristina Chodorow91876f02015-02-27 17:14:12 +0000862 pkg.workspaceName = workspaceName;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000863 return this;
Kristina Chodorow91876f02015-02-27 17:14:12 +0000864 }
865
866 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100867 * Returns whether the "package" function has been called yet
868 */
869 public boolean isPackageFunctionUsed() {
870 return packageFunctionUsed;
871 }
872
873 public void setPackageFunctionUsed() {
874 packageFunctionUsed = true;
875 }
876
877 /**
878 * Sets the default header checking mode.
879 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000880 public Builder setDefaultHdrsCheck(String hdrsCheck) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100881 // Note that this setting is propagated directly to the package because
882 // other code needs the ability to read this info directly from the
883 // under-construction package. See {@link Package#setDefaultHdrsCheck}.
884 pkg.setDefaultHdrsCheck(hdrsCheck);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000885 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100886 }
887
888 /**
889 * Sets the default value of copts. Rule-level copts will append to this.
890 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000891 public Builder setDefaultCopts(List<String> defaultCopts) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100892 this.defaultCopts = defaultCopts;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000893 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100894 }
895
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000896 public Builder addFeatures(Iterable<String> features) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100897 Iterables.addAll(this.features, features);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000898 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100899 }
900
901 /**
902 * Declares that errors were encountering while loading this package.
903 */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000904 public Builder setContainsErrors() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100905 containsErrors = true;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000906 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100907 }
908
909 public boolean containsErrors() {
910 return containsErrors;
911 }
912
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000913 public Builder addEvents(Iterable<Event> events) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100914 for (Event event : events) {
915 addEvent(event);
916 }
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000917 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100918 }
919
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000920 public Builder addEvent(Event event) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100921 this.events.add(event);
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000922 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100923 }
924
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000925 Builder setSkylarkFileDependencies(ImmutableList<Label> skylarkFileDependencies) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100926 this.skylarkFileDependencies = skylarkFileDependencies;
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +0000927 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100928 }
929
930 /**
931 * Sets the default license for this package.
932 */
933 void setDefaultLicense(License license) {
934 this.defaultLicense = license;
935 }
936
937 License getDefaultLicense() {
938 return defaultLicense;
939 }
940
941 /**
942 * Initializes the default set of distributions for targets in this package.
943 *
Laurent Le Brun41079bd2015-06-19 09:37:25 +0000944 * <p> TODO(bazel-team): (2011) consider moving the license & distribs info into Metadata--maybe
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100945 * even in the Build language.
946 */
947 void setDefaultDistribs(Set<DistributionType> dists) {
948 this.defaultDistributionSet = dists;
949 }
950
951 Set<DistributionType> getDefaultDistribs() {
952 return defaultDistributionSet;
953 }
954
955 /**
956 * Sets the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
957 * attribute when not explicitly specified by the rule. Records a package error if
958 * any labels are duplicated.
959 */
960 void setDefaultCompatibleWith(List<Label> environments, String attrName, Location location) {
961 if (!checkForDuplicateLabels(environments, "package " + pkg.getName(), attrName, location,
962 builderEventHandler)) {
963 setContainsErrors();
964 }
965 pkg.setDefaultCompatibleWith(ImmutableSet.copyOf(environments));
966 }
967
968 /**
969 * Sets the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
970 * attribute when not explicitly specified by the rule. Records a package error if
971 * any labels are duplicated.
972 */
973 void setDefaultRestrictedTo(List<Label> environments, String attrName, Location location) {
974 if (!checkForDuplicateLabels(environments, "package " + pkg.getName(), attrName, location,
975 builderEventHandler)) {
976 setContainsErrors();
977 }
978
979 pkg.setDefaultRestrictedTo(ImmutableSet.copyOf(environments));
980 }
981
982 /**
Mark Schalleree624452016-01-13 18:41:24 +0000983 * Creates a new {@link Rule} {@code r} where {@code r.getPackage()} is the {@link Package}
984 * associated with this {@link Builder}.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100985 *
Mark Schalleree624452016-01-13 18:41:24 +0000986 * <p>The created {@link Rule} will have no attribute values, no output files, and therefore
987 * will be in an invalid state.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100988 */
Mark Schalleree624452016-01-13 18:41:24 +0000989 Rule createRule(
990 Label label,
991 RuleClass ruleClass,
992 Location location,
Michajlo Matijkiw1a759072015-10-22 13:52:00 +0000993 AttributeContainer attributeContainer) {
994 return new Rule(pkg, label, ruleClass, location, attributeContainer);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100995 }
996
997 /**
998 * Called by the parser when a "mocksubinclude" is encountered, to record the
999 * mappings from labels to absolute paths upon which that the validity of
1000 * this package depends.
1001 */
1002 void addSubinclude(Label label, Path resolvedPath) {
1003 if (subincludes == null) {
1004 // This is a TreeMap because the order needs to be deterministic.
1005 subincludes = Maps.newTreeMap();
1006 }
1007
1008 Path oldResolvedPath = subincludes.put(label, resolvedPath);
1009 if (oldResolvedPath != null && !oldResolvedPath.equals(resolvedPath)){
1010 // The same label should have been resolved to the same path
1011 throw new IllegalStateException("Ambiguous subinclude path");
1012 }
1013 }
1014
1015 public Set<Label> getSubincludeLabels() {
1016 return subincludes == null ? Sets.<Label>newHashSet() : subincludes.keySet();
1017 }
1018
1019 public Map<Label, Path> getSubincludes() {
1020 return subincludes == null ? Maps.<Label, Path>newHashMap() : subincludes;
1021 }
1022
1023 public Collection<Target> getTargets() {
1024 return Package.getTargets(targets);
1025 }
1026
Han-Wen Nienhuys67e6f982016-01-15 15:54:12 +00001027 @Nullable
1028 public Target getTarget(String name) {
1029 return targets.get(name);
1030 }
1031
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001032 /**
1033 * Returns an (immutable, unordered) view of all the targets belonging to
1034 * this package which are instances of the specified class.
1035 */
1036 <T extends Target> Iterable<T> getTargets(Class<T> targetClass) {
1037 return Package.getTargets(targets, targetClass);
1038 }
1039
1040 /**
1041 * An input file name conflicts with an existing package member.
1042 */
1043 static class GeneratedLabelConflict extends NameConflictException {
1044 private GeneratedLabelConflict(String message) {
1045 super(message);
1046 }
1047 }
1048
1049 /**
1050 * Creates an input file target in this package with the specified name.
1051 *
1052 * @param targetName name of the input file. This must be a valid target
1053 * name as defined by {@link
1054 * com.google.devtools.build.lib.cmdline.LabelValidator#validateTargetName}.
1055 * @return the newly-created InputFile, or the old one if it already existed.
1056 * @throws GeneratedLabelConflict if the name was already taken by a Rule or
1057 * an OutputFile target.
1058 * @throws IllegalArgumentException if the name is not a valid label
1059 */
1060 InputFile createInputFile(String targetName, Location location)
1061 throws GeneratedLabelConflict {
1062 Target existing = targets.get(targetName);
1063 if (existing == null) {
1064 try {
1065 return addInputFile(createLabel(targetName), location);
Lukacs Berkia6434362015-09-15 13:56:14 +00001066 } catch (LabelSyntaxException e) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001067 throw new IllegalArgumentException("FileTarget in package " + pkg.getName()
1068 + " has illegal name: " + targetName);
1069 }
1070 } else if (existing instanceof InputFile) {
1071 return (InputFile) existing; // idempotent
1072 } else {
1073 throw new GeneratedLabelConflict("generated label '//" + pkg.getName() + ":"
1074 + targetName + "' conflicts with existing "
1075 + existing.getTargetKind());
1076 }
1077 }
1078
1079 /**
1080 * Sets the visibility and license for an input file. The input file must already exist as
1081 * a member of this package.
1082 * @throws IllegalArgumentException if the input file doesn't exist in this
1083 * package's target map.
1084 */
1085 void setVisibilityAndLicense(InputFile inputFile, RuleVisibility visibility, License license) {
1086 String filename = inputFile.getName();
1087 Target cacheInstance = targets.get(filename);
Ulf Adams07dba942015-03-05 14:47:37 +00001088 if (!(cacheInstance instanceof InputFile)) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001089 throw new IllegalArgumentException("Can't set visibility for nonexistent FileTarget "
1090 + filename + " in package " + pkg.getName() + ".");
1091 }
1092 if (!((InputFile) cacheInstance).isVisibilitySpecified()
1093 || cacheInstance.getVisibility() != visibility
1094 || cacheInstance.getLicense() != license) {
1095 targets.put(filename, new InputFile(
1096 pkg, cacheInstance.getLabel(), cacheInstance.getLocation(), visibility, license));
1097 }
1098 }
1099
1100 /**
1101 * Creates a label for a target inside this package.
1102 *
Lukacs Berkia6434362015-09-15 13:56:14 +00001103 * @throws LabelSyntaxException if the {@code targetName} is invalid
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001104 */
Lukacs Berkia6434362015-09-15 13:56:14 +00001105 Label createLabel(String targetName) throws LabelSyntaxException {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001106 return Label.create(pkg.getPackageIdentifier(), targetName);
1107 }
1108
1109 /**
1110 * Adds a package group to the package.
1111 */
1112 void addPackageGroup(String name, Collection<String> packages, Collection<Label> includes,
1113 EventHandler eventHandler, Location location)
Lukacs Berkia6434362015-09-15 13:56:14 +00001114 throws NameConflictException, LabelSyntaxException {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001115 PackageGroup group =
1116 new PackageGroup(createLabel(name), pkg, packages, includes, eventHandler, location);
1117 Target existing = targets.get(group.getName());
1118 if (existing != null) {
1119 throw nameConflict(group, existing);
1120 }
1121
1122 targets.put(group.getName(), group);
1123
1124 if (group.containsErrors()) {
1125 setContainsErrors();
1126 }
1127 }
1128
1129 /**
1130 * Checks if any labels in the given list appear multiple times and reports an appropriate
1131 * error message if so. Returns true if no duplicates were found, false otherwise.
1132 *
Laurent Le Brun41079bd2015-06-19 09:37:25 +00001133 * <p> TODO(bazel-team): apply this to all build functions (maybe automatically?), possibly
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001134 * integrate with RuleClass.checkForDuplicateLabels.
1135 */
1136 private static boolean checkForDuplicateLabels(Collection<Label> labels, String owner,
1137 String attrName, Location location, EventHandler eventHandler) {
1138 Set<Label> dupes = CollectionUtils.duplicatedElementsOf(labels);
1139 for (Label dupe : dupes) {
1140 eventHandler.handle(Event.error(location, String.format(
1141 "label '%s' is duplicated in the '%s' list of '%s'", dupe, attrName, owner)));
1142 }
1143 return dupes.isEmpty();
1144 }
1145
1146 /**
1147 * Adds an environment group to the package.
1148 */
1149 void addEnvironmentGroup(String name, List<Label> environments, List<Label> defaults,
1150 EventHandler eventHandler, Location location)
Lukacs Berkia6434362015-09-15 13:56:14 +00001151 throws NameConflictException, LabelSyntaxException {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001152
1153 if (!checkForDuplicateLabels(environments, name, "environments", location, eventHandler)
1154 || !checkForDuplicateLabels(defaults, name, "defaults", location, eventHandler)) {
1155 setContainsErrors();
1156 return;
1157 }
1158
1159 EnvironmentGroup group = new EnvironmentGroup(createLabel(name), pkg, environments,
1160 defaults, location);
1161 Target existing = targets.get(group.getName());
1162 if (existing != null) {
1163 throw nameConflict(group, existing);
1164 }
1165
1166 targets.put(group.getName(), group);
1167 Collection<Event> membershipErrors = group.validateMembership();
1168 if (!membershipErrors.isEmpty()) {
1169 for (Event error : membershipErrors) {
1170 eventHandler.handle(error);
1171 }
1172 setContainsErrors();
1173 return;
1174 }
1175
1176 // For each declared environment, make sure it doesn't also belong to some other group.
1177 for (Label environment : group.getEnvironments()) {
1178 EnvironmentGroup otherGroup = environmentGroups.get(environment);
1179 if (otherGroup != null) {
1180 eventHandler.handle(Event.error(location, "environment " + environment + " belongs to"
1181 + " both " + group.getLabel() + " and " + otherGroup.getLabel()));
1182 setContainsErrors();
1183 } else {
1184 environmentGroups.put(environment, group);
1185 }
1186 }
1187 }
1188
1189 void addRule(Rule rule) throws NameConflictException {
Lukacs Berki4867ef72016-05-17 11:15:19 +00001190 Preconditions.checkArgument(rule.getPackage() == pkg);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001191 checkForConflicts(rule);
1192 // Now, modify the package:
1193 for (OutputFile outputFile : rule.getOutputFiles()) {
1194 targets.put(outputFile.getName(), outputFile);
1195 PathFragment outputFileFragment = new PathFragment(outputFile.getName());
1196 for (int i = 1; i < outputFileFragment.segmentCount(); i++) {
1197 String prefix = outputFileFragment.subFragment(0, i).toString();
1198 if (!outputFilePrefixes.containsKey(prefix)) {
1199 outputFilePrefixes.put(prefix, outputFile);
1200 }
1201 }
1202 }
1203 targets.put(rule.getName(), rule);
1204 if (rule.containsErrors()) {
1205 this.setContainsErrors();
1206 }
1207 }
1208
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001209 private Builder beforeBuild() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001210 Preconditions.checkNotNull(pkg);
1211 Preconditions.checkNotNull(filename);
1212 Preconditions.checkNotNull(buildFileLabel);
1213 Preconditions.checkNotNull(makeEnv);
1214 // Freeze subincludes.
1215 subincludes = (subincludes == null)
1216 ? Collections.<Label, Path>emptyMap()
1217 : Collections.unmodifiableMap(subincludes);
1218
1219 // We create the original BUILD InputFile when the package filename is set; however, the
1220 // visibility may be overridden with an exports_files directive, so we need to obtain the
1221 // current instance here.
1222 buildFile = (InputFile) Preconditions.checkNotNull(targets.get(buildFileLabel.getName()));
1223
1224 List<Rule> rules = Lists.newArrayList(getTargets(Rule.class));
1225
1226 // All labels mentioned in a rule that refer to an unknown target in the
1227 // current package are assumed to be InputFiles, so let's create them:
1228 for (final Rule rule : rules) {
1229 AggregatingAttributeMapper.of(rule).visitLabels(new AcceptsLabelAttribute() {
1230 @Override
1231 public void acceptLabelAttribute(Label label, Attribute attribute) {
1232 createInputFileMaybe(label, rule.getAttributeLocation(attribute.getName()));
1233 }
1234 });
1235 }
1236
1237 // "test_suite" rules have the idiosyncratic semantics of implicitly
1238 // depending on all tests in the package, iff tests=[] and suites=[].
1239 // Note, we implement this here when the Package is fully constructed,
1240 // since clearly this information isn't available at Rule construction
1241 // time, as forward references are permitted.
1242 List<Label> allTests = new ArrayList<>();
1243 for (Rule rule : rules) {
Greg Estrenc48214d2015-03-17 22:43:19 +00001244 if (TargetUtils.isTestRule(rule) && !TargetUtils.hasManualTag(rule)) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001245 allTests.add(rule.getLabel());
1246 }
1247 }
Laurent Le Brund6ff51c2015-09-03 23:45:05 +00001248 Collections.sort(allTests);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001249 for (Rule rule : rules) {
1250 AttributeMap attributes = NonconfigurableAttributeMapper.of(rule);
1251 if (rule.getRuleClass().equals("test_suite")
Lukacs Berkiffa73ad2015-09-18 11:40:12 +00001252 && attributes.get("tests", BuildType.LABEL_LIST).isEmpty()
1253 && (!attributes.has("suites", BuildType.LABEL_LIST)
1254 || attributes.get("suites", BuildType.LABEL_LIST).isEmpty())) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001255 rule.setAttributeValueByName("$implicit_tests", allTests);
1256 }
1257 }
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001258 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001259 }
1260
Michajlo Matijkiw91407182015-05-29 14:27:09 +00001261 /** Intended for use by {@link com.google.devtools.build.lib.skyframe.PackageFunction} only. */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001262 public Builder buildPartial() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001263 if (alreadyBuilt) {
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001264 return this;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001265 }
1266 return beforeBuild();
1267 }
1268
Nathan Harmata6b253fa2016-05-24 15:27:18 +00001269 /**
1270 * Removes a target from the {@link Package} under construction. Intended to be used only by
1271 * {@link com.google.devtools.build.lib.skyframe.PackageFunction} to remove targets whose
1272 * labels cross subpackage boundaries.
1273 */
1274 public void removeTarget(Target target) {
1275 if (target.getPackage() == pkg) {
1276 this.targets.remove(target.getName());
1277 }
1278 }
1279
Michajlo Matijkiw91407182015-05-29 14:27:09 +00001280 /** Intended for use by {@link com.google.devtools.build.lib.skyframe.PackageFunction} only. */
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001281 public Package finishBuild() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001282 if (alreadyBuilt) {
1283 return pkg;
1284 }
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +00001285
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001286 // Freeze targets and distributions.
1287 targets = ImmutableMap.copyOf(targets);
1288 defaultDistributionSet =
1289 Collections.unmodifiableSet(defaultDistributionSet);
1290
Greg Estrend0f10dc2015-03-17 22:19:37 +00001291 // Now all targets have been loaded, so we validate the group's member environments.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001292 for (EnvironmentGroup envGroup : ImmutableSet.copyOf(environmentGroups.values())) {
Greg Estrend0f10dc2015-03-17 22:19:37 +00001293 Collection<Event> errors = envGroup.processMemberEnvironments(targets);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001294 if (!errors.isEmpty()) {
1295 addEvents(errors);
1296 setContainsErrors();
1297 }
1298 }
1299
1300 // Build the package.
1301 pkg.finishInit(this);
1302 alreadyBuilt = true;
1303 return pkg;
1304 }
1305
Damien Martin-Guillerez653df882016-02-17 21:46:22 +00001306 public ExternalPackageBuilder externalPackageData() {
Han-Wen Nienhuysf172d392015-09-18 17:37:51 +00001307 return externalPackageData;
1308 }
1309
Han-Wen Nienhuys086e5f22015-04-21 11:39:59 +00001310 public Package build() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001311 if (alreadyBuilt) {
1312 return pkg;
1313 }
1314 beforeBuild();
1315 return finishBuild();
1316 }
1317
1318 /**
1319 * If "label" refers to a non-existent target in the current package, create
1320 * an InputFile target.
1321 */
1322 void createInputFileMaybe(Label label, Location location) {
Damien Martin-Guillerez49978052016-01-21 19:55:07 +00001323 if (label != null && label.getPackageIdentifier().equals(pkg.getPackageIdentifier())) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001324 if (!targets.containsKey(label.getName())) {
1325 addInputFile(label, location);
1326 }
1327 }
1328 }
1329
1330 private InputFile addInputFile(Label label, Location location) {
1331 InputFile inputFile = new InputFile(pkg, label, location);
1332 Target prev = targets.put(label.getName(), inputFile);
1333 Preconditions.checkState(prev == null);
1334 return inputFile;
1335 }
1336
1337 /**
1338 * Precondition check for addRule. We must maintain these invariants of the
1339 * package:
1340 * - Each name refers to at most one target.
1341 * - No rule with errors is inserted into the package.
1342 * - The generating rule of every output file in the package must itself be
1343 * in the package.
1344 */
1345 private void checkForConflicts(Rule rule) throws NameConflictException {
1346 String name = rule.getName();
1347 Target existing = targets.get(name);
1348 if (existing != null) {
1349 throw nameConflict(rule, existing);
1350 }
1351 Map<String, OutputFile> outputFiles = new HashMap<>();
1352
1353 for (OutputFile outputFile : rule.getOutputFiles()) {
1354 String outputFileName = outputFile.getName();
1355 if (outputFiles.put(outputFileName, outputFile) != null) { // dups within a single rule:
1356 throw duplicateOutputFile(outputFile, outputFile);
1357 }
1358 existing = targets.get(outputFileName);
1359 if (existing != null) {
1360 throw duplicateOutputFile(outputFile, existing);
1361 }
1362
1363 // Check if this output file is the prefix of an already existing one
1364 if (outputFilePrefixes.containsKey(outputFileName)) {
1365 throw conflictingOutputFile(outputFile, outputFilePrefixes.get(outputFileName));
1366 }
1367
1368 // Check if a prefix of this output file matches an already existing one
1369 PathFragment outputFileFragment = new PathFragment(outputFileName);
1370 for (int i = 1; i < outputFileFragment.segmentCount(); i++) {
1371 String prefix = outputFileFragment.subFragment(0, i).toString();
1372 if (outputFiles.containsKey(prefix)) {
1373 throw conflictingOutputFile(outputFile, outputFiles.get(prefix));
1374 }
1375 if (targets.containsKey(prefix)
1376 && targets.get(prefix) instanceof OutputFile) {
1377 throw conflictingOutputFile(outputFile, (OutputFile) targets.get(prefix));
1378 }
1379
1380 if (!outputFilePrefixes.containsKey(prefix)) {
1381 outputFilePrefixes.put(prefix, outputFile);
1382 }
1383 }
1384 }
1385
1386 checkForInputOutputConflicts(rule, outputFiles.keySet());
1387 }
1388
1389 /**
1390 * A utility method that checks for conflicts between
1391 * input file names and output file names for a rule from a build
1392 * file.
1393 * @param rule the rule whose inputs and outputs are
1394 * to be checked for conflicts.
1395 * @param outputFiles a set containing the names of output
1396 * files to be generated by the rule.
1397 * @throws NameConflictException if a conflict is found.
1398 */
1399 private void checkForInputOutputConflicts(Rule rule, Set<String> outputFiles)
1400 throws NameConflictException {
1401 PathFragment packageFragment = rule.getLabel().getPackageFragment();
1402 for (Label inputLabel : rule.getLabels()) {
1403 if (packageFragment.equals(inputLabel.getPackageFragment())
1404 && outputFiles.contains(inputLabel.getName())) {
1405 throw inputOutputNameConflict(rule, inputLabel.getName());
1406 }
1407 }
1408 }
1409
1410 /** An output file conflicts with another output file or the BUILD file. */
1411 private NameConflictException duplicateOutputFile(OutputFile duplicate, Target existing) {
1412 return new NameConflictException(duplicate.getTargetKind() + " '" + duplicate.getName()
1413 + "' in rule '" + duplicate.getGeneratingRule().getName() + "' "
1414 + conflictsWith(existing));
1415 }
1416
1417 /** The package contains two targets with the same name. */
1418 private NameConflictException nameConflict(Target duplicate, Target existing) {
1419 return new NameConflictException(duplicate.getTargetKind() + " '" + duplicate.getName()
1420 + "' in package '" + duplicate.getLabel().getPackageName() + "' "
1421 + conflictsWith(existing));
1422 }
1423
1424 /** A a rule has a input/output name conflict. */
1425 private NameConflictException inputOutputNameConflict(Rule rule, String conflictingName) {
1426 return new NameConflictException("rule '" + rule.getName() + "' has file '"
1427 + conflictingName + "' as both an input and an output");
1428 }
1429
1430 private static NameConflictException conflictingOutputFile(
1431 OutputFile added, OutputFile existing) {
1432 if (added.getGeneratingRule() == existing.getGeneratingRule()) {
1433 return new NameConflictException(String.format(
1434 "rule '%s' has conflicting output files '%s' and '%s'", added.getGeneratingRule()
1435 .getName(), added.getName(), existing.getName()));
1436 } else {
1437 return new NameConflictException(String.format(
1438 "output file '%s' of rule '%s' conflicts with output file '%s' of rule '%s'", added
1439 .getName(), added.getGeneratingRule().getName(), existing.getName(), existing
1440 .getGeneratingRule().getName()));
1441 }
1442 }
1443
1444 /**
1445 * Utility function for generating exception messages.
1446 */
1447 private static String conflictsWith(Target target) {
1448 String message = "conflicts with existing ";
1449 if (target instanceof OutputFile) {
1450 return message + "generated file from rule '"
1451 + ((OutputFile) target).getGeneratingRule().getName()
1452 + "'";
1453 } else {
1454 return message + target.getTargetKind();
1455 }
1456 }
1457 }
1458}