blob: 644fe24347bd793900bfe9efabcf043296eb6748 [file] [log] [blame]
// Copyright 2019 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.base.Preconditions;
import com.google.common.base.Strings;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.lib.actions.FileValue;
import com.google.devtools.build.lib.vfs.Dirent;
import com.google.devtools.build.lib.vfs.RootedPath;
import com.google.devtools.build.lib.vfs.RootedPathAndCasing;
import com.google.devtools.build.skyframe.SkyFunction;
import com.google.devtools.build.skyframe.SkyFunctionException;
import com.google.devtools.build.skyframe.SkyKey;
import com.google.devtools.build.skyframe.SkyValue;
import java.io.IOException;
import java.util.Map;
import javax.annotation.Nullable;
/** SkyFunction for {@link PathCasingLookupValue}s. */
public final class PathCasingLookupFunction implements SkyFunction {
@Override
public PathCasingLookupValue compute(SkyKey skyKey, Environment env)
throws PathComponentIsNotDirectoryException, InterruptedException {
RootedPathAndCasing arg = (RootedPathAndCasing) skyKey.argument();
if (arg.getPath().getRootRelativePath().isEmpty()) {
// This is a Root, e.g. "[/foo/bar]/[]".
// As of 2019-12-04, PathCasingLookupValue is not used anywhere. But it's planned to be used
// in PackageLookupFunction to validate the package part's casing, so the RootedPath's Root's
// casing doesn't even matter, so if the relative part is empty then for our use case this is
// a correctly cased RootedPath.
return PathCasingLookupValue.GOOD;
}
RootedPath parent = arg.getPath().getParentDirectory();
Preconditions.checkNotNull(parent, arg.getPath());
Preconditions.checkNotNull(parent.getRootRelativePath(), arg.getPath());
SkyKey parentCasingKey = PathCasingLookupValue.key(parent);
SkyKey parentFileKey = FileValue.key(parent);
SkyKey childFileKey = FileValue.key(arg.getPath());
Map<SkyKey, SkyValue> values =
env.getValues(ImmutableList.of(parentCasingKey, parentFileKey, childFileKey));
if (env.valuesMissing()) {
return null;
}
if (!((PathCasingLookupValue) values.get(parentCasingKey)).isCorrect()) {
// Parent's casing is bad, so this path's casing is also bad.
return PathCasingLookupValue.BAD;
}
FileValue parentFile = (FileValue) values.get(parentFileKey);
if (!parentFile.exists()) {
// Parent's casing is good, because it's missing.
// That means this path is also missing, so by definition its casing is good.
return PathCasingLookupValue.GOOD;
}
if (!parentFile.isDirectory()) {
// Parent's casing is good, but it's not a directory.
throw new PathComponentIsNotDirectoryException(
new IOException(
"Cannot check path casing of "
+ arg.getPath()
+ ": its parent exists but is not a directory"));
}
FileValue childFile = (FileValue) values.get(childFileKey);
if (!childFile.exists()) {
// Parent's casing is good, but this file is missing.
// That means this path is missing, so by definition its casing is good.
return PathCasingLookupValue.GOOD;
}
// The parent file must exist, otherwise the DirectoryListingFunction will throw.
SkyKey parentListingKey = DirectoryListingValue.key(parent);
DirectoryListingValue parentList = (DirectoryListingValue) env.getValue(parentListingKey);
if (parentList == null) {
return null;
}
String expected = arg.getPath().getRootRelativePath().getBaseName();
// We already handled RootedPaths with empty relative part.
Preconditions.checkState(!Strings.isNullOrEmpty(expected), arg.getPath());
Dirent child = parentList.getDirents().maybeGetDirent(expected);
return (child != null && expected.equals(child.getName()))
? PathCasingLookupValue.GOOD
: PathCasingLookupValue.BAD;
}
@Nullable
@Override
public String extractTag(SkyKey skyKey) {
return null;
}
/** Thrown if a non-terminal path component exists, but it's not a directory. */
public static final class PathComponentIsNotDirectoryException extends SkyFunctionException {
public PathComponentIsNotDirectoryException(IOException e) {
super(e, Transience.PERSISTENT);
}
}
}