blob: 603764c5293b74a2a64fbd056a81cf5f9ec8f2a7 [file] [log] [blame]
// Copyright 2018 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 static com.google.common.base.MoreObjects.firstNonNull;
import com.google.common.base.Preconditions;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.analysis.RuleDefinitionEnvironment;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.cmdline.RepositoryName;
import java.util.Optional;
import javax.annotation.Nullable;
import net.starlark.java.eval.EvalException;
import net.starlark.java.eval.Starlark;
import net.starlark.java.eval.StarlarkThread;
/**
* Bazel-specific contextual information associated with a Starlark evaluation thread.
*
* <p>This is stored in the StarlarkThread object as a thread-local. A subclass of this class may be
* used for certain kinds of Starlark evaluations; in that case it is still keyed in the
* thread-locals under {@code BazelStarlarkContext.class}.
*
* <p>This object is mutable and should not be reused for more than one Starlark thread.
*/
// TODO(b/236456122): rename BazelThreadContext, for symmetry with BazelModuleContext.
// TODO(b/236456122): We should break this class up into subclasses for each kind of evaluation, as
// opposed to storing specialized fields on this class and setting them to null for inapplicable
// environments. Subclasses should define {@code from(StarlarkThread)} and {@code
// fromOrFailFunction(StarlarkThread, String)} methods to be used in place of the {@code
// check*Phase} methods in this class. Kinds of evaluation include:
// - .bzl loading
// - BUILD evaluation (should absorb PackageFactory.PackageContext -- no need to store multiple
// thread-locals in StarlarkThread)
// - WORKSPACE evaluation (shares logic with BUILD)
// - rule and aspect analysis implementation (can/should we store the RuleContext here?)
// - implicit outputs
// - computed defaults
// - transition implementation
// - Args.map_each
// - probably others
// TODO(b/236456122): The inheritance of RuleDefinitionEnvironment should be replaced by
// composition, in an appropriate subclass of this class. Things like the tools repository, network
// allowlist, etc. can be accessed from the RDE. (If any info needs to be duplicated between RDE and
// here, we should assert consistency with a precondition check.)
public class BazelStarlarkContext
implements RuleDefinitionEnvironment, StarlarkThread.UncheckedExceptionContext {
/** The phase to which this Starlark thread belongs. */
// TODO(b/236456122): Eliminate.
public enum Phase {
WORKSPACE,
LOADING,
ANALYSIS
}
/**
* Retrieves this context from a Starlark thread, or throws {@link IllegalStateException} if
* unavailable.
*/
public static BazelStarlarkContext from(StarlarkThread thread) {
BazelStarlarkContext ctx = thread.getThreadLocal(BazelStarlarkContext.class);
// ISE rather than NPE for symmetry with subclasses.
Preconditions.checkState(
ctx != null, "Expected BazelStarlarkContext to be available in this Starlark thread");
return ctx;
}
/**
* Saves this {@code BazelStarlarkContext} in the specified Starlark thread. Call only once,
* before evaluation begins.
*/
public void storeInThread(StarlarkThread thread) {
Preconditions.checkState(thread.getThreadLocal(BazelStarlarkContext.class) == null);
thread.setThreadLocal(BazelStarlarkContext.class, this);
thread.setUncheckedExceptionContext(this);
}
// A generic counter for uniquely identifying symbols created in this Starlark evaluation.
private final SymbolGenerator<?> symbolGenerator;
// TODO(b/236456122): Migrate the below fields to subclasses.
private final Phase phase;
// Only necessary for loading phase threads.
@Nullable private final RepositoryName toolsRepository;
// Only necessary for loading phase threads to construct configuration_field.
@Nullable private final ImmutableMap<String, Class<?>> fragmentNameToClass;
@Nullable private final Label analysisRuleLabel;
// TODO(b/192694287): Remove once we migrate all tests from the allowlist
@Nullable private final Label networkAllowlistForTests;
/**
* @param phase the phase to which this Starlark thread belongs
* @param toolsRepository the name of the tools repository, such as "@bazel_tools" for loading
* phase threads, null for other threads.
* @param fragmentNameToClass a map from configuration fragment name to configuration fragment
* class, such as "apple" to AppleConfiguration.class for loading phase threads, null for
* other threads.
* @param symbolGenerator a {@link SymbolGenerator} to be used when creating objects to be
* compared using reference equality.
* @param analysisRuleLabel is the label of the rule for an analysis phase (rule or aspect
*/
public BazelStarlarkContext(
Phase phase,
@Nullable RepositoryName toolsRepository,
@Nullable ImmutableMap<String, Class<?>> fragmentNameToClass,
SymbolGenerator<?> symbolGenerator,
@Nullable Label analysisRuleLabel,
@Nullable Label networkAllowlistForTests) {
this.phase = Preconditions.checkNotNull(phase);
this.toolsRepository = toolsRepository;
this.fragmentNameToClass = fragmentNameToClass;
this.symbolGenerator = Preconditions.checkNotNull(symbolGenerator);
this.analysisRuleLabel = analysisRuleLabel;
this.networkAllowlistForTests = networkAllowlistForTests;
}
/** Returns the phase associated with this context. */
public Phase getPhase() {
return phase;
}
/** Returns the name of the tools repository, such as "@bazel_tools". */
@Nullable
@Override
public RepositoryName getToolsRepository() {
return toolsRepository;
}
/** Returns a map from configuration fragment name to configuration fragment class. */
@Nullable
public ImmutableMap<String, Class<?>> getFragmentNameToClass() {
return fragmentNameToClass;
}
public SymbolGenerator<?> getSymbolGenerator() {
return symbolGenerator;
}
/**
* Returns the label of the rule, if this is an analysis-phase (rule or aspect 'implementation')
* thread, or null otherwise.
*/
@Nullable
public Label getAnalysisRuleLabel() {
return analysisRuleLabel;
}
@Override
public String getContextForUncheckedException() {
return firstNonNull(analysisRuleLabel, phase).toString();
}
@Override
public Optional<Label> getNetworkAllowlistForTests() {
return Optional.ofNullable(networkAllowlistForTests);
}
/**
* Checks that the Starlark thread is in the loading or the workspace phase.
*
* @param function name of a function that requires this check
*/
// TODO(b/236456122): The Phase enum is incomplete. Ex: `Args.map_each` evaluation happens at
// execution time. So this is a misnomer and possibly wrong in those contexts.
public void checkLoadingOrWorkspacePhase(String function) throws EvalException {
if (phase == Phase.ANALYSIS) {
throw Starlark.errorf("'%s' cannot be called during the analysis phase", function);
}
}
/**
* Checks that the current StarlarkThread is in the loading phase.
*
* @param function name of a function that requires this check
*/
public void checkLoadingPhase(String function) throws EvalException {
if (phase != Phase.LOADING) {
throw Starlark.errorf("'%s' can only be called during the loading phase", function);
}
}
/**
* Checks that the current StarlarkThread is in the workspace phase.
*
* @param function name of a function that requires this check
*/
public void checkWorkspacePhase(String function) throws EvalException {
if (phase != Phase.WORKSPACE) {
throw Starlark.errorf("'%s' can only be called during workspace loading", function);
}
}
}