|  | // Copyright 2015 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.Function; | 
|  | import com.google.common.base.Preconditions; | 
|  | import com.google.common.base.Predicate; | 
|  | import com.google.common.collect.ImmutableList; | 
|  | import com.google.common.collect.ImmutableSet; | 
|  | import com.google.common.collect.Iterables; | 
|  | import com.google.common.collect.Sets; | 
|  | import com.google.devtools.build.lib.actions.FilesetOutputSymlink; | 
|  | import com.google.devtools.build.lib.actions.FilesetTraversalParams; | 
|  | import com.google.devtools.build.lib.actions.FilesetTraversalParams.DirectTraversal; | 
|  | import com.google.devtools.build.lib.actions.HasDigest; | 
|  | import com.google.devtools.build.lib.skyframe.RecursiveFilesystemTraversalFunction.DanglingSymlinkException; | 
|  | import com.google.devtools.build.lib.skyframe.RecursiveFilesystemTraversalFunction.RecursiveFilesystemTraversalException; | 
|  | import com.google.devtools.build.lib.skyframe.RecursiveFilesystemTraversalValue.ResolvedFile; | 
|  | import com.google.devtools.build.lib.skyframe.RecursiveFilesystemTraversalValue.TraversalRequest; | 
|  | import com.google.devtools.build.lib.vfs.Path; | 
|  | import com.google.devtools.build.lib.vfs.PathFragment; | 
|  | 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.util.LinkedHashMap; | 
|  | import java.util.Map; | 
|  | import java.util.Set; | 
|  | import java.util.TreeMap; | 
|  |  | 
|  | /** SkyFunction for {@link FilesetEntryValue}. */ | 
|  | public final class FilesetEntryFunction implements SkyFunction { | 
|  |  | 
|  | private static final class MissingDepException extends Exception {} | 
|  |  | 
|  | private static final class FilesetEntryFunctionException extends SkyFunctionException { | 
|  | FilesetEntryFunctionException(RecursiveFilesystemTraversalException e) { | 
|  | super(e, Transience.PERSISTENT); | 
|  | } | 
|  | } | 
|  |  | 
|  | private final Function<String, Path> getExecRoot; | 
|  |  | 
|  | public FilesetEntryFunction(Function<String, Path> getExecRoot) { | 
|  | this.getExecRoot = getExecRoot; | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public SkyValue compute(SkyKey key, Environment env) | 
|  | throws FilesetEntryFunctionException, InterruptedException { | 
|  | WorkspaceNameValue workspaceNameValue = | 
|  | (WorkspaceNameValue) env.getValue(WorkspaceNameValue.key()); | 
|  | if (workspaceNameValue == null) { | 
|  | return null; | 
|  | } | 
|  |  | 
|  | FilesetTraversalParams t = (FilesetTraversalParams) key.argument(); | 
|  | Preconditions.checkState( | 
|  | t.getDirectTraversal().isPresent() && t.getNestedArtifact() == null, | 
|  | "FilesetEntry does not support nested traversal: %s", t); | 
|  |  | 
|  | // The map of output symlinks. Each key is the path of a output symlink that the Fileset must | 
|  | // create, relative to the Fileset.out directory, and each value specifies extra information | 
|  | // about the link (its target, associated metadata and again its name). | 
|  | Map<PathFragment, FilesetOutputSymlink> outputSymlinks = new LinkedHashMap<>(); | 
|  |  | 
|  | // The "direct" traversal params are present, which is the case when the FilesetEntry | 
|  | // specifies a package's BUILD file, a directory or a list of files. | 
|  |  | 
|  | // The root of the direct traversal is defined as follows. | 
|  | // | 
|  | // If FilesetEntry.files is specified, then a TraversalRequest is created for each entry, the | 
|  | // root being the respective entry itself. These are all traversed for they may be | 
|  | // directories or symlinks to directories, and we need to establish Skyframe dependencies on | 
|  | // their contents for incremental correctness. If an entry is indeed a directory (but not when | 
|  | // it's a symlink to one) then we have to create symlinks to each of their children. | 
|  | // (NB: there seems to be no good reason for this, it's just how legacy Fileset works. We may | 
|  | // want to consider creating a symlink just for the directory and not for its child elements.) | 
|  | // | 
|  | // If FilesetEntry.files is not specified, then srcdir refers to either a BUILD file or a | 
|  | // directory. For the former, the root will be the parent of the BUILD file. For the latter, | 
|  | // the root will be srcdir itself. | 
|  | DirectTraversal direct = t.getDirectTraversal().get(); | 
|  |  | 
|  | RecursiveFilesystemTraversalValue rftv; | 
|  | try { | 
|  | // Traverse the filesystem to establish skyframe dependencies. | 
|  | rftv = traverse(env, createErrorInfo(t), direct); | 
|  | } catch (MissingDepException e) { | 
|  | return null; | 
|  | } | 
|  |  | 
|  | // The root can only be absent for the EMPTY rftv instance. | 
|  | if (!rftv.getResolvedRoot().isPresent()) { | 
|  | return FilesetEntryValue.EMPTY; | 
|  | } | 
|  |  | 
|  | ResolvedFile resolvedRoot = rftv.getResolvedRoot().get(); | 
|  |  | 
|  | // Handle dangling symlinks gracefully be returning empty results. | 
|  | if (!resolvedRoot.getType().exists()) { | 
|  | return FilesetEntryValue.EMPTY; | 
|  | } | 
|  |  | 
|  | // The prefix to remove is the entire path of the root. This is OK: | 
|  | // - when the root is a file, this removes the entire path, but the traversal's destination | 
|  | //   path is actually the name of the output symlink, so this works out correctly | 
|  | // - when the root is a directory or a symlink to one then we need to strip off the | 
|  | //   directory's path from every result (this is how the output symlinks must be created) | 
|  | //   before making them relative to the destination path | 
|  | PathFragment prefixToRemove = direct.getRoot().getRelativePart(); | 
|  |  | 
|  | Iterable<ResolvedFile> results = null; | 
|  |  | 
|  | if (direct.isRecursive() | 
|  | || (resolvedRoot.getType().isDirectory() && !resolvedRoot.getType().isSymlink())) { | 
|  | // The traversal is recursive (requested for an entire FilesetEntry.srcdir) or it was | 
|  | // requested for a FilesetEntry.files entry which turned out to be a directory. We need to | 
|  | // create an output symlink for every file in it and all of its subdirectories. Only | 
|  | // exception is when the subdirectory is really a symlink to a directory -- no output | 
|  | // shall be created for the contents of those. | 
|  | // Now we create Dir objects to model the filesystem tree. The object employs a trick to | 
|  | // find directory symlinks: directory symlinks have corresponding ResolvedFile entries and | 
|  | // are added as files too, while their children, also added as files, contain the path of | 
|  | // the parent. Finding and discarding the children is easy if we traverse the tree from | 
|  | // root to leaf. | 
|  | DirectoryTree root = new DirectoryTree(); | 
|  | for (ResolvedFile f : rftv.getTransitiveFiles().toList()) { | 
|  | PathFragment path = f.getNameInSymlinkTree().relativeTo(prefixToRemove); | 
|  | if (!path.isEmpty()) { | 
|  | path = t.getDestPath().getRelative(path); | 
|  | DirectoryTree dir = root; | 
|  | for (int i = 0; i < path.segmentCount() - 1; ++i) { | 
|  | dir = dir.addOrGetSubdir(path.getSegment(i)); | 
|  | } | 
|  | dir.maybeAddFile(f); | 
|  | } | 
|  | } | 
|  | // Here's where the magic happens. The returned iterable will yield all files in the | 
|  | // directory that are not under symlinked directories, as well as all directory symlinks. | 
|  | results = root.iterateFiles(); | 
|  | } else { | 
|  | // If we're on this branch then the traversal was done for just one entry in | 
|  | // FilesetEntry.files (which was not a directory, so it was either a file, a symlink to one | 
|  | // or a symlink to a directory), meaning we'll have only one output symlink. | 
|  | results = ImmutableList.of(resolvedRoot); | 
|  | } | 
|  |  | 
|  | // Create the set of excluded files. Only top-level files can be excluded, i.e. ones that are | 
|  | // directly under the root if the root is a directory. | 
|  | Set<String> exclusions = | 
|  | Sets.filter(t.getExcludedFiles(), e -> PathFragment.create(e).segmentCount() == 1); | 
|  |  | 
|  | // Create one output symlink for each entry in the results. | 
|  | for (ResolvedFile f : results) { | 
|  | // The linkName has to be under the traversal's root, which is also the prefix to remove. | 
|  | PathFragment linkName = f.getNameInSymlinkTree().relativeTo(prefixToRemove); | 
|  |  | 
|  | // Check whether the symlink is excluded before attempting to resolve it. | 
|  | // It may be dangling, but excluding it is still fine. | 
|  | // TODO(b/64754128): Investigate if we could have made the exclude earlier before | 
|  | //                   unnecessarily iterating over all the files in an excluded directory. | 
|  | if (linkName.segmentCount() > 0 && exclusions.contains(linkName.getSegment(0))) { | 
|  | continue; | 
|  | } | 
|  |  | 
|  | PathFragment targetName; | 
|  | try { | 
|  | targetName = f.getTargetInSymlinkTree(direct.isFollowingSymlinks()); | 
|  | } catch (DanglingSymlinkException e) { | 
|  | throw new FilesetEntryFunctionException(e); | 
|  | } | 
|  |  | 
|  | maybeStoreSymlink( | 
|  | linkName, | 
|  | targetName, | 
|  | f.getMetadata(), | 
|  | t.getDestPath(), | 
|  | direct.isGenerated(), | 
|  | outputSymlinks, | 
|  | getExecRoot.apply(workspaceNameValue.getName())); | 
|  | } | 
|  |  | 
|  | return FilesetEntryValue.of(ImmutableSet.copyOf(outputSymlinks.values())); | 
|  | } | 
|  |  | 
|  | /** Stores an output symlink unless it would overwrite an existing one. */ | 
|  | private void maybeStoreSymlink( | 
|  | PathFragment linkName, | 
|  | PathFragment linkTarget, | 
|  | HasDigest metadata, | 
|  | PathFragment destPath, | 
|  | boolean isGenerated, | 
|  | Map<PathFragment, FilesetOutputSymlink> result, | 
|  | Path execRoot) { | 
|  | linkName = destPath.getRelative(linkName); | 
|  | if (!result.containsKey(linkName)) { | 
|  | result.put( | 
|  | linkName, | 
|  | FilesetOutputSymlink.create( | 
|  | linkName, linkTarget, metadata, isGenerated, execRoot.asFragment())); | 
|  | } | 
|  | } | 
|  |  | 
|  | @Override | 
|  | public String extractTag(SkyKey skyKey) { | 
|  | return null; | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Returns the {@link TraversalRequest} node used to compute the Skyframe value for {@code | 
|  | * filesetEntryKey}. Should only be called to determine which nodes need to be rewound, and only | 
|  | * when {@code filesetEntryKey.isGenerated()}. | 
|  | */ | 
|  | public static TraversalRequest getDependencyForRewinding(FilesetEntryKey filesetEntryKey) { | 
|  | FilesetTraversalParams t = filesetEntryKey.argument(); | 
|  | Preconditions.checkState( | 
|  | t.getDirectTraversal().isPresent() && t.getNestedArtifact() == null, | 
|  | "FilesetEntry does not support nested traversal: %s", | 
|  | t); | 
|  | Preconditions.checkState( | 
|  | t.getDirectTraversal().get().isGenerated(), | 
|  | "Rewinding is only supported for outputs: %s", | 
|  | t); | 
|  | // Traversals in the output tree inline any recursive TraversalRequest evaluations, i.e. there | 
|  | // won't be any transitively depended-on TraversalRequests. | 
|  | return createTraversalRequestKey(createErrorInfo(t), t.getDirectTraversal().get()); | 
|  | } | 
|  |  | 
|  | private static TraversalRequest createTraversalRequestKey( | 
|  | String errorInfo, DirectTraversal traversal) { | 
|  | return TraversalRequest.create( | 
|  | traversal.getRoot(), | 
|  | traversal.isGenerated(), | 
|  | traversal.getPackageBoundaryMode(), | 
|  | traversal.isStrictFilesetOutput(), | 
|  | traversal.isPackage(), | 
|  | errorInfo); | 
|  | } | 
|  |  | 
|  | private static RecursiveFilesystemTraversalValue traverse( | 
|  | Environment env, String errorInfo, DirectTraversal traversal) | 
|  | throws MissingDepException, InterruptedException { | 
|  | RecursiveFilesystemTraversalValue v = | 
|  | (RecursiveFilesystemTraversalValue) | 
|  | env.getValue(createTraversalRequestKey(errorInfo, traversal)); | 
|  | if (env.valuesMissing()) { | 
|  | throw new MissingDepException(); | 
|  | } | 
|  | return v; | 
|  | } | 
|  |  | 
|  | private static String createErrorInfo(FilesetTraversalParams t) { | 
|  | if (t.getDirectTraversal().isPresent()) { | 
|  | DirectTraversal direct = t.getDirectTraversal().get(); | 
|  | return String.format( | 
|  | "Fileset '%s' traversing %s '%s'", | 
|  | t.getOwnerLabelForErrorMessages(), | 
|  | direct.isPackage() ? "package" : "file (or directory)", | 
|  | direct.getRoot().getRelativePart().getPathString()); | 
|  | } else { | 
|  | return String.format( | 
|  | "Fileset '%s' traversing another Fileset", t.getOwnerLabelForErrorMessages()); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Models a FilesetEntryFunction's portion of the symlink output tree created by a Fileset rule. | 
|  | * | 
|  | * <p>A Fileset rule's output is computed by zero or more {@link FilesetEntryFunction}s, resulting | 
|  | * in one {@link FilesetEntryValue} for each. Each of those represents a portion of the grand | 
|  | * output tree of the Fileset. These portions are later merged and written to the fileset manifest | 
|  | * file, which is then consumed by a tool that ultimately creates the symlinks in the filesystem. | 
|  | * | 
|  | * <p>Because the Fileset doesn't process the lists in the FilesetEntryValues any further than | 
|  | * merging them, they have to adhere to the conventions of the manifest file. One of these is that | 
|  | * files are alphabetically ordered (enables the consumer of the manifest to work faster than | 
|  | * otherwise) and another is that the contents of regular directories are listed, but contents | 
|  | * of directory symlinks are not, only the symlinks are. (Other details of the manifest file are | 
|  | * not relevant here.) | 
|  | * | 
|  | * <p>See {@link DirectoryTree#iterateFiles()} for more details. | 
|  | */ | 
|  | private static final class DirectoryTree { | 
|  | // Use TreeMaps for the benefit of alphabetically ordered iteration. | 
|  | public final Map<String, ResolvedFile> files = new TreeMap<>(); | 
|  | public final Map<String, DirectoryTree> dirs = new TreeMap<>(); | 
|  |  | 
|  | DirectoryTree addOrGetSubdir(String name) { | 
|  | DirectoryTree result = dirs.get(name); | 
|  | if (result == null) { | 
|  | result = new DirectoryTree(); | 
|  | dirs.put(name, result); | 
|  | } | 
|  | return result; | 
|  | } | 
|  |  | 
|  | void maybeAddFile(ResolvedFile r) { | 
|  | String name = r.getNameInSymlinkTree().getBaseName(); | 
|  | if (!files.containsKey(name)) { | 
|  | files.put(name, r); | 
|  | } | 
|  | } | 
|  |  | 
|  | /** | 
|  | * Lazily yields all files in this directory and all of its subdirectories. | 
|  | * | 
|  | * <p>The function first yields all the files directly under this directory, in alphabetical | 
|  | * order. Then come the contents of subdirectories, processed recursively in the same fashion | 
|  | * as this directory, and also in alphabetical order. | 
|  | * | 
|  | * <p>If a directory symlink is encountered its contents are not listed, only the symlink is. | 
|  | */ | 
|  | Iterable<ResolvedFile> iterateFiles() { | 
|  | // 1. Filter directory symlinks. If the symlink target contains files, those were added | 
|  | // as normal files so their parent directory (the symlink) would show up in the dirs map | 
|  | // (as a directory) as well as in the files map (as a symlink to a directory). | 
|  | final Set<String> fileNames = files.keySet(); | 
|  | Iterable<Map.Entry<String, DirectoryTree>> noDirSymlinkes = Iterables.filter(dirs.entrySet(), | 
|  | new Predicate<Map.Entry<String, DirectoryTree>>() { | 
|  | @Override | 
|  | public boolean apply(Map.Entry<String, DirectoryTree> input) { | 
|  | return !fileNames.contains(input.getKey()); | 
|  | } | 
|  | }); | 
|  |  | 
|  | // 2. Extract the iterables of the true subdirectories. | 
|  | Iterable<Iterable<ResolvedFile>> subdirIters = | 
|  | Iterables.transform( | 
|  | noDirSymlinkes, | 
|  | new Function<Map.Entry<String, DirectoryTree>, Iterable<ResolvedFile>>() { | 
|  | @Override | 
|  | public Iterable<ResolvedFile> apply(Map.Entry<String, DirectoryTree> input) { | 
|  | return input.getValue().iterateFiles(); | 
|  | } | 
|  | }); | 
|  |  | 
|  | // 3. Just concat all subdirectory iterations for one, seamless iteration. | 
|  | Iterable<ResolvedFile> dirsIter = Iterables.concat(subdirIters); | 
|  |  | 
|  | return Iterables.concat(files.values(), dirsIter); | 
|  | } | 
|  | } | 
|  | } |