blob: 58a29a25388910b72bb1f667819fab088160fa3b [file] [log] [blame]
// Copyright 2021 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.skyframe;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.devtools.build.lib.analysis.BlazeDirectories;
import com.google.devtools.build.lib.bazel.bzlmod.BzlmodRepoRuleHelper;
import com.google.devtools.build.lib.bazel.bzlmod.BzlmodRepoRuleValue;
import com.google.devtools.build.lib.bazel.bzlmod.RepoSpec;
import com.google.devtools.build.lib.cmdline.LabelConstants;
import com.google.devtools.build.lib.packages.NoSuchPackageException;
import com.google.devtools.build.lib.packages.Package;
import com.google.devtools.build.lib.packages.PackageFactory;
import com.google.devtools.build.lib.packages.Rule;
import com.google.devtools.build.lib.packages.RuleClass;
import com.google.devtools.build.lib.packages.RuleClassProvider;
import com.google.devtools.build.lib.packages.RuleFactory;
import com.google.devtools.build.lib.packages.RuleFactory.BuildLangTypedAttributeValuesMap;
import com.google.devtools.build.lib.packages.RuleFactory.InvalidRuleException;
import com.google.devtools.build.lib.vfs.Root;
import com.google.devtools.build.lib.vfs.RootedPath;
import com.google.devtools.build.skyframe.SkyFunction;
import com.google.devtools.build.skyframe.SkyFunctionException;
import com.google.devtools.build.skyframe.SkyFunctionException.Transience;
import com.google.devtools.build.skyframe.SkyKey;
import com.google.devtools.build.skyframe.SkyValue;
import java.util.Optional;
import javax.annotation.Nullable;
import net.starlark.java.eval.StarlarkSemantics;
import net.starlark.java.eval.StarlarkThread.CallStackEntry;
import net.starlark.java.syntax.Location;
/**
* Look up the {link@ RepoSpec} of a given repository name and create its repository rule instance.
*/
public final class BzlmodRepoRuleFunction implements SkyFunction {
private static final String TOOLS_REPO = "bazel_tools";
private final PackageFactory packageFactory;
private final RuleClassProvider ruleClassProvider;
private final BlazeDirectories directories;
private final BzlmodRepoRuleHelper bzlmodRepoRuleHelper;
public BzlmodRepoRuleFunction(
PackageFactory packageFactory,
RuleClassProvider ruleClassProvider,
BlazeDirectories directories,
BzlmodRepoRuleHelper bzlmodRepoRuleHelper) {
this.packageFactory = packageFactory;
this.ruleClassProvider = ruleClassProvider;
this.directories = directories;
this.bzlmodRepoRuleHelper = bzlmodRepoRuleHelper;
}
@Nullable
@Override
public SkyValue compute(SkyKey skyKey, Environment env)
throws SkyFunctionException, InterruptedException {
StarlarkSemantics starlarkSemantics = PrecomputedValue.STARLARK_SEMANTICS.get(env);
if (starlarkSemantics == null) {
return null;
}
String repositoryName = ((BzlmodRepoRuleValue.Key) skyKey).argument();
// @bazel_tools is a special repo that we pull from the extracted install dir.
if (repositoryName.equals(TOOLS_REPO)) {
RepoSpec repoSpec =
RepoSpec.builder()
.setRuleClassName("local_repository")
.setAttributes(
ImmutableMap.of(
"name",
"bazel_tools",
"path",
directories
.getEmbeddedBinariesRoot()
.getChild("embedded_tools")
.getPathString()))
.build();
return createRuleFromSpec(repoSpec, starlarkSemantics, env);
}
Optional<RepoSpec> result = bzlmodRepoRuleHelper.getRepoSpec(env, repositoryName);
if (env.valuesMissing()) {
return null;
}
if (result.isPresent()) {
return createRuleFromSpec(result.get(), starlarkSemantics, env);
}
return BzlmodRepoRuleValue.REPO_RULE_NOT_FOUND_VALUE;
}
private BzlmodRepoRuleValue createRuleFromSpec(
RepoSpec repoSpec, StarlarkSemantics starlarkSemantics, Environment env)
throws BzlmodRepoRuleFunctionException, InterruptedException {
if (repoSpec.isNativeRepoRule()) {
return createNativeRepoRule(repoSpec, starlarkSemantics, env);
}
// TODO(pcloudy): Implement creating starlark repository rule
return BzlmodRepoRuleValue.REPO_RULE_NOT_FOUND_VALUE;
}
private BzlmodRepoRuleValue createNativeRepoRule(
RepoSpec repoSpec, StarlarkSemantics semantics, Environment env)
throws InterruptedException, BzlmodRepoRuleFunctionException {
RuleClass ruleClass = ruleClassProvider.getRuleClassMap().get(repoSpec.ruleClassName());
BuildLangTypedAttributeValuesMap attributeValues =
new BuildLangTypedAttributeValuesMap(repoSpec.attributes());
ImmutableList.Builder<CallStackEntry> callStack = ImmutableList.builder();
callStack.add(new CallStackEntry("BzlmodRepoRuleFunction.getNativeRepoRule", Location.BUILTIN));
// TODO(bazel-team): Don't use the {@link Rule} class for repository rule.
// Currently, the repository rule is represented with the {@link Rule} class that's designed
// for build rules. Therefore, we have to create a package instance for it, which doesn't make
// sense. We should migrate away from this implementation so that we don't refer to any build
// rule specific things in repository rule.
Rule rule;
Package.Builder pkg = createExternalPackageBuilder(semantics);
try {
rule =
RuleFactory.createRule(
pkg, ruleClass, attributeValues, env.getListener(), semantics, callStack.build());
// We need to actually build the package so that the rule has the correct package reference.
pkg.build();
} catch (InvalidRuleException e) {
throw new BzlmodRepoRuleFunctionException(e, Transience.PERSISTENT);
} catch (NoSuchPackageException e) {
throw new BzlmodRepoRuleFunctionException(e, Transience.PERSISTENT);
}
return new BzlmodRepoRuleValue(rule);
}
/**
* Create the external package builder, which is only for the convenience of creating repository
* rules.
*/
private Package.Builder createExternalPackageBuilder(StarlarkSemantics semantics) {
RootedPath bzlmodFile =
RootedPath.toRootedPath(
Root.fromPath(directories.getWorkspace()), LabelConstants.MODULE_DOT_BAZEL_FILE_NAME);
return packageFactory.newExternalPackageBuilder(
bzlmodFile, ruleClassProvider.getRunfilesPrefix(), semantics);
}
@Nullable
@Override
public String extractTag(SkyKey skyKey) {
// default behaviour
return null;
}
private static final class BzlmodRepoRuleFunctionException extends SkyFunctionException {
BzlmodRepoRuleFunctionException(InvalidRuleException e, Transience transience) {
super(e, transience);
}
BzlmodRepoRuleFunctionException(NoSuchPackageException e, Transience transience) {
super(e, transience);
}
}
}