blob: 3e676ecfdb8d78c3f0d8faf169125a28da1a67da [file] [log] [blame]
// Copyright 2016 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.bazel.repository.starlark;
import com.google.common.collect.ImmutableList;
import com.google.devtools.build.docgen.annot.DocCategory;
import com.google.devtools.build.lib.bazel.repository.starlark.StarlarkBaseExternalContext.ShouldWatch;
import com.google.devtools.build.lib.concurrent.ThreadSafety.Immutable;
import com.google.devtools.build.lib.rules.repository.RepositoryFunction.RepositoryFunctionException;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.build.lib.vfs.PathFragment;
import com.google.devtools.build.skyframe.SkyFunctionException.Transience;
import java.io.IOException;
import javax.annotation.Nullable;
import net.starlark.java.annot.Param;
import net.starlark.java.annot.StarlarkBuiltin;
import net.starlark.java.annot.StarlarkMethod;
import net.starlark.java.eval.EvalException;
import net.starlark.java.eval.Printer;
import net.starlark.java.eval.Sequence;
import net.starlark.java.eval.Starlark;
import net.starlark.java.eval.StarlarkValue;
import net.starlark.java.eval.Tuple;
/**
* A Path object to be used in repo rules and module extensions.
*
* <p>This path object enable non-hermetic operations from Starlark and should not be returned by
* something other than a StarlarkBaseExternalContext.
*/
@Immutable
@StarlarkBuiltin(
name = "path",
category = DocCategory.BUILTIN,
doc = "A structure representing a file to be used inside a repository.")
public final class StarlarkPath implements StarlarkValue {
private final StarlarkBaseExternalContext ctx;
private final Path path;
StarlarkPath(StarlarkBaseExternalContext ctx, Path path) {
this.ctx = ctx;
this.path = path;
}
Path getPath() {
return path;
}
@Override
public boolean isImmutable() {
return true; // immutable and Starlark-hashable
}
@Override
public boolean equals(Object obj) {
return (obj instanceof StarlarkPath) && path.equals(((StarlarkPath) obj).path);
}
@Override
public int hashCode() {
return path.hashCode();
}
@StarlarkMethod(
name = "basename",
structField = true,
doc = "A string giving the basename of the file.")
public String getBasename() {
return path.getBaseName();
}
@StarlarkMethod(
name = "readdir",
doc =
"Returns the list of entries in the directory denoted by this path. Each entry is a "
+ "<code>path</code> object itself.",
parameters = {
@Param(
name = "watch",
defaultValue = "'auto'",
positional = false,
named = true,
doc =
"whether Bazel should watch the list of entries in this directory and refetch the "
+ "repository or re-evaluate the module extension next time when any changes "
+ "are detected. Changes to detect include entry creation, deletion, and "
+ "renaming. Note that this doesn't watch the <em>contents</em> of any entries "
+ "in the directory.<p>Can be the string 'yes', 'no', or 'auto'. If set to "
+ "'auto', Bazel will only watch this directory when it is legal to do so (see "
+ "<a "
+ "href=\"repository_ctx.html#watch\"><code>repository_ctx.watch()</code></a> "
+ "docs for more information)."),
})
public ImmutableList<StarlarkPath> readdir(String watch)
throws EvalException, RepositoryFunctionException, InterruptedException {
if (!isDir()) {
throw Starlark.errorf("can't readdir(), not a directory: %s", path);
}
ctx.maybeWatchDirents(path, ShouldWatch.fromString(watch));
try {
ImmutableList.Builder<StarlarkPath> builder = ImmutableList.builder();
for (Path p : path.getDirectoryEntries()) {
builder.add(new StarlarkPath(ctx, p));
}
return builder.build();
} catch (IOException e) {
throw new RepositoryFunctionException(e, Transience.TRANSIENT);
}
}
@StarlarkMethod(
name = "dirname",
structField = true,
allowReturnNones = true,
doc = "The parent directory of this file, or None if this file does not have a parent.")
@Nullable
public StarlarkPath getDirname() {
Path parentPath = path.getParentDirectory();
return parentPath == null ? null : new StarlarkPath(ctx, parentPath);
}
@StarlarkMethod(
name = "get_child",
doc = "Returns the path obtained by joining this path with the given relative paths.",
extraPositionals =
@Param(
name = "relative_paths",
doc =
"Zero or more relative path strings to append to this path with path separators"
+ "added as needed."))
public StarlarkPath getChild(Tuple relativePaths) throws EvalException {
return new StarlarkPath(
ctx,
path.getRelative(
String.join(
Character.toString(PathFragment.SEPARATOR_CHAR),
Sequence.cast(relativePaths, String.class, "relative_paths"))));
}
@StarlarkMethod(
name = "exists",
structField = true,
doc =
"Returns true if the file or directory denoted by this path exists.<p>Note that "
+ "accessing this field does <em>not</em> cause the path to be watched. If you'd "
+ "like the repo rule or module extension to be sensitive to the path's existence, "
+ "use the <code>watch()</code> method on the context object.")
public boolean exists() {
return path.exists();
}
@StarlarkMethod(
name = "is_dir",
structField = true,
doc =
"Returns true if this path points to a directory.<p>Note that accessing this field does "
+ "<em>not</em> cause the path to be watched. If you'd like the repo rule or module "
+ "extension to be sensitive to whether the path is a directory or a file, use the "
+ "<code>watch()</code> method on the context object.")
public boolean isDir() {
return path.isDirectory();
}
@StarlarkMethod(
name = "realpath",
structField = true,
doc =
"Returns the canonical path for this path by repeatedly replacing all symbolic links "
+ "with their referents.")
public StarlarkPath realpath() throws IOException {
return new StarlarkPath(ctx, path.resolveSymbolicLinks());
}
@Override
public String toString() {
return path.toString();
}
@Override
public void repr(Printer printer) {
printer.append(toString());
}
}