blob: 8fd2ecbd035c1e27bfef8eda8aa42ed9d32c3ca3 [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.includescanning;
import com.google.devtools.build.lib.actions.EnvironmentalExecException;
import com.google.devtools.build.lib.actions.FileValue;
import com.google.devtools.build.lib.cmdline.PackageIdentifier;
import com.google.devtools.build.lib.includescanning.IncludeParser.Hints;
import com.google.devtools.build.lib.packages.BuildFileNotFoundException;
import com.google.devtools.build.lib.server.FailureDetails.FailureDetail;
import com.google.devtools.build.lib.server.FailureDetails.IncludeScanning;
import com.google.devtools.build.lib.server.FailureDetails.IncludeScanning.Code;
import com.google.devtools.build.lib.skyframe.ContainingPackageLookupValue;
import com.google.devtools.build.lib.skyframe.serialization.autocodec.SerializationConstant;
import com.google.devtools.build.lib.vfs.PathFragment;
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.SkyKey;
import java.io.IOException;
import javax.annotation.Nullable;
/**
* Creates a {@link IncludeParser.HintsRules} object. Done in Skyframe to track dependence on
* INCLUDE_HINTS file.
*/
public class IncludeHintsFunction implements SkyFunction {
// TODO(b/111722810): the action cache is not sensitive to changes in the INCLUDE_HINTS file, so
// even though Skyframe handles changes, we may still not re-execute an affected action.
@SerializationConstant
public static final SkyKey INCLUDE_HINTS_KEY =
(SkyKey) () -> IncludeScanningSkyFunctions.INCLUDE_HINTS;
private final PathFragment hintsFile;
public IncludeHintsFunction(PathFragment hintsFile) {
this.hintsFile = hintsFile;
}
@Nullable
@Override
public IncludeParser.HintsRules compute(SkyKey skyKey, Environment env)
throws IncludeHintsFunctionException, InterruptedException {
Root hintsPackageRoot;
try {
ContainingPackageLookupValue hintsLookupValue =
(ContainingPackageLookupValue) env.getValueOrThrow(ContainingPackageLookupValue.key(
PackageIdentifier.createInMainRepo(hintsFile.getParentDirectory())),
IOException.class, BuildFileNotFoundException.class);
if (env.valuesMissing()) {
return null;
}
if (!hintsLookupValue.hasContainingPackage()) {
String reasonForNoContainingPackage = hintsLookupValue.getReasonForNoContainingPackage();
String message =
String.format(
"INCLUDE_HINTS file %s was not in a package%s",
hintsFile,
reasonForNoContainingPackage != null ? ": " + reasonForNoContainingPackage : "");
throw new IncludeHintsFunctionException(
new EnvironmentalExecException(
createFailureDetail(message, Code.INCLUDE_HINTS_FILE_NOT_IN_PACKAGE)));
}
hintsPackageRoot = hintsLookupValue.getContainingPackageRoot();
env.getValueOrThrow(FileValue.key(RootedPath.toRootedPath(hintsPackageRoot, hintsFile)),
IOException.class);
} catch (IOException | BuildFileNotFoundException e) {
throw new IncludeHintsFunctionException(
new EnvironmentalExecException(
e,
createFailureDetail(
"could not read INCLUDE_HINTS file", Code.INCLUDE_HINTS_READ_FAILURE)));
}
if (env.valuesMissing()) {
return null;
}
try {
return Hints.getRules(hintsPackageRoot.getRelative(hintsFile));
} catch (IOException e) {
throw new IncludeHintsFunctionException(
new EnvironmentalExecException(
e,
createFailureDetail(
"could not read INCLUDE_HINTS file", Code.INCLUDE_HINTS_READ_FAILURE)));
}
}
private static FailureDetail createFailureDetail(String message, Code detailedCode) {
return FailureDetail.newBuilder()
.setMessage(message)
.setIncludeScanning(IncludeScanning.newBuilder().setCode(detailedCode))
.build();
}
/**
* Used to declare the exception type that can be wrapped in the exception thrown by
* {@link IncludeHintsFunction#compute}.
*/
private static final class IncludeHintsFunctionException extends SkyFunctionException {
IncludeHintsFunctionException(EnvironmentalExecException e) {
super(e, Transience.PERSISTENT);
}
}
}