blob: 1a19d281d33eb53496cd2f461a0d56cd7f44094c [file] [log] [blame]
Damien Martin-Guillerezf88f4d82015-09-25 13:56:55 +00001// Copyright 2014 The Bazel Authors. All rights reserved.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +01002//
3// Licensed under the Apache License, Version 2.0 (the "License");
4// you may not use this file except in compliance with the License.
5// You may obtain a copy of the License at
6//
7// http://www.apache.org/licenses/LICENSE-2.0
8//
9// Unless required by applicable law or agreed to in writing, software
10// distributed under the License is distributed on an "AS IS" BASIS,
11// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
12// See the License for the specific language governing permissions and
13// limitations under the License.
14
15package com.google.devtools.build.lib.actions;
16
17import com.google.common.annotations.VisibleForTesting;
18import com.google.common.base.Function;
19import com.google.common.base.Functions;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010020import com.google.common.collect.Collections2;
Googlerece75722016-02-11 17:55:41 +000021import com.google.common.collect.ImmutableSet;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010022import com.google.common.collect.Iterables;
Michael Thvedt434e68e2016-02-09 00:57:46 +000023import com.google.devtools.build.lib.actions.Artifact.ArtifactExpander;
Rumou Duana77f32c2016-04-13 21:59:21 +000024import com.google.devtools.build.lib.actions.Artifact.TreeFileArtifact;
Mark Schaller6df81792015-12-10 18:47:47 +000025import com.google.devtools.build.lib.util.Preconditions;
Michael Thvedte3b1cb72016-02-08 23:32:27 +000026import com.google.devtools.build.lib.vfs.PathFragment;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010027import java.util.ArrayList;
28import java.util.Collection;
29import java.util.List;
Googlerece75722016-02-11 17:55:41 +000030import java.util.Set;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010031
32/**
33 * Helper utility to create ActionInput instances.
34 */
35public final class ActionInputHelper {
36 private ActionInputHelper() {
37 }
38
39 @VisibleForTesting
Michael Thvedt434e68e2016-02-09 00:57:46 +000040 public static ArtifactExpander actionGraphArtifactExpander(
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010041 final ActionGraph actionGraph) {
Michael Thvedt434e68e2016-02-09 00:57:46 +000042 return new ArtifactExpander() {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010043 @Override
Rumou Duana77f32c2016-04-13 21:59:21 +000044 public void expand(Artifact mm, Collection<? super Artifact> output) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010045 // Skyframe is stricter in that it checks that "mm" is a input of the action, because
46 // it cannot expand arbitrary middlemen without access to a global action graph.
47 // We could check this constraint here too, but it seems unnecessary. This code is
48 // going away anyway.
49 Preconditions.checkArgument(mm.isMiddlemanArtifact(),
50 "%s is not a middleman artifact", mm);
Rumou Duan33bab462016-04-25 17:55:12 +000051 ActionAnalysisMetadata middlemanAction = actionGraph.getGeneratingAction(mm);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010052 Preconditions.checkState(middlemanAction != null, mm);
53 // TODO(bazel-team): Consider expanding recursively or throwing an exception here.
54 // Most likely, this code will cause silent errors if we ever have a middleman that
55 // contains a middleman.
56 if (middlemanAction.getActionType() == Action.MiddlemanType.AGGREGATING_MIDDLEMAN) {
57 Artifact.addNonMiddlemanArtifacts(middlemanAction.getInputs(), output,
58 Functions.<Artifact>identity());
59 }
60
61 }
62 };
63 }
64
65 /**
66 * Most ActionInputs are created and never used again. On the off chance that one is, however, we
67 * implement equality via path comparison. Since file caches are keyed by ActionInput, equality
68 * checking does come up.
69 */
70 private static class BasicActionInput implements ActionInput {
71 private final String path;
Ulf Adams14db5072017-03-20 16:00:51 +000072
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010073 public BasicActionInput(String path) {
74 this.path = Preconditions.checkNotNull(path);
75 }
76
77 @Override
78 public String getExecPathString() {
79 return path;
80 }
81
82 @Override
Ulf Adams14db5072017-03-20 16:00:51 +000083 public PathFragment getExecPath() {
nharmatab4060b62017-04-04 17:11:39 +000084 return PathFragment.create(path);
Ulf Adams14db5072017-03-20 16:00:51 +000085 }
86
87 @Override
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010088 public int hashCode() {
89 return path.hashCode();
90 }
91
92 @Override
93 public boolean equals(Object other) {
94 if (this == other) {
95 return true;
96 }
97 if (other == null) {
98 return false;
99 }
100 if (!this.getClass().equals(other.getClass())) {
101 return false;
102 }
103 return this.path.equals(((BasicActionInput) other).path);
104 }
105
106 @Override
107 public String toString() {
108 return "BasicActionInput: " + path;
109 }
110 }
111
112 /**
113 * Creates an ActionInput with just the given relative path and no digest.
114 *
115 * @param path the relative path of the input.
116 * @return a ActionInput.
117 */
118 public static ActionInput fromPath(String path) {
119 return new BasicActionInput(path);
120 }
121
Ulf Adams11e52b42017-01-11 11:00:24 +0000122 /**
123 * Creates an ActionInput with just the given relative path and no digest.
124 *
125 * @param path the relative path of the input.
126 * @return a ActionInput.
127 */
128 public static ActionInput fromPath(PathFragment path) {
129 return fromPath(path.getPathString());
130 }
131
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100132 /**
133 * Creates a sequence of {@link ActionInput}s from a sequence of string paths.
134 */
135 public static Collection<ActionInput> fromPaths(Collection<String> paths) {
laurentlb3d2a68c2017-06-30 00:32:04 +0200136 return Collections2.transform(paths, ActionInputHelper::fromPath);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100137 }
138
139 /**
Rumou Duana77f32c2016-04-13 21:59:21 +0000140 * Instantiates a concrete TreeFileArtifact with the given parent Artifact and path
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000141 * relative to that Artifact.
142 */
Rumou Duana77f32c2016-04-13 21:59:21 +0000143 public static TreeFileArtifact treeFileArtifact(
144 Artifact parent, PathFragment relativePath) {
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000145 Preconditions.checkState(parent.isTreeArtifact(),
146 "Given parent %s must be a TreeArtifact", parent);
Rumou Duana77f32c2016-04-13 21:59:21 +0000147 return new TreeFileArtifact(parent, relativePath);
148 }
149
150 public static TreeFileArtifact treeFileArtifact(
151 Artifact parent, PathFragment relativePath, ArtifactOwner artifactOwner) {
152 Preconditions.checkState(parent.isTreeArtifact(),
153 "Given parent %s must be a TreeArtifact", parent);
154 return new TreeFileArtifact(
155 parent,
156 relativePath,
157 artifactOwner);
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000158 }
159
160 /**
Rumou Duana77f32c2016-04-13 21:59:21 +0000161 * Instantiates a concrete TreeFileArtifact with the given parent Artifact and path
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000162 * relative to that Artifact.
163 */
Rumou Duana77f32c2016-04-13 21:59:21 +0000164 public static TreeFileArtifact treeFileArtifact(Artifact parent, String relativePath) {
nharmatab4060b62017-04-04 17:11:39 +0000165 return treeFileArtifact(parent, PathFragment.create(relativePath));
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000166 }
167
Rumou Duana77f32c2016-04-13 21:59:21 +0000168 /** Returns an Iterable of TreeFileArtifacts with the given parent and parent relative paths. */
169 public static Iterable<TreeFileArtifact> asTreeFileArtifacts(
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000170 final Artifact parent, Iterable<? extends PathFragment> parentRelativePaths) {
171 Preconditions.checkState(parent.isTreeArtifact(),
172 "Given parent %s must be a TreeArtifact", parent);
laurentlb3d2a68c2017-06-30 00:32:04 +0200173 return Iterables.transform(
174 parentRelativePaths, pathFragment -> treeFileArtifact(parent, pathFragment));
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000175 }
176
Rumou Duana77f32c2016-04-13 21:59:21 +0000177 /** Returns a Set of TreeFileArtifacts with the given parent and parent-relative paths. */
178 public static Set<TreeFileArtifact> asTreeFileArtifacts(
Googlerece75722016-02-11 17:55:41 +0000179 final Artifact parent, Set<? extends PathFragment> parentRelativePaths) {
180 Preconditions.checkState(parent.isTreeArtifact(),
181 "Given parent %s must be a TreeArtifact", parent);
182
Rumou Duana77f32c2016-04-13 21:59:21 +0000183 ImmutableSet.Builder<TreeFileArtifact> builder = ImmutableSet.builder();
Googlerece75722016-02-11 17:55:41 +0000184 for (PathFragment path : parentRelativePaths) {
Rumou Duana77f32c2016-04-13 21:59:21 +0000185 builder.add(treeFileArtifact(parent, path));
Googlerece75722016-02-11 17:55:41 +0000186 }
187
188 return builder.build();
189 }
190
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000191 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100192 * Expands middleman artifacts in a sequence of {@link ActionInput}s.
193 *
194 * <p>Non-middleman artifacts are returned untouched.
195 */
Michael Thvedt434e68e2016-02-09 00:57:46 +0000196 public static List<ActionInput> expandArtifacts(Iterable<? extends ActionInput> inputs,
197 ArtifactExpander artifactExpander) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100198
199 List<ActionInput> result = new ArrayList<>();
200 List<Artifact> containedArtifacts = new ArrayList<>();
201 for (ActionInput input : inputs) {
202 if (!(input instanceof Artifact)) {
203 result.add(input);
204 continue;
205 }
206 containedArtifacts.add((Artifact) input);
207 }
Michael Thvedt434e68e2016-02-09 00:57:46 +0000208 Artifact.addExpandedArtifacts(containedArtifacts, result, artifactExpander);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100209 return result;
210 }
211
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000212 /** Formatter for execPath String output. Public because {@link Artifact} uses it directly. */
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100213 public static final Function<ActionInput, String> EXEC_PATH_STRING_FORMATTER =
laurentlb3d2a68c2017-06-30 00:32:04 +0200214 ActionInput::getExecPathString;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100215
216 public static Iterable<String> toExecPaths(Iterable<? extends ActionInput> artifacts) {
217 return Iterables.transform(artifacts, EXEC_PATH_STRING_FORMATTER);
218 }
219}