blob: acfc8178f1e5d608a1c14aac92a71c67f363ec95 [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.skyframe;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.MoreObjects;
import com.google.common.base.Preconditions;
import com.google.devtools.build.lib.cmdline.Label;
import com.google.devtools.build.lib.cmdline.PackageIdentifier;
import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec;
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.skyframe.AbstractSkyKey;
import com.google.devtools.build.skyframe.SkyFunctionName;
import com.google.devtools.build.skyframe.SkyKey;
import com.google.devtools.build.skyframe.SkyValue;
import javax.annotation.Nonnull;
/**
* A value that represents the result of looking for the existence of a package that owns a
* specific directory path. Compare with {@link PackageLookupValue}, which deals with existence of
* a specific package.
*/
public abstract class ContainingPackageLookupValue implements SkyValue {
@SerializationConstant public static final NoContainingPackage NONE = new NoContainingPackage();
/** Returns whether there is a containing package. */
public abstract boolean hasContainingPackage();
/** If there is a containing package, returns its name. */
public abstract PackageIdentifier getContainingPackageName();
/** If there is a containing package, returns its package root */
public abstract Root getContainingPackageRoot();
/**
* If there is not a containing package, returns a reason why (this is usually the reason the
* outer-most directory isn't a package).
*/
public String getReasonForNoContainingPackage() {
throw new IllegalStateException();
}
public static Key key(PackageIdentifier id) {
Preconditions.checkArgument(!id.getPackageFragment().isAbsolute(), id);
return Key.create(id);
}
/**
* Creates the error message for the input {@linkplain Label label} if label itself is a
* subpackage crosses boundary when an outer package exists.
*/
static String getErrorMessageForLabelSubpackageCrossesBoundary(
ContainingPackageLookupValue containingPkgLookupValue, Label label) {
PackageIdentifier containingPkg = containingPkgLookupValue.getContainingPackageName();
Preconditions.checkState(
label.getPackageIdentifier().getSourceRoot().startsWith(containingPkg.getSourceRoot()),
"Label's path should start with outer package's path.");
String message =
String.format(
"Label '%s' is invalid because '%s' is not a package", label, label.getPackageName());
PathFragment labelNameInContainingPackage =
label.toPathFragment().relativeTo(containingPkg.getPackageFragment());
message += "; perhaps you meant to put the colon here: '";
if (containingPkg.getRepository().isMain()) {
message += "//";
}
message += containingPkg + ":" + labelNameInContainingPackage + "'?";
return message;
}
/** {@link com.google.devtools.build.skyframe.SkyKey} for {@code ContainingPackageLookupValue}. */
@AutoCodec
public static class Key extends AbstractSkyKey<PackageIdentifier> {
private static final SkyKeyInterner<Key> interner = SkyKey.newInterner();
private Key(PackageIdentifier arg) {
super(arg);
}
@AutoCodec.VisibleForSerialization
@AutoCodec.Instantiator
static Key create(PackageIdentifier arg) {
return interner.intern(new Key(arg));
}
@Override
public SkyFunctionName functionName() {
return SkyFunctions.CONTAINING_PACKAGE_LOOKUP;
}
@Override
public SkyKeyInterner<Key> getSkyKeyInterner() {
return interner;
}
}
public static ContainingPackage withContainingPackage(PackageIdentifier pkgId, Root root) {
return new ContainingPackage(pkgId, root);
}
/** Value indicating there is no containing package. */
public static class NoContainingPackage extends ContainingPackageLookupValue {
private final String reason;
private NoContainingPackage() {
this.reason = null;
}
NoContainingPackage(@Nonnull String reason) {
this.reason = reason;
}
@Override
public boolean hasContainingPackage() {
return false;
}
@Override
public PackageIdentifier getContainingPackageName() {
throw new IllegalStateException();
}
@Override
public Root getContainingPackageRoot() {
throw new IllegalStateException();
}
@Override
public String toString() {
return getClass().getName();
}
@Override
public String getReasonForNoContainingPackage() {
return reason;
}
}
/** A successful lookup value. */
@VisibleForTesting
public static class ContainingPackage extends ContainingPackageLookupValue {
private final PackageIdentifier containingPackage;
private final Root containingPackageRoot;
private ContainingPackage(PackageIdentifier containingPackage, Root containingPackageRoot) {
this.containingPackage = containingPackage;
this.containingPackageRoot = containingPackageRoot;
}
@Override
public boolean hasContainingPackage() {
return true;
}
@Override
public PackageIdentifier getContainingPackageName() {
return containingPackage;
}
@Override
public Root getContainingPackageRoot() {
return containingPackageRoot;
}
@Override
public boolean equals(Object obj) {
if (this == obj) {
return true;
}
if (!(obj instanceof ContainingPackage)) {
return false;
}
ContainingPackage other = (ContainingPackage) obj;
return containingPackage.equals(other.containingPackage)
&& containingPackageRoot.equals(other.containingPackageRoot);
}
@Override
public int hashCode() {
return containingPackage.hashCode();
}
@Override
public String toString() {
return MoreObjects.toStringHelper(this)
.add("containingPackage", containingPackage)
.add("containingPackageRoot", containingPackageRoot)
.toString();
}
}
}