blob: 9dadb42b0e3cc29c8d34ac05e6690f688bfd5464 [file] [log] [blame]
// Copyright 2014 The Bazel Authors. All rights reserved.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package com.google.devtools.build.lib.packages;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.cmdline.PackageIdentifier;
import com.google.devtools.build.lib.packages.License.DistributionType;
import java.util.List;
import java.util.Set;
import javax.annotation.Nullable;
/**
* A node in the build dependency graph, identified by a Label.
*
* <p>This StarlarkBuiltin does not contain any documentation since Starlark's Target type refers to
* TransitiveInfoCollection.class, which contains the appropriate documentation.
*/
public interface Target extends TargetData {
/** Returns the Package to which this target belongs. */
Package getPackage();
/**
* Returns the innermost symbolic macro that declared this target, or null if it was declared
* outside any symbolic macro (i.e. directly in a BUILD file or only in one or more legacy
* macros).
*
* <p>For targets in deserialized packages, throws {@link IllegalStateException}.
*/
@Nullable
default MacroInstance getDeclaringMacro() {
// TODO: #19922 - We might replace Package#getDeclaringMacroForTarget by storing a reference to
// the declaring macro in implementations of this interface (sharing memory with the field for
// the package).
return getPackage().getDeclaringMacroForTarget(getName());
}
/**
* Returns the package that is considered to be the declaring location of this target.
*
* <p>For targets created inside a symbolic macro, this is the package containing the .bzl code of
* the innermost running symbolic macro. For targets not in any symbolic macro, this is the same
* as the package the target lives in.
*/
default PackageIdentifier getDeclaringPackage() {
PackageIdentifier pkgId = getPackage().getDeclaringPackageForTargetIfInMacro(getName());
return pkgId != null ? pkgId : getPackage().getPackageIdentifier();
}
/**
* Returns true if this target was declared within one or more symbolic macros, or false if it was
* the product of running only a BUILD file and the legacy macros it called.
*/
default boolean isCreatedInSymbolicMacro() {
return getPackage().getDeclaringPackageForTargetIfInMacro(getName()) != null;
}
/**
* Returns the rule associated with this target, if any.
*
* <p>If this is a Rule, returns itself; it this is an OutputFile, returns its generating rule; if
* this is an input file, returns null.
*/
@Nullable
Rule getAssociatedRule();
/**
* Returns the license associated with this target.
*/
License getLicense();
/** Returns the set of distribution types associated with this target. */
Set<DistributionType> getDistributions();
/**
* Returns the visibility that was supplied at the point of this target's declaration -- e.g. the
* {@code visibility} attribute/argument for a rule target or {@code exports_files()} declaration)
* -- or null if none was given.
*
* <p>Although this value is "raw", it is still normalized through {@link
* RuleVisibility#validateAndSimplify RuleVisibility parsing}, e.g. eliminating redundant {@code
* //visibility:private} items and replacing the list with a single {@code //visibility:public}
* item if at least one such item appears.
*
* <p>This value may be useful to tooling that wants to introspect a target's visibility via
* {@code bazel query} and feed the result back into a modified target declaration, without
* picking up the package's default visibility, or the added location of the package or symbolic
* macro the target was declared in. It is not useful as a direct input to the visibility
* semantics; for that see {@link #getActualVisibility}.
*
* <p>This is also the value that is introspected through {@code native.existing_rules()}, except
* that null is replaced by an empty visibility.
*/
@Nullable
RuleVisibility getRawVisibility();
/**
* Returns the default visibility value to fall back on if this target does not have a raw
* visibility.
*
* <p>Usually this is just the package's default visibility value for targets not declared in
* symbolic macros, and private for targets within symbolic macros. (In other words, a package's
* default visibility does not propagate to within a symbolic macro.) However, some targets may
* inject additional default visibility behavior here.
*/
default RuleVisibility getDefaultVisibility() {
return isCreatedInSymbolicMacro()
? RuleVisibility.PRIVATE
: getPackage().getPackageArgs().defaultVisibility();
}
/**
* Returns the {@link #getRawVisibility raw visibility} of this target, falling back on a {@link
* #getDefaultVisibility default value} if no raw visibility was supplied.
*
* <p>Due to the fallback, the result cannot be null.
*
* <p>This value may be useful for introspecting a target's visibility and reporting it in a
* context where the package's default visibility is not known. It is not useful as a direct input
* to the visibility semantics; for that see {@link #getActualVisibility}.
*/
// TODO(brandjon): Perhaps the default value within a symbolic macro should be the value of the
// `--default_visibility` flag / PrecomputedValue. This would ensure targets within macros are
// always visible within unit tests or escape-hatched builds.
// TODO(jhorvitz): Usually one of the following two methods suffice. Try to remove this.
default RuleVisibility getVisibility() {
RuleVisibility result = getRawVisibility();
return result != null ? result : getDefaultVisibility();
}
/**
* Equivalent to calling {@link RuleVisibility#getDependencyLabels} on the value returned by
* {@link #getVisibility}, but potentially more efficient.
*
* <p>Prefer this method over {@link #getVisibility} when only the dependency labels are needed
* and not a {@link RuleVisibility} instance.
*/
default Iterable<Label> getVisibilityDependencyLabels() {
return getVisibility().getDependencyLabels();
}
/**
* Equivalent to calling {@link RuleVisibility#getDeclaredLabels} on the value returned by {@link
* #getVisibility}, but potentially more efficient.
*
* <p>Prefer this method over {@link #getVisibility} when only the declared labels are needed and
* not a {@link RuleVisibility} instance.
*/
default List<Label> getVisibilityDeclaredLabels() {
return getVisibility().getDeclaredLabels();
}
/**
* Returns the visibility of this target, as understood by the visibility semantics.
*
* <p>This is the result of {@link #getVisibility} unioned with the package where this target was
* instantiated (which can differ from the package where this target lives if the target was
* created inside a symbolic macro).
*
* <p>This is the value that feeds into visibility checking in the analysis phase. See {@link
* ConfiguredTargetFactory#convertVisibility} and {@link
* CommonPrerequisiteValidator#isVisibleToLocation}.
*/
default RuleVisibility getActualVisibility() {
RuleVisibility visibility = getVisibility();
MacroInstance declaringMacro = getDeclaringMacro();
PackageIdentifier instantiatingLoc =
declaringMacro == null
? getPackage().getPackageIdentifier()
: declaringMacro.getDefinitionPackage();
return visibility.concatWithPackage(instantiatingLoc);
}
/** Returns whether this target type can be configured (e.g. accepts non-null configurations). */
boolean isConfigurable();
/**
* Creates a compact representation of this target with enough information for dependent parents.
*/
TargetData reduceForSerialization();
}