blob: 4ca4a1b162396ccc43db4479002650e731fba6b9 [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;
72 public BasicActionInput(String path) {
73 this.path = Preconditions.checkNotNull(path);
74 }
75
76 @Override
77 public String getExecPathString() {
78 return path;
79 }
80
81 @Override
82 public int hashCode() {
83 return path.hashCode();
84 }
85
86 @Override
87 public boolean equals(Object other) {
88 if (this == other) {
89 return true;
90 }
91 if (other == null) {
92 return false;
93 }
94 if (!this.getClass().equals(other.getClass())) {
95 return false;
96 }
97 return this.path.equals(((BasicActionInput) other).path);
98 }
99
100 @Override
101 public String toString() {
102 return "BasicActionInput: " + path;
103 }
104 }
105
106 /**
107 * Creates an ActionInput with just the given relative path and no digest.
108 *
109 * @param path the relative path of the input.
110 * @return a ActionInput.
111 */
112 public static ActionInput fromPath(String path) {
113 return new BasicActionInput(path);
114 }
115
Ulf Adams11e52b42017-01-11 11:00:24 +0000116 /**
117 * Creates an ActionInput with just the given relative path and no digest.
118 *
119 * @param path the relative path of the input.
120 * @return a ActionInput.
121 */
122 public static ActionInput fromPath(PathFragment path) {
123 return fromPath(path.getPathString());
124 }
125
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100126 private static final Function<String, ActionInput> FROM_PATH =
127 new Function<String, ActionInput>() {
128 @Override
129 public ActionInput apply(String path) {
130 return fromPath(path);
131 }
132 };
133
134 /**
135 * Creates a sequence of {@link ActionInput}s from a sequence of string paths.
136 */
137 public static Collection<ActionInput> fromPaths(Collection<String> paths) {
138 return Collections2.transform(paths, FROM_PATH);
139 }
140
141 /**
Rumou Duana77f32c2016-04-13 21:59:21 +0000142 * Instantiates a concrete TreeFileArtifact with the given parent Artifact and path
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000143 * relative to that Artifact.
144 */
Rumou Duana77f32c2016-04-13 21:59:21 +0000145 public static TreeFileArtifact treeFileArtifact(
146 Artifact parent, PathFragment relativePath) {
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000147 Preconditions.checkState(parent.isTreeArtifact(),
148 "Given parent %s must be a TreeArtifact", parent);
Rumou Duana77f32c2016-04-13 21:59:21 +0000149 return new TreeFileArtifact(parent, relativePath);
150 }
151
152 public static TreeFileArtifact treeFileArtifact(
153 Artifact parent, PathFragment relativePath, ArtifactOwner artifactOwner) {
154 Preconditions.checkState(parent.isTreeArtifact(),
155 "Given parent %s must be a TreeArtifact", parent);
156 return new TreeFileArtifact(
157 parent,
158 relativePath,
159 artifactOwner);
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000160 }
161
162 /**
Rumou Duana77f32c2016-04-13 21:59:21 +0000163 * Instantiates a concrete TreeFileArtifact with the given parent Artifact and path
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000164 * relative to that Artifact.
165 */
Rumou Duana77f32c2016-04-13 21:59:21 +0000166 public static TreeFileArtifact treeFileArtifact(Artifact parent, String relativePath) {
167 return treeFileArtifact(parent, new PathFragment(relativePath));
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000168 }
169
Rumou Duana77f32c2016-04-13 21:59:21 +0000170 /** Returns an Iterable of TreeFileArtifacts with the given parent and parent relative paths. */
171 public static Iterable<TreeFileArtifact> asTreeFileArtifacts(
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000172 final Artifact parent, Iterable<? extends PathFragment> parentRelativePaths) {
173 Preconditions.checkState(parent.isTreeArtifact(),
174 "Given parent %s must be a TreeArtifact", parent);
175 return Iterables.transform(parentRelativePaths,
Rumou Duana77f32c2016-04-13 21:59:21 +0000176 new Function<PathFragment, TreeFileArtifact>() {
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000177 @Override
Rumou Duana77f32c2016-04-13 21:59:21 +0000178 public TreeFileArtifact apply(PathFragment pathFragment) {
179 return treeFileArtifact(parent, pathFragment);
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000180 }
181 });
182 }
183
Rumou Duana77f32c2016-04-13 21:59:21 +0000184 /** Returns a Set of TreeFileArtifacts with the given parent and parent-relative paths. */
185 public static Set<TreeFileArtifact> asTreeFileArtifacts(
Googlerece75722016-02-11 17:55:41 +0000186 final Artifact parent, Set<? extends PathFragment> parentRelativePaths) {
187 Preconditions.checkState(parent.isTreeArtifact(),
188 "Given parent %s must be a TreeArtifact", parent);
189
Rumou Duana77f32c2016-04-13 21:59:21 +0000190 ImmutableSet.Builder<TreeFileArtifact> builder = ImmutableSet.builder();
Googlerece75722016-02-11 17:55:41 +0000191 for (PathFragment path : parentRelativePaths) {
Rumou Duana77f32c2016-04-13 21:59:21 +0000192 builder.add(treeFileArtifact(parent, path));
Googlerece75722016-02-11 17:55:41 +0000193 }
194
195 return builder.build();
196 }
197
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000198 /**
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100199 * Expands middleman artifacts in a sequence of {@link ActionInput}s.
200 *
201 * <p>Non-middleman artifacts are returned untouched.
202 */
Michael Thvedt434e68e2016-02-09 00:57:46 +0000203 public static List<ActionInput> expandArtifacts(Iterable<? extends ActionInput> inputs,
204 ArtifactExpander artifactExpander) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100205
206 List<ActionInput> result = new ArrayList<>();
207 List<Artifact> containedArtifacts = new ArrayList<>();
208 for (ActionInput input : inputs) {
209 if (!(input instanceof Artifact)) {
210 result.add(input);
211 continue;
212 }
213 containedArtifacts.add((Artifact) input);
214 }
Michael Thvedt434e68e2016-02-09 00:57:46 +0000215 Artifact.addExpandedArtifacts(containedArtifacts, result, artifactExpander);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100216 return result;
217 }
218
Michael Thvedte3b1cb72016-02-08 23:32:27 +0000219 /** Formatter for execPath String output. Public because {@link Artifact} uses it directly. */
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100220 public static final Function<ActionInput, String> EXEC_PATH_STRING_FORMATTER =
221 new Function<ActionInput, String>() {
222 @Override
223 public String apply(ActionInput input) {
224 return input.getExecPathString();
225 }
226 };
227
228 public static Iterable<String> toExecPaths(Iterable<? extends ActionInput> artifacts) {
229 return Iterables.transform(artifacts, EXEC_PATH_STRING_FORMATTER);
230 }
231}