blob: a2216e051b6abe5efb63af0d8dabf3b7baee7f27 [file] [log] [blame]
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01001// Copyright 2014 Google Inc. All rights reserved.
2//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package com.google.devtools.build.lib.packages;
16
17import com.google.common.annotations.VisibleForTesting;
18import com.google.common.base.Preconditions;
19import com.google.common.collect.ImmutableList;
20import com.google.common.collect.ImmutableMap;
21import com.google.common.collect.ImmutableSet;
22import com.google.common.collect.ImmutableSortedSet;
23import com.google.common.collect.Iterables;
24import com.google.common.collect.Lists;
25import com.google.common.collect.Maps;
26import com.google.common.collect.Sets;
27import com.google.devtools.build.lib.collect.CollectionUtils;
28import com.google.devtools.build.lib.collect.ImmutableSortedKeyMap;
29import com.google.devtools.build.lib.events.Event;
30import com.google.devtools.build.lib.events.EventHandler;
31import com.google.devtools.build.lib.events.Location;
32import com.google.devtools.build.lib.packages.AttributeMap.AcceptsLabelAttribute;
33import com.google.devtools.build.lib.packages.License.DistributionType;
34import com.google.devtools.build.lib.packages.PackageDeserializer.PackageDeserializationException;
35import com.google.devtools.build.lib.packages.PackageFactory.Globber;
36
37import com.google.devtools.build.lib.syntax.FuncallExpression;
38import com.google.devtools.build.lib.syntax.Label;
39import com.google.devtools.build.lib.syntax.Label.SyntaxException;
40import com.google.devtools.build.lib.util.Pair;
41import com.google.devtools.build.lib.vfs.Canonicalizer;
42import com.google.devtools.build.lib.vfs.Path;
43import com.google.devtools.build.lib.vfs.PathFragment;
44
45import java.io.IOException;
46import java.io.ObjectInputStream;
47import java.io.ObjectOutputStream;
48import java.io.PrintStream;
49import java.io.Serializable;
50import java.util.ArrayList;
51import java.util.Collection;
52import java.util.Collections;
53import java.util.HashMap;
54import java.util.List;
55import java.util.Map;
56import java.util.Set;
57
58/**
59 * A package, which is a container of {@link Rule}s, each of
60 * which contains a dictionary of named attributes.
61 *
62 * <p>Package instances are intended to be immutable and for all practical
63 * purposes can be treated as such. Note, however, that some member variables
64 * exposed via the public interface are not strictly immutable, so until their
65 * types are guaranteed immutable we're not applying the {@code @Immutable}
66 * annotation here.
67 */
68public class Package implements Serializable {
69
70 /**
71 * Common superclass for all name-conflict exceptions.
72 */
73 public static class NameConflictException extends Exception {
74 protected NameConflictException(String message) {
75 super(message);
76 }
77 }
78
79 /**
80 * The repository identifier for this package.
81 */
82 private final PackageIdentifier packageIdentifier;
83
84 /**
85 * The name of the package, e.g. "foo/bar".
86 */
87 protected final String name;
88
89 /**
90 * Like name, but in the form of a PathFragment.
91 */
92 private final PathFragment nameFragment;
93
94 /**
95 * The filename of this package's BUILD file.
96 */
97 protected Path filename;
98
99 /**
100 * The directory in which this package's BUILD file resides. All InputFile
101 * members of the packages are located relative to this directory.
102 */
103 private Path packageDirectory;
104
105 /**
106 * The root of the source tree in which this package was found. It is an invariant that
107 * {@code sourceRoot.getRelative(name).equals(packageDirectory)}.
108 */
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 /**
123 * Default visibility for rules that do not specify it. null is interpreted
124 * as VISIBILITY_PRIVATE.
125 */
126 private RuleVisibility defaultVisibility;
127 private boolean defaultVisibilitySet;
128
129 /**
130 * Default package-level 'obsolete' value for rules that do not specify it.
131 */
132 private boolean defaultObsolete = false;
133
134 /**
135 * Default package-level 'testonly' value for rules that do not specify it.
136 */
137 private boolean defaultTestOnly = false;
138
139 /**
140 * Default package-level 'deprecation' value for rules that do not specify it.
141 */
142 private String defaultDeprecation;
143
144 /**
145 * Default header strictness checking for rules that do not specify it.
146 */
147 private String defaultHdrsCheck;
148
149 /**
150 * Default copts for cc_* rules. The rules' individual copts will append to
151 * this value.
152 */
153 private ImmutableList<String> defaultCopts;
154
155 /**
156 * The InputFile target corresponding to this package's BUILD file.
157 */
158 private InputFile buildFile;
159
160 /**
161 * True iff this package's BUILD files contained lexical or grammatical
162 * errors, or experienced errors during evaluation, or semantic errors during
163 * the construction of any rule.
164 *
165 * <p>Note: A package containing errors does not necessarily prevent a build;
166 * if all the rules needed for a given build were constructed prior to the
167 * first error, the build may proceed.
168 */
169 private boolean containsErrors;
170
171 /**
172 * True iff this package contains errors that were caused by temporary conditions (e.g. an I/O
173 * error). If this is true, {@link #containsErrors} is also true.
174 */
175 private boolean containsTemporaryErrors;
176
177 /**
178 * The set of labels subincluded by this package.
179 */
180 private Set<Label> subincludes;
181
182 /**
183 * The list of transitive closure of the Skylark file dependencies.
184 */
185 private ImmutableList<Label> skylarkFileDependencies;
186
187 /**
188 * The package's default "licenses" and "distribs" attributes, as specified
189 * in calls to licenses() and distribs() in the BUILD file.
190 */
191 // These sets contain the values specified by the most recent licenses() or
192 // distribs() declarations encountered during package parsing:
193 private License defaultLicense;
194 private Set<License.DistributionType> defaultDistributionSet;
195
196
197 /**
198 * The names of the package() attributes that declare default values for rule
199 * {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR} and {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
200 * values when not explicitly specified.
201 */
202 public static final String DEFAULT_COMPATIBLE_WITH_ATTRIBUTE = "default_compatible_with";
203 public static final String DEFAULT_RESTRICTED_TO_ATTRIBUTE = "default_restricted_to";
204
205 private Set<Label> defaultCompatibleWith = ImmutableSet.of();
206 private Set<Label> defaultRestrictedTo = ImmutableSet.of();
207
208 private ImmutableSet<String> features;
209
210 private ImmutableList<Event> events;
211
212 // Hack to avoid having to copy every attribute. See #readObject and #readResolve.
213 // This will always be null for externally observable instances.
214 private Package deserializedPkg = null;
215
216 /**
217 * Package initialization, part 1 of 3: instantiates a new package with the
218 * given name.
219 *
220 * <p>As part of initialization, {@link Builder} constructs {@link InputFile}
221 * and {@link PackageGroup} instances that require a valid Package instance where
222 * {@link Package#getNameFragment()} is accessible. That's why these settings are
223 * applied here at the start.
224 *
225 * @precondition {@code name} must be a suffix of
226 * {@code filename.getParentDirectory())}.
227 */
228 protected Package(PackageIdentifier packageId) {
229 this.packageIdentifier = packageId;
230 this.nameFragment = Canonicalizer.fragments().intern(packageId.getPackageFragment());
231 this.name = nameFragment.getPathString();
232 }
233
234 private void writeObject(ObjectOutputStream out) {
235 com.google.devtools.build.lib.query2.proto.proto2api.Build.Package pb =
236 PackageSerializer.serializePackage(this);
237 try {
238 pb.writeDelimitedTo(out);
239 } catch (IOException e) {
240 throw new IllegalStateException(e);
241 }
242 }
243
244 private void readObject(ObjectInputStream in) throws IOException {
245 com.google.devtools.build.lib.query2.proto.proto2api.Build.Package pb =
246 com.google.devtools.build.lib.query2.proto.proto2api.Build.Package.parseDelimitedFrom(in);
247 Package pkg;
248 try {
249 pkg = new PackageDeserializer(null, null).deserialize(pb);
250 } catch (PackageDeserializationException e) {
251 throw new IllegalStateException(e);
252 }
253 deserializedPkg = pkg;
254 }
255
256 protected Object readResolve() {
257 // This method needs to be protected so serialization works for subclasses.
258 return deserializedPkg;
259 }
260
261 // See: http://docs.oracle.com/javase/6/docs/platform/serialization/spec/input.html#6053
262 @SuppressWarnings("unused")
263 private void readObjectNoData() {
264 throw new IllegalStateException();
265 }
266
267 /** Returns this packages' identifier. */
268 public PackageIdentifier getPackageIdentifier() {
269 return packageIdentifier;
270 }
271
272 /**
273 * Package initialization: part 2 of 3: sets this package's default header
274 * strictness checking.
275 *
276 * <p>This is needed to support C++-related rule classes
277 * which accesses {@link #getDefaultHdrsCheck} from the still-under-construction
278 * package.
279 */
280 protected void setDefaultHdrsCheck(String defaultHdrsCheck) {
281 this.defaultHdrsCheck = defaultHdrsCheck;
282 }
283
284 /**
285 * Set the default 'obsolete' value for this package.
286 */
287 protected void setDefaultObsolete(boolean obsolete) {
288 defaultObsolete = obsolete;
289 }
290
291 /**
292 * Set the default 'testonly' value for this package.
293 */
294 protected void setDefaultTestOnly(boolean testOnly) {
295 defaultTestOnly = testOnly;
296 }
297
298 /**
299 * Set the default 'deprecation' value for this package.
300 */
301 protected void setDefaultDeprecation(String deprecation) {
302 defaultDeprecation = deprecation;
303 }
304
305 /**
306 * Sets the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
307 * attribute when not explicitly specified by the rule.
308 */
309 protected void setDefaultCompatibleWith(Set<Label> environments) {
310 defaultCompatibleWith = environments;
311 }
312
313 /**
314 * Sets the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
315 * attribute when not explicitly specified by the rule.
316 */
317 protected void setDefaultRestrictedTo(Set<Label> environments) {
318 defaultRestrictedTo = environments;
319 }
320
321 public static Path getSourceRoot(Path buildFile, PathFragment nameFragment) {
322 Path current = buildFile.getParentDirectory();
323 for (int i = 0, len = nameFragment.segmentCount(); i < len && current != null; i++) {
324 current = current.getParentDirectory();
325 }
326 return current;
327 }
328
329 /**
330 * Package initialization: part 3 of 3: applies all other settings and completes
331 * initialization of the package.
332 *
333 * <p>Only after this method is called can this package be considered "complete"
334 * and be shared publicly.
335 */
336 protected void finishInit(AbstractBuilder<?, ?> builder) {
337 // If any error occurred during evaluation of this package, consider all
338 // rules in the package to be "in error" also (even if they were evaluated
339 // prior to the error). This behaviour is arguably stricter than need be,
340 // but stopping a build only for some errors but not others creates user
341 // confusion.
342 if (builder.containsErrors) {
343 for (Rule rule : builder.getTargets(Rule.class)) {
344 rule.setContainsErrors();
345 }
346 }
347 this.filename = builder.filename;
348 this.packageDirectory = filename.getParentDirectory();
349
350 this.sourceRoot = getSourceRoot(filename, nameFragment);
351 if ((sourceRoot == null
352 || !sourceRoot.getRelative(nameFragment).equals(packageDirectory))
353 && !filename.getBaseName().equals("WORKSPACE")) {
354 throw new IllegalArgumentException(
355 "Invalid BUILD file name for package '" + name + "': " + filename);
356 }
357
358 this.makeEnv = builder.makeEnv.build();
359 this.targets = ImmutableSortedKeyMap.copyOf(builder.targets);
360 this.defaultVisibility = builder.defaultVisibility;
361 this.defaultVisibilitySet = builder.defaultVisibilitySet;
362 if (builder.defaultCopts == null) {
363 this.defaultCopts = ImmutableList.of();
364 } else {
365 this.defaultCopts = ImmutableList.copyOf(builder.defaultCopts);
366 }
367 this.buildFile = builder.buildFile;
368 this.containsErrors = builder.containsErrors;
369 this.containsTemporaryErrors = builder.containsTemporaryErrors;
370 this.subincludes = builder.subincludes.keySet();
371 this.skylarkFileDependencies = builder.skylarkFileDependencies;
372 this.defaultLicense = builder.defaultLicense;
373 this.defaultDistributionSet = builder.defaultDistributionSet;
374 this.features = ImmutableSortedSet.copyOf(builder.features);
375 this.events = ImmutableList.copyOf(builder.events);
376 }
377
378 /**
379 * Returns the list of subincluded labels on which the validity of this package depends.
380 */
381 public Set<Label> getSubincludeLabels() {
382 return subincludes;
383 }
384
385 /**
386 * Returns the list of transitive closure of the Skylark file dependencies of this package.
387 */
388 public ImmutableList<Label> getSkylarkFileDependencies() {
389 return skylarkFileDependencies;
390 }
391
392 /**
393 * Returns the filename of the BUILD file which defines this package. The
394 * parent directory of the BUILD file is the package directory.
395 */
396 public Path getFilename() {
397 return filename;
398 }
399
400 /**
401 * Returns the source root (a directory) beneath which this package's BUILD file was found.
402 *
403 * Assumes invariant:
404 * {@code getSourceRoot().getRelative(getName()).equals(getPackageDirectory())}
405 */
406 public Path getSourceRoot() {
407 return sourceRoot;
408 }
409
410 /**
411 * Returns the directory containing the package's BUILD file.
412 */
413 public Path getPackageDirectory() {
414 return packageDirectory;
415 }
416
417 /**
418 * Returns the name of this package. If this build is using external repositories then this name
419 * may not be unique!
420 */
421 public String getName() {
422 return name;
423 }
424
425 /**
426 * Like {@link #getName}, but has type {@code PathFragment}.
427 */
428 public PathFragment getNameFragment() {
429 return nameFragment;
430 }
431
432 /**
433 * Returns the "Make" value from the package's make environment whose name
434 * is "varname", or null iff the variable is not defined in the environment.
435 */
436 public String lookupMakeVariable(String varname, String platform) {
437 return makeEnv.lookup(varname, platform);
438 }
439
440 /**
441 * Returns the make environment. This should only ever be used for serialization -- how the
442 * make variables are implemented is an implementation detail.
443 */
444 MakeEnvironment getMakeEnvironment() {
445 return makeEnv;
446 }
447
448 /**
449 * Returns the label of this package's BUILD file.
450 *
451 * Typically <code>getBuildFileLabel().getName().equals("BUILD")</code> --
452 * though not necessarily: data in a subdirectory of a test package may use a
453 * different filename to avoid inadvertently creating a new package.
454 */
455 Label getBuildFileLabel() {
456 return buildFile.getLabel();
457 }
458
459 /**
460 * Returns the InputFile target for this package's BUILD file.
461 */
462 public InputFile getBuildFile() {
463 return buildFile;
464 }
465
466 /**
467 * Returns true if errors were encountered during evaluation of this package.
468 * (The package may be incomplete and its contents should not be relied upon
469 * for critical operations. However, any Rules belonging to the package are
470 * guaranteed to be intact, unless their <code>containsErrors()</code> flag
471 * is set.)
472 */
473 public boolean containsErrors() {
474 return containsErrors;
475 }
476
477 /**
478 * True iff this package contains errors that were caused by temporary conditions (e.g. an I/O
479 * error). If this is true, {@link #containsErrors()} also returns true.
480 */
481 public boolean containsTemporaryErrors() {
482 return containsTemporaryErrors;
483 }
484
485 public List<Event> getEvents() {
486 return events;
487 }
488
489 /**
490 * Returns an (immutable, unordered) view of all the targets belonging to this package.
491 */
492 public Collection<Target> getTargets() {
493 return getTargets(targets);
494 }
495
496 /**
497 * Common getTargets implementation, accessible by both {@link Package} and
498 * {@link Package.AbstractBuilder}.
499 */
500 private static Collection<Target> getTargets(Map<String, Target> targetMap) {
501 return Collections.unmodifiableCollection(targetMap.values());
502 }
503
504 /**
505 * Returns a (read-only, unordered) iterator of all the targets belonging
506 * to this package which are instances of the specified class.
507 */
508 public <T extends Target> Iterable<T> getTargets(Class<T> targetClass) {
509 return getTargets(targets, targetClass);
510 }
511
512 /**
513 * Common getTargets implementation, accessible by both {@link Package} and
514 * {@link Package.AbstractBuilder}.
515 */
516 private static <T extends Target> Iterable<T> getTargets(Map<String, Target> targetMap,
517 Class<T> targetClass) {
518 return Iterables.filter(targetMap.values(), targetClass);
519 }
520
521 /**
522 * Returns a (read-only, unordered) iterator over the rules in this package.
523 */
524 @VisibleForTesting // Legacy. Production code should use getTargets(Class) instead
525 Iterable<? extends Rule> getRules() {
526 return getTargets(Rule.class);
527 }
528
529 /**
530 * Returns a (read-only, unordered) iterator over the files in this package.
531 */
532 @VisibleForTesting // Legacy. Production code should use getTargets(Class) instead
533 Iterable<? extends FileTarget> getFiles() {
534 return getTargets(FileTarget.class);
535 }
536
537 /**
538 * Returns the rule that corresponds to a particular BUILD target name. Useful
539 * for walking through the dependency graph of a target.
540 * Fails if the target is not a Rule.
541 */
542 @VisibleForTesting
543 Rule getRule(String targetName) {
544 return (Rule) targets.get(targetName);
545 }
546
547 /**
548 * Returns the features specified in the <code>package()</code> declaration.
549 */
550 public ImmutableSet<String> getFeatures() {
551 return features;
552 }
553
554 /**
555 * Returns the target (a member of this package) whose name is "targetName".
556 * First rules are searched, then output files, then input files. The target
557 * name must be valid, as defined by {@code LabelValidator#validateTargetName}.
558 *
559 * @throws NoSuchTargetException if the specified target was not found.
560 */
561 public Target getTarget(String targetName) throws NoSuchTargetException {
562 Target target = targets.get(targetName);
563 if (target != null) {
564 return target;
565 }
566
567 // No such target.
568
569 // If there's a file on the disk that's not mentioned in the BUILD file,
570 // produce a more informative error. NOTE! this code path is only executed
571 // on failure, which is (relatively) very rare. In the common case no
572 // stat(2) is executed.
573 Path filename = getPackageDirectory().getRelative(targetName);
574 String suffix;
575 if (!new PathFragment(targetName).isNormalized()) {
576 // Don't check for file existence in this case because the error message
577 // would be confusing and wrong. If the targetName is "foo/bar/.", and
578 // there is a directory "foo/bar", it doesn't mean that "//pkg:foo/bar/."
579 // is a valid label.
580 suffix = "";
581 } else if (filename.isDirectory()) {
582 suffix = "; however, a source directory of this name exists. (Perhaps add "
583 + "'exports_files([\"" + targetName + "\"])' to " + name + "/BUILD, or define a "
584 + "filegroup?)";
585 } else if (filename.exists()) {
586 suffix = "; however, a source file of this name exists. (Perhaps add "
587 + "'exports_files([\"" + targetName + "\"])' to " + name + "/BUILD?)";
588 } else {
589 suffix = "";
590 }
591
592 try {
593 throw new NoSuchTargetException(createLabel(targetName), "target '" + targetName
594 + "' not declared in package '" + name + "'" + suffix + " defined by "
595 + this.filename);
596 } catch (Label.SyntaxException e) {
597 throw new IllegalArgumentException(targetName);
598 }
599 }
600
601 /**
602 * Creates a label for a target inside this package.
603 *
604 * @throws SyntaxException if the {@code targetName} is invalid
605 */
606 public Label createLabel(String targetName) throws SyntaxException {
607 return Label.create(packageIdentifier, targetName);
608 }
609
610 /**
611 * Returns the default visibility for this package.
612 */
613 public RuleVisibility getDefaultVisibility() {
614 if (defaultVisibility != null) {
615 return defaultVisibility;
616 } else {
617 return ConstantRuleVisibility.PRIVATE;
618 }
619 }
620
621 /**
622 * Returns the default obsolete value.
623 */
624 public Boolean getDefaultObsolete() {
625 return defaultObsolete;
626 }
627
628 /**
629 * Returns the default testonly value.
630 */
631 public Boolean getDefaultTestOnly() {
632 return defaultTestOnly;
633 }
634
635 /**
636 * Returns the default obsolete value.
637 */
638 public String getDefaultDeprecation() {
639 return defaultDeprecation;
640 }
641
642 /**
643 * Gets the default header checking mode.
644 */
645 public String getDefaultHdrsCheck() {
646 return defaultHdrsCheck != null ? defaultHdrsCheck : "loose";
647 }
648
649 /**
650 * Returns the default copts value, to which rules should append their
651 * specific copts.
652 */
653 public ImmutableList<String> getDefaultCopts() {
654 return defaultCopts;
655 }
656
657 /**
658 * Returns whether the default header checking mode has been set or it is the
659 * default value.
660 */
661 public boolean isDefaultHdrsCheckSet() {
662 return defaultHdrsCheck != null;
663 }
664
665 public boolean isDefaultVisibilitySet() {
666 return defaultVisibilitySet;
667 }
668
669 /**
670 * Gets the parsed license object for the default license
671 * declared by this package.
672 */
673 public License getDefaultLicense() {
674 return defaultLicense;
675 }
676
677 /**
678 * Returns the parsed set of distributions declared as the default for this
679 * package.
680 */
681 public Set<License.DistributionType> getDefaultDistribs() {
682 return defaultDistributionSet;
683 }
684
685 /**
686 * Returns the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
687 * attribute when not explicitly specified by the rule.
688 */
689 public Set<Label> getDefaultCompatibleWith() {
690 return defaultCompatibleWith;
691 }
692
693 /**
694 * Returns the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
695 * attribute when not explicitly specified by the rule.
696 */
697 public Set<Label> getDefaultRestrictedTo() {
698 return defaultRestrictedTo;
699 }
700
701 @Override
702 public String toString() {
703 return "Package(" + name + ")=" + (targets != null ? getRules() : "initializing...");
704 }
705
706 /**
707 * Dumps the package for debugging. Do not depend on the exact format/contents of this debugging
708 * output.
709 */
710 public void dump(PrintStream out) {
711 out.println(" Package " + getName() + " (" + getFilename() + ")");
712
713 // Rules:
714 out.println(" Rules");
715 for (Rule rule : getTargets(Rule.class)) {
716 out.println(" " + rule.getTargetKind() + " " + rule.getLabel());
717 for (Attribute attr : rule.getAttributes()) {
718 for (Object possibleValue : AggregatingAttributeMapper.of(rule)
719 .visitAttribute(attr.getName(), attr.getType())) {
720 out.println(" " + attr.getName() + " = " + possibleValue);
721 }
722 }
723 }
724
725 // Files:
726 out.println(" Files");
727 for (FileTarget file : getTargets(FileTarget.class)) {
728 out.print(" " + file.getTargetKind() + " " + file.getLabel());
729 if (file instanceof OutputFile) {
730 out.println(" (generated by " + ((OutputFile) file).getGeneratingRule().getLabel() + ")");
731 } else {
732 out.println();
733 }
734 }
735
736 // TODO(bazel-team): (2009) perhaps dump also:
737 // - subincludes
738 // - globs
739 // - containsErrors
740 // - makeEnv
741 }
742
743 /**
744 * Builder class for {@link Package}.
745 *
746 * <p>Should only be used by the package loading and the package deserialization machineries.
747 */
748 static class Builder extends AbstractBuilder<Package, Builder> {
749 Builder(PackageIdentifier packageId) {
750 super(new Package(packageId));
751 }
752
753 @Override
754 protected Builder self() {
755 return this;
756 }
757 }
758
759 /** Builder class for {@link Package} that does its own globbing. */
760 public static class LegacyBuilder extends AbstractBuilder<Package, LegacyBuilder> {
761
762 private Globber globber = null;
763
764 LegacyBuilder(PackageIdentifier packageId) {
765 super(AbstractBuilder.newPackage(packageId));
766 }
767
768 @Override
769 protected LegacyBuilder self() {
770 return this;
771 }
772
773 /**
774 * Sets the globber used for this package's glob expansions.
775 */
776 LegacyBuilder setGlobber(Globber globber) {
777 this.globber = globber;
778 return this;
779 }
780
781 /**
782 * Removes a target from the {@link Package} under construction. Intended to be used only by
783 * {@link PackageFunction} to remove targets whose labels cross subpackage boundaries.
784 */
785 public void removeTarget(Target target) {
786 if (target.getPackage() == pkg) {
787 this.targets.remove(target.getName());
788 }
789 }
790
791 /**
792 * Returns the glob patterns requested by {@link PackageFactory} during evaluation of this
793 * package's BUILD file. Intended to be used only by {@link PackageFunction} to mark the
794 * appropriate Skyframe dependencies after the fact.
795 */
796 public Set<Pair<String, Boolean>> getGlobPatterns() {
797 return globber.getGlobPatterns();
798 }
799 }
800
801 abstract static class AbstractBuilder<P extends Package, B extends AbstractBuilder<P, B>> {
802 /**
803 * The output instance for this builder. Needs to be instantiated and
804 * available with name info throughout initialization. All other settings
805 * are applied during {@link #build}. See {@link Package#Package(String)}
806 * and {@link Package#finishInit} for details.
807 */
808 protected P pkg;
809
810 protected Path filename = null;
811 private Label buildFileLabel = null;
812 private InputFile buildFile = null;
813 private MakeEnvironment.Builder makeEnv = null;
814 private RuleVisibility defaultVisibility = null;
815 private boolean defaultVisibilitySet;
816 private List<String> defaultCopts = null;
817 private List<String> features = new ArrayList<>();
818 private List<Event> events = Lists.newArrayList();
819 private boolean containsErrors = false;
820 private boolean containsTemporaryErrors = false;
821
822 private License defaultLicense = License.NO_LICENSE;
823 private Set<License.DistributionType> defaultDistributionSet = License.DEFAULT_DISTRIB;
824
825 protected Map<String, Target> targets = new HashMap<>();
826 protected Map<Label, EnvironmentGroup> environmentGroups = new HashMap<>();
827
828 protected Map<Label, Path> subincludes = null;
829 protected ImmutableList<Label> skylarkFileDependencies = null;
830
831 /**
832 * True iff the "package" function has already been called in this package.
833 */
834 private boolean packageFunctionUsed;
835
836 /**
837 * The collection of the prefixes of every output file. Maps every prefix
838 * to an output file whose prefix it is.
839 *
840 * <p>This is needed to make the output file prefix conflict check be
841 * reasonably fast. However, since it can potentially take a lot of memory and
842 * is useless after the package has been loaded, it isn't passed to the
843 * package itself.
844 */
845 private Map<String, OutputFile> outputFilePrefixes = new HashMap<>();
846
847 private boolean alreadyBuilt = false;
848
849 private EventHandler builderEventHandler = new EventHandler() {
850 @Override
851 public void handle(Event event) {
852 addEvent(event);
853 }
854 };
855
856 protected AbstractBuilder(P pkg) {
857 this.pkg = pkg;
858 if (pkg.getName().startsWith("javatests/")) {
859 setDefaultTestonly(true);
860 }
861 }
862
863 protected static Package newPackage(PackageIdentifier packageId) {
864 return new Package(packageId);
865 }
866
867 protected abstract B self();
868
869 protected PackageIdentifier getPackageIdentifier() {
870 return pkg.getPackageIdentifier();
871 }
872
873 /**
874 * Sets the name of this package's BUILD file.
875 */
876 B setFilename(Path filename) {
877 this.filename = filename;
878 try {
879 buildFileLabel = createLabel(filename.getBaseName());
880 addInputFile(buildFileLabel, Location.fromFile(filename));
881 } catch (Label.SyntaxException e) {
882 // This can't actually happen.
883 throw new AssertionError("Package BUILD file has an illegal name: " + filename);
884 }
885 return self();
886 }
887
888 public Label getBuildFileLabel() {
889 return buildFileLabel;
890 }
891
892 Path getFilename() {
893 return filename;
894 }
895
896 /**
897 * Sets this package's Make environment.
898 */
899 B setMakeEnv(MakeEnvironment.Builder makeEnv) {
900 this.makeEnv = makeEnv;
901 return self();
902 }
903
904 /**
905 * Sets the default visibility for this package. Called at most once per
906 * package from PackageFactory.
907 */
908 B setDefaultVisibility(RuleVisibility visibility) {
909 this.defaultVisibility = visibility;
910 this.defaultVisibilitySet = true;
911 return self();
912 }
913
914 /**
915 * Sets whether the default visibility is set in the BUILD file.
916 */
917 B setDefaultVisibilitySet(boolean defaultVisibilitySet) {
918 this.defaultVisibilitySet = defaultVisibilitySet;
919 return self();
920 }
921
922 /**
923 * Sets the default value of 'obsolete'. Rule-level 'obsolete' will override this.
924 */
925 B setDefaultObsolete(boolean defaultObsolete) {
926 pkg.setDefaultObsolete(defaultObsolete);
927 return self();
928 }
929
930 /** Sets the default value of 'testonly'. Rule-level 'testonly' will override this. */
931 B setDefaultTestonly(boolean defaultTestonly) {
932 pkg.setDefaultTestOnly(defaultTestonly);
933 return self();
934 }
935
936 /**
937 * Sets the default value of 'deprecation'. Rule-level 'deprecation' will append to this.
938 */
939 B setDefaultDeprecation(String defaultDeprecation) {
940 pkg.setDefaultDeprecation(defaultDeprecation);
941 return self();
942 }
943
944 /**
945 * Returns whether the "package" function has been called yet
946 */
947 public boolean isPackageFunctionUsed() {
948 return packageFunctionUsed;
949 }
950
951 public void setPackageFunctionUsed() {
952 packageFunctionUsed = true;
953 }
954
955 /**
956 * Sets the default header checking mode.
957 */
958 public B setDefaultHdrsCheck(String hdrsCheck) {
959 // Note that this setting is propagated directly to the package because
960 // other code needs the ability to read this info directly from the
961 // under-construction package. See {@link Package#setDefaultHdrsCheck}.
962 pkg.setDefaultHdrsCheck(hdrsCheck);
963 return self();
964 }
965
966 /**
967 * Sets the default value of copts. Rule-level copts will append to this.
968 */
969 public B setDefaultCopts(List<String> defaultCopts) {
970 this.defaultCopts = defaultCopts;
971 return self();
972 }
973
974 public B addFeatures(Iterable<String> features) {
975 Iterables.addAll(this.features, features);
976 return self();
977 }
978
979 /**
980 * Declares that errors were encountering while loading this package.
981 */
982 public B setContainsErrors() {
983 containsErrors = true;
984 return self();
985 }
986
987 public boolean containsErrors() {
988 return containsErrors;
989 }
990
991 B setContainsTemporaryErrors() {
992 setContainsErrors();
993 containsTemporaryErrors = true;
994 return self();
995 }
996
997 public B addEvents(Iterable<Event> events) {
998 for (Event event : events) {
999 addEvent(event);
1000 }
1001 return self();
1002 }
1003
1004 public B addEvent(Event event) {
1005 this.events.add(event);
1006 return self();
1007 }
1008
1009 B setSkylarkFileDependencies(ImmutableList<Label> skylarkFileDependencies) {
1010 this.skylarkFileDependencies = skylarkFileDependencies;
1011 return self();
1012 }
1013
1014 /**
1015 * Sets the default license for this package.
1016 */
1017 void setDefaultLicense(License license) {
1018 this.defaultLicense = license;
1019 }
1020
1021 License getDefaultLicense() {
1022 return defaultLicense;
1023 }
1024
1025 /**
1026 * Initializes the default set of distributions for targets in this package.
1027 *
1028 * TODO(bazel-team): (2011) consider moving the license & distribs info into Metadata--maybe
1029 * even in the Build language.
1030 */
1031 void setDefaultDistribs(Set<DistributionType> dists) {
1032 this.defaultDistributionSet = dists;
1033 }
1034
1035 Set<DistributionType> getDefaultDistribs() {
1036 return defaultDistributionSet;
1037 }
1038
1039 /**
1040 * Sets the default value to use for a rule's {@link RuleClass#COMPATIBLE_ENVIRONMENT_ATTR}
1041 * attribute when not explicitly specified by the rule. Records a package error if
1042 * any labels are duplicated.
1043 */
1044 void setDefaultCompatibleWith(List<Label> environments, String attrName, Location location) {
1045 if (!checkForDuplicateLabels(environments, "package " + pkg.getName(), attrName, location,
1046 builderEventHandler)) {
1047 setContainsErrors();
1048 }
1049 pkg.setDefaultCompatibleWith(ImmutableSet.copyOf(environments));
1050 }
1051
1052 /**
1053 * Sets the default value to use for a rule's {@link RuleClass#RESTRICTED_ENVIRONMENT_ATTR}
1054 * attribute when not explicitly specified by the rule. Records a package error if
1055 * any labels are duplicated.
1056 */
1057 void setDefaultRestrictedTo(List<Label> environments, String attrName, Location location) {
1058 if (!checkForDuplicateLabels(environments, "package " + pkg.getName(), attrName, location,
1059 builderEventHandler)) {
1060 setContainsErrors();
1061 }
1062
1063 pkg.setDefaultRestrictedTo(ImmutableSet.copyOf(environments));
1064 }
1065
1066 /**
1067 * Returns a new Rule belonging to this package instance, and uses the given Label.
1068 *
1069 * <p>Useful for RuleClass instantiation, where the rule name is checked by trying to create a
1070 * Label. This label can then be used again here.
1071 */
1072 Rule newRuleWithLabel(Label label, RuleClass ruleClass, FuncallExpression ast,
1073 Location location) {
1074 return new Rule(pkg, label, ruleClass, ast, location);
1075 }
1076
1077 /**
1078 * Called by the parser when a "mocksubinclude" is encountered, to record the
1079 * mappings from labels to absolute paths upon which that the validity of
1080 * this package depends.
1081 */
1082 void addSubinclude(Label label, Path resolvedPath) {
1083 if (subincludes == null) {
1084 // This is a TreeMap because the order needs to be deterministic.
1085 subincludes = Maps.newTreeMap();
1086 }
1087
1088 Path oldResolvedPath = subincludes.put(label, resolvedPath);
1089 if (oldResolvedPath != null && !oldResolvedPath.equals(resolvedPath)){
1090 // The same label should have been resolved to the same path
1091 throw new IllegalStateException("Ambiguous subinclude path");
1092 }
1093 }
1094
1095 public Set<Label> getSubincludeLabels() {
1096 return subincludes == null ? Sets.<Label>newHashSet() : subincludes.keySet();
1097 }
1098
1099 public Map<Label, Path> getSubincludes() {
1100 return subincludes == null ? Maps.<Label, Path>newHashMap() : subincludes;
1101 }
1102
1103 public Collection<Target> getTargets() {
1104 return Package.getTargets(targets);
1105 }
1106
1107 /**
1108 * Returns an (immutable, unordered) view of all the targets belonging to
1109 * this package which are instances of the specified class.
1110 */
1111 <T extends Target> Iterable<T> getTargets(Class<T> targetClass) {
1112 return Package.getTargets(targets, targetClass);
1113 }
1114
1115 /**
1116 * An input file name conflicts with an existing package member.
1117 */
1118 static class GeneratedLabelConflict extends NameConflictException {
1119 private GeneratedLabelConflict(String message) {
1120 super(message);
1121 }
1122 }
1123
1124 /**
1125 * Creates an input file target in this package with the specified name.
1126 *
1127 * @param targetName name of the input file. This must be a valid target
1128 * name as defined by {@link
1129 * com.google.devtools.build.lib.cmdline.LabelValidator#validateTargetName}.
1130 * @return the newly-created InputFile, or the old one if it already existed.
1131 * @throws GeneratedLabelConflict if the name was already taken by a Rule or
1132 * an OutputFile target.
1133 * @throws IllegalArgumentException if the name is not a valid label
1134 */
1135 InputFile createInputFile(String targetName, Location location)
1136 throws GeneratedLabelConflict {
1137 Target existing = targets.get(targetName);
1138 if (existing == null) {
1139 try {
1140 return addInputFile(createLabel(targetName), location);
1141 } catch (Label.SyntaxException e) {
1142 throw new IllegalArgumentException("FileTarget in package " + pkg.getName()
1143 + " has illegal name: " + targetName);
1144 }
1145 } else if (existing instanceof InputFile) {
1146 return (InputFile) existing; // idempotent
1147 } else {
1148 throw new GeneratedLabelConflict("generated label '//" + pkg.getName() + ":"
1149 + targetName + "' conflicts with existing "
1150 + existing.getTargetKind());
1151 }
1152 }
1153
1154 /**
1155 * Sets the visibility and license for an input file. The input file must already exist as
1156 * a member of this package.
1157 * @throws IllegalArgumentException if the input file doesn't exist in this
1158 * package's target map.
1159 */
1160 void setVisibilityAndLicense(InputFile inputFile, RuleVisibility visibility, License license) {
1161 String filename = inputFile.getName();
1162 Target cacheInstance = targets.get(filename);
1163 if (cacheInstance == null || !(cacheInstance instanceof InputFile)) {
1164 throw new IllegalArgumentException("Can't set visibility for nonexistent FileTarget "
1165 + filename + " in package " + pkg.getName() + ".");
1166 }
1167 if (!((InputFile) cacheInstance).isVisibilitySpecified()
1168 || cacheInstance.getVisibility() != visibility
1169 || cacheInstance.getLicense() != license) {
1170 targets.put(filename, new InputFile(
1171 pkg, cacheInstance.getLabel(), cacheInstance.getLocation(), visibility, license));
1172 }
1173 }
1174
1175 /**
1176 * Creates a label for a target inside this package.
1177 *
1178 * @throws SyntaxException if the {@code targetName} is invalid
1179 */
1180 Label createLabel(String targetName) throws SyntaxException {
1181 return Label.create(pkg.getPackageIdentifier(), targetName);
1182 }
1183
1184 /**
1185 * Adds a package group to the package.
1186 */
1187 void addPackageGroup(String name, Collection<String> packages, Collection<Label> includes,
1188 EventHandler eventHandler, Location location)
1189 throws NameConflictException, Label.SyntaxException {
1190 PackageGroup group =
1191 new PackageGroup(createLabel(name), pkg, packages, includes, eventHandler, location);
1192 Target existing = targets.get(group.getName());
1193 if (existing != null) {
1194 throw nameConflict(group, existing);
1195 }
1196
1197 targets.put(group.getName(), group);
1198
1199 if (group.containsErrors()) {
1200 setContainsErrors();
1201 }
1202 }
1203
1204 /**
1205 * Checks if any labels in the given list appear multiple times and reports an appropriate
1206 * error message if so. Returns true if no duplicates were found, false otherwise.
1207 *
1208 * TODO(bazel-team): apply this to all build functions (maybe automatically?), possibly
1209 * integrate with RuleClass.checkForDuplicateLabels.
1210 */
1211 private static boolean checkForDuplicateLabels(Collection<Label> labels, String owner,
1212 String attrName, Location location, EventHandler eventHandler) {
1213 Set<Label> dupes = CollectionUtils.duplicatedElementsOf(labels);
1214 for (Label dupe : dupes) {
1215 eventHandler.handle(Event.error(location, String.format(
1216 "label '%s' is duplicated in the '%s' list of '%s'", dupe, attrName, owner)));
1217 }
1218 return dupes.isEmpty();
1219 }
1220
1221 /**
1222 * Adds an environment group to the package.
1223 */
1224 void addEnvironmentGroup(String name, List<Label> environments, List<Label> defaults,
1225 EventHandler eventHandler, Location location)
1226 throws NameConflictException, SyntaxException {
1227
1228 if (!checkForDuplicateLabels(environments, name, "environments", location, eventHandler)
1229 || !checkForDuplicateLabels(defaults, name, "defaults", location, eventHandler)) {
1230 setContainsErrors();
1231 return;
1232 }
1233
1234 EnvironmentGroup group = new EnvironmentGroup(createLabel(name), pkg, environments,
1235 defaults, location);
1236 Target existing = targets.get(group.getName());
1237 if (existing != null) {
1238 throw nameConflict(group, existing);
1239 }
1240
1241 targets.put(group.getName(), group);
1242 Collection<Event> membershipErrors = group.validateMembership();
1243 if (!membershipErrors.isEmpty()) {
1244 for (Event error : membershipErrors) {
1245 eventHandler.handle(error);
1246 }
1247 setContainsErrors();
1248 return;
1249 }
1250
1251 // For each declared environment, make sure it doesn't also belong to some other group.
1252 for (Label environment : group.getEnvironments()) {
1253 EnvironmentGroup otherGroup = environmentGroups.get(environment);
1254 if (otherGroup != null) {
1255 eventHandler.handle(Event.error(location, "environment " + environment + " belongs to"
1256 + " both " + group.getLabel() + " and " + otherGroup.getLabel()));
1257 setContainsErrors();
1258 } else {
1259 environmentGroups.put(environment, group);
1260 }
1261 }
1262 }
1263
1264 void addRule(Rule rule) throws NameConflictException {
1265 checkForConflicts(rule);
1266 // Now, modify the package:
1267 for (OutputFile outputFile : rule.getOutputFiles()) {
1268 targets.put(outputFile.getName(), outputFile);
1269 PathFragment outputFileFragment = new PathFragment(outputFile.getName());
1270 for (int i = 1; i < outputFileFragment.segmentCount(); i++) {
1271 String prefix = outputFileFragment.subFragment(0, i).toString();
1272 if (!outputFilePrefixes.containsKey(prefix)) {
1273 outputFilePrefixes.put(prefix, outputFile);
1274 }
1275 }
1276 }
1277 targets.put(rule.getName(), rule);
1278 if (rule.containsErrors()) {
1279 this.setContainsErrors();
1280 }
1281 }
1282
1283 private B beforeBuild() {
1284 Preconditions.checkNotNull(pkg);
1285 Preconditions.checkNotNull(filename);
1286 Preconditions.checkNotNull(buildFileLabel);
1287 Preconditions.checkNotNull(makeEnv);
1288 // Freeze subincludes.
1289 subincludes = (subincludes == null)
1290 ? Collections.<Label, Path>emptyMap()
1291 : Collections.unmodifiableMap(subincludes);
1292
1293 // We create the original BUILD InputFile when the package filename is set; however, the
1294 // visibility may be overridden with an exports_files directive, so we need to obtain the
1295 // current instance here.
1296 buildFile = (InputFile) Preconditions.checkNotNull(targets.get(buildFileLabel.getName()));
1297
1298 List<Rule> rules = Lists.newArrayList(getTargets(Rule.class));
1299
1300 // All labels mentioned in a rule that refer to an unknown target in the
1301 // current package are assumed to be InputFiles, so let's create them:
1302 for (final Rule rule : rules) {
1303 AggregatingAttributeMapper.of(rule).visitLabels(new AcceptsLabelAttribute() {
1304 @Override
1305 public void acceptLabelAttribute(Label label, Attribute attribute) {
1306 createInputFileMaybe(label, rule.getAttributeLocation(attribute.getName()));
1307 }
1308 });
1309 }
1310
1311 // "test_suite" rules have the idiosyncratic semantics of implicitly
1312 // depending on all tests in the package, iff tests=[] and suites=[].
1313 // Note, we implement this here when the Package is fully constructed,
1314 // since clearly this information isn't available at Rule construction
1315 // time, as forward references are permitted.
1316 List<Label> allTests = new ArrayList<>();
1317 for (Rule rule : rules) {
1318 if (TargetUtils.isTestRule(rule) && !TargetUtils.hasManualTag(rule)
1319 && !TargetUtils.isObsolete(rule)) {
1320 allTests.add(rule.getLabel());
1321 }
1322 }
1323 for (Rule rule : rules) {
1324 AttributeMap attributes = NonconfigurableAttributeMapper.of(rule);
1325 if (rule.getRuleClass().equals("test_suite")
1326 && attributes.get("tests", Type.LABEL_LIST).isEmpty()
1327 && attributes.get("suites", Type.LABEL_LIST).isEmpty()) {
1328 rule.setAttributeValueByName("$implicit_tests", allTests);
1329 }
1330 }
1331 return self();
1332 }
1333
1334 /** Intended to be used only by {@link PackageFunction}. */
1335 public B buildPartial() {
1336 if (alreadyBuilt) {
1337 return self();
1338 }
1339 return beforeBuild();
1340 }
1341
1342 /** Intended to be used only by {@link PackageFunction}. */
1343 public P finishBuild() {
1344 if (alreadyBuilt) {
1345 return pkg;
1346 }
1347 // Freeze targets and distributions.
1348 targets = ImmutableMap.copyOf(targets);
1349 defaultDistributionSet =
1350 Collections.unmodifiableSet(defaultDistributionSet);
1351
1352 // Now all targets have been loaded, so we can check all declared environments in an
1353 // environment group exist.
1354 for (EnvironmentGroup envGroup : ImmutableSet.copyOf(environmentGroups.values())) {
1355 Collection<Event> errors = envGroup.checkEnvironmentsExist(targets);
1356 if (!errors.isEmpty()) {
1357 addEvents(errors);
1358 setContainsErrors();
1359 }
1360 }
1361
1362 // Build the package.
1363 pkg.finishInit(this);
1364 alreadyBuilt = true;
1365 return pkg;
1366 }
1367
1368 public P build() {
1369 if (alreadyBuilt) {
1370 return pkg;
1371 }
1372 beforeBuild();
1373 return finishBuild();
1374 }
1375
1376 /**
1377 * If "label" refers to a non-existent target in the current package, create
1378 * an InputFile target.
1379 */
1380 void createInputFileMaybe(Label label, Location location) {
1381 if (label != null && label.getPackageFragment().equals(pkg.getNameFragment())) {
1382 if (!targets.containsKey(label.getName())) {
1383 addInputFile(label, location);
1384 }
1385 }
1386 }
1387
1388 private InputFile addInputFile(Label label, Location location) {
1389 InputFile inputFile = new InputFile(pkg, label, location);
1390 Target prev = targets.put(label.getName(), inputFile);
1391 Preconditions.checkState(prev == null);
1392 return inputFile;
1393 }
1394
1395 /**
1396 * Precondition check for addRule. We must maintain these invariants of the
1397 * package:
1398 * - Each name refers to at most one target.
1399 * - No rule with errors is inserted into the package.
1400 * - The generating rule of every output file in the package must itself be
1401 * in the package.
1402 */
1403 private void checkForConflicts(Rule rule) throws NameConflictException {
1404 String name = rule.getName();
1405 Target existing = targets.get(name);
1406 if (existing != null) {
1407 throw nameConflict(rule, existing);
1408 }
1409 Map<String, OutputFile> outputFiles = new HashMap<>();
1410
1411 for (OutputFile outputFile : rule.getOutputFiles()) {
1412 String outputFileName = outputFile.getName();
1413 if (outputFiles.put(outputFileName, outputFile) != null) { // dups within a single rule:
1414 throw duplicateOutputFile(outputFile, outputFile);
1415 }
1416 existing = targets.get(outputFileName);
1417 if (existing != null) {
1418 throw duplicateOutputFile(outputFile, existing);
1419 }
1420
1421 // Check if this output file is the prefix of an already existing one
1422 if (outputFilePrefixes.containsKey(outputFileName)) {
1423 throw conflictingOutputFile(outputFile, outputFilePrefixes.get(outputFileName));
1424 }
1425
1426 // Check if a prefix of this output file matches an already existing one
1427 PathFragment outputFileFragment = new PathFragment(outputFileName);
1428 for (int i = 1; i < outputFileFragment.segmentCount(); i++) {
1429 String prefix = outputFileFragment.subFragment(0, i).toString();
1430 if (outputFiles.containsKey(prefix)) {
1431 throw conflictingOutputFile(outputFile, outputFiles.get(prefix));
1432 }
1433 if (targets.containsKey(prefix)
1434 && targets.get(prefix) instanceof OutputFile) {
1435 throw conflictingOutputFile(outputFile, (OutputFile) targets.get(prefix));
1436 }
1437
1438 if (!outputFilePrefixes.containsKey(prefix)) {
1439 outputFilePrefixes.put(prefix, outputFile);
1440 }
1441 }
1442 }
1443
1444 checkForInputOutputConflicts(rule, outputFiles.keySet());
1445 }
1446
1447 /**
1448 * A utility method that checks for conflicts between
1449 * input file names and output file names for a rule from a build
1450 * file.
1451 * @param rule the rule whose inputs and outputs are
1452 * to be checked for conflicts.
1453 * @param outputFiles a set containing the names of output
1454 * files to be generated by the rule.
1455 * @throws NameConflictException if a conflict is found.
1456 */
1457 private void checkForInputOutputConflicts(Rule rule, Set<String> outputFiles)
1458 throws NameConflictException {
1459 PathFragment packageFragment = rule.getLabel().getPackageFragment();
1460 for (Label inputLabel : rule.getLabels()) {
1461 if (packageFragment.equals(inputLabel.getPackageFragment())
1462 && outputFiles.contains(inputLabel.getName())) {
1463 throw inputOutputNameConflict(rule, inputLabel.getName());
1464 }
1465 }
1466 }
1467
1468 /** An output file conflicts with another output file or the BUILD file. */
1469 private NameConflictException duplicateOutputFile(OutputFile duplicate, Target existing) {
1470 return new NameConflictException(duplicate.getTargetKind() + " '" + duplicate.getName()
1471 + "' in rule '" + duplicate.getGeneratingRule().getName() + "' "
1472 + conflictsWith(existing));
1473 }
1474
1475 /** The package contains two targets with the same name. */
1476 private NameConflictException nameConflict(Target duplicate, Target existing) {
1477 return new NameConflictException(duplicate.getTargetKind() + " '" + duplicate.getName()
1478 + "' in package '" + duplicate.getLabel().getPackageName() + "' "
1479 + conflictsWith(existing));
1480 }
1481
1482 /** A a rule has a input/output name conflict. */
1483 private NameConflictException inputOutputNameConflict(Rule rule, String conflictingName) {
1484 return new NameConflictException("rule '" + rule.getName() + "' has file '"
1485 + conflictingName + "' as both an input and an output");
1486 }
1487
1488 private static NameConflictException conflictingOutputFile(
1489 OutputFile added, OutputFile existing) {
1490 if (added.getGeneratingRule() == existing.getGeneratingRule()) {
1491 return new NameConflictException(String.format(
1492 "rule '%s' has conflicting output files '%s' and '%s'", added.getGeneratingRule()
1493 .getName(), added.getName(), existing.getName()));
1494 } else {
1495 return new NameConflictException(String.format(
1496 "output file '%s' of rule '%s' conflicts with output file '%s' of rule '%s'", added
1497 .getName(), added.getGeneratingRule().getName(), existing.getName(), existing
1498 .getGeneratingRule().getName()));
1499 }
1500 }
1501
1502 /**
1503 * Utility function for generating exception messages.
1504 */
1505 private static String conflictsWith(Target target) {
1506 String message = "conflicts with existing ";
1507 if (target instanceof OutputFile) {
1508 return message + "generated file from rule '"
1509 + ((OutputFile) target).getGeneratingRule().getName()
1510 + "'";
1511 } else {
1512 return message + target.getTargetKind();
1513 }
1514 }
1515 }
1516}