| // 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.collect.ImmutableList; |
| import com.google.common.collect.ImmutableSet; |
| import com.google.devtools.build.lib.util.OS; |
| import com.google.devtools.build.lib.vfs.ModifiedFileSet; |
| import com.google.devtools.build.lib.vfs.PathFragment; |
| import com.google.devtools.build.lib.vfs.Root; |
| import com.google.devtools.common.options.Option; |
| import com.google.devtools.common.options.OptionDocumentationCategory; |
| import com.google.devtools.common.options.OptionEffectTag; |
| import com.google.devtools.common.options.OptionsBase; |
| import java.io.IOException; |
| import java.nio.file.FileSystems; |
| import java.nio.file.Path; |
| import java.util.Set; |
| import javax.annotation.Nullable; |
| |
| /** |
| * File system watcher for local filesystems. It's able to provide a list of changed files between |
| * two consecutive calls. On Linux, uses the standard Java WatchService, which uses 'inotify' and, |
| * on OS X, uses {@link MacOSXFsEventsDiffAwareness}, which use FSEvents. |
| * |
| * <p> |
| * This is an abstract class, specialized by {@link MacOSXFsEventsDiffAwareness} and |
| * {@link WatchServiceDiffAwareness}. |
| */ |
| public abstract class LocalDiffAwareness implements DiffAwareness { |
| /** |
| * Option to enable / disable local diff awareness. |
| */ |
| public static final class Options extends OptionsBase { |
| @Option( |
| name = "watchfs", |
| defaultValue = "false", |
| documentationCategory = OptionDocumentationCategory.UNCATEGORIZED, |
| effectTags = {OptionEffectTag.UNKNOWN}, |
| help = |
| "On Linux/macOS: If true, %{product} tries to use the operating system's file watch " |
| + "service for local changes instead of scanning every file for a change. On " |
| + "Windows: this flag currently is a non-op but can be enabled in conjunction " |
| + "with --experimental_windows_watchfs. On any OS: The behavior is undefined " |
| + "if your workspace is on a network file system, and files are edited on a " |
| + "remote machine.") |
| public boolean watchFS; |
| |
| @Option( |
| name = "experimental_windows_watchfs", |
| defaultValue = "false", |
| documentationCategory = OptionDocumentationCategory.UNCATEGORIZED, |
| effectTags = {OptionEffectTag.UNKNOWN}, |
| help = |
| "If true, experimental Windows support for --watchfs is enabled. Otherwise --watchfs" |
| + "is a non-op on Windows. Make sure to also enable --watchfs.") |
| public boolean windowsWatchFS; |
| } |
| |
| /** Factory for creating {@link LocalDiffAwareness} instances. */ |
| public static class Factory implements DiffAwareness.Factory { |
| private final ImmutableList<String> excludedNetworkFileSystemsPrefixes; |
| |
| /** |
| * Creates a new factory; the file system watcher may not work on all file systems, particularly |
| * for network file systems. The prefix list can be used to exclude known paths that point to |
| * network file systems. |
| */ |
| public Factory(ImmutableList<String> excludedNetworkFileSystemsPrefixes) { |
| this.excludedNetworkFileSystemsPrefixes = excludedNetworkFileSystemsPrefixes; |
| } |
| |
| @Override |
| @Nullable |
| public DiffAwareness maybeCreate(Root pathEntry) { |
| com.google.devtools.build.lib.vfs.Path resolvedPathEntry; |
| try { |
| resolvedPathEntry = pathEntry.asPath().resolveSymbolicLinks(); |
| } catch (IOException e) { |
| return null; |
| } |
| PathFragment resolvedPathEntryFragment = resolvedPathEntry.asFragment(); |
| // There's no good way to automatically detect network file systems. We rely on a list of |
| // paths to exclude for now (and maybe add a command-line option in the future?). |
| for (String prefix : excludedNetworkFileSystemsPrefixes) { |
| if (resolvedPathEntryFragment.startsWith(PathFragment.create(prefix))) { |
| return null; |
| } |
| } |
| // On OSX uses FsEvents due to https://bugs.openjdk.java.net/browse/JDK-7133447 |
| if (OS.getCurrent() == OS.DARWIN) { |
| return new MacOSXFsEventsDiffAwareness(resolvedPathEntryFragment.toString()); |
| } |
| |
| return new WatchServiceDiffAwareness(resolvedPathEntryFragment.toString()); |
| } |
| } |
| |
| /** |
| * A view that results in any subsequent getDiff calls returning |
| * {@link ModifiedFileSet#EVERYTHING_MODIFIED}. Use this if --watchFs is disabled. |
| * |
| * <p>The position is set to -2 in order for {@link #areInSequence} below to always return false |
| * if this view is passed to it. Any negative number would work; we don't use -1 as the other |
| * view may have a position of 0. |
| */ |
| protected static final View EVERYTHING_MODIFIED = |
| new SequentialView(/*owner=*/null, /*position=*/-2, ImmutableSet.<Path>of()); |
| |
| public static boolean areInSequence(SequentialView oldView, SequentialView newView) { |
| // Keep this in sync with the EVERYTHING_MODIFIED View above. |
| return oldView.owner == newView.owner && (oldView.position + 1) == newView.position; |
| } |
| |
| private int numGetCurrentViewCalls = 0; |
| |
| /** Root directory to watch. This is an absolute path. */ |
| protected final Path watchRootPath; |
| |
| protected LocalDiffAwareness(String watchRoot) { |
| this.watchRootPath = FileSystems.getDefault().getPath(watchRoot); |
| } |
| |
| /** |
| * The WatchService is inherently sequential and side-effectful, so we enforce this by only |
| * supporting {@link #getDiff} calls that happen to be sequential. |
| */ |
| @VisibleForTesting |
| static class SequentialView implements DiffAwareness.View { |
| private final LocalDiffAwareness owner; |
| private final int position; |
| private final Set<Path> modifiedAbsolutePaths; |
| |
| public SequentialView(LocalDiffAwareness owner, int position, Set<Path> modifiedAbsolutePaths) { |
| this.owner = owner; |
| this.position = position; |
| this.modifiedAbsolutePaths = modifiedAbsolutePaths; |
| } |
| |
| @Override |
| public String toString() { |
| return String.format("SequentialView[owner=%s, position=%d, modifiedAbsolutePaths=%s]", owner, |
| position, modifiedAbsolutePaths); |
| } |
| } |
| |
| /** |
| * Returns true on any call before first call to {@link #newView}. |
| */ |
| protected boolean isFirstCall() { |
| return numGetCurrentViewCalls == 0; |
| } |
| |
| /** |
| * Create a new views using a list of modified absolute paths. This will increase the view |
| * counter. |
| */ |
| protected SequentialView newView(Set<Path> modifiedAbsolutePaths) { |
| numGetCurrentViewCalls++; |
| return new SequentialView(this, numGetCurrentViewCalls, modifiedAbsolutePaths); |
| } |
| |
| @Override |
| public ModifiedFileSet getDiff(View oldView, View newView) |
| throws IncompatibleViewException, BrokenDiffAwarenessException { |
| SequentialView oldSequentialView; |
| SequentialView newSequentialView; |
| try { |
| oldSequentialView = (SequentialView) oldView; |
| newSequentialView = (SequentialView) newView; |
| } catch (ClassCastException e) { |
| throw new IncompatibleViewException("Given views are not from LocalDiffAwareness"); |
| } |
| if (!areInSequence(oldSequentialView, newSequentialView)) { |
| return ModifiedFileSet.EVERYTHING_MODIFIED; |
| } |
| |
| ModifiedFileSet.Builder resultBuilder = ModifiedFileSet.builder(); |
| for (Path modifiedPath : newSequentialView.modifiedAbsolutePaths) { |
| if (!modifiedPath.startsWith(watchRootPath)) { |
| throw new BrokenDiffAwarenessException( |
| String.format("%s is not under %s", modifiedPath, watchRootPath)); |
| } |
| PathFragment relativePath = |
| PathFragment.create(watchRootPath.relativize(modifiedPath).toString()); |
| if (!relativePath.isEmpty()) { |
| resultBuilder.modify(relativePath); |
| } |
| } |
| return resultBuilder.build(); |
| } |
| |
| @Override |
| public String name() { |
| return "local"; |
| } |
| } |