blob: eac03d982b2b2de8d46e89c6aa1f907f72dee01b [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.
14package com.google.devtools.build.lib.actions;
15
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010016import com.google.common.base.Predicate;
Janak Ramakrishnan90f3d342015-03-27 19:45:18 +000017import com.google.common.collect.ImmutableList;
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +000018import com.google.common.collect.ImmutableMap;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010019import com.google.common.collect.Iterables;
Rumou Duan33bab462016-04-25 17:55:12 +000020import com.google.devtools.build.lib.actions.ActionAnalysisMetadata.MiddlemanType;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010021import com.google.devtools.build.lib.actions.cache.ActionCache;
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +000022import com.google.devtools.build.lib.actions.cache.ActionCache.Entry;
Shreya Bhattarai141b6c22016-08-22 22:00:24 +000023import com.google.devtools.build.lib.actions.cache.DigestUtils;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010024import com.google.devtools.build.lib.actions.cache.Metadata;
25import com.google.devtools.build.lib.actions.cache.MetadataHandler;
26import com.google.devtools.build.lib.events.Event;
27import com.google.devtools.build.lib.events.EventHandler;
28import com.google.devtools.build.lib.events.EventKind;
Mark Schaller6df81792015-12-10 18:47:47 +000029import com.google.devtools.build.lib.util.Preconditions;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010030import com.google.devtools.build.lib.vfs.PathFragment;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010031import java.io.IOException;
32import java.util.ArrayList;
33import java.util.HashMap;
34import java.util.List;
35import java.util.Map;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010036import javax.annotation.Nullable;
37
38/**
39 * Checks whether an {@link Action} needs to be executed, or whether it has not changed since it was
40 * last stored in the action cache. Must be informed of the new Action data after execution as well.
41 *
42 * <p>The fingerprint, input files names, and metadata (either mtimes or MD5sums) of each action are
43 * cached in the action cache to avoid unnecessary rebuilds. Middleman artifacts are handled
44 * specially, avoiding the need to create actual files corresponding to the middleman artifacts.
45 * Instead of that, results of MiddlemanAction dependency checks are cached internally and then
46 * reused whenever an input middleman artifact is encountered.
47 *
48 * <p>While instances of this class hold references to action and metadata cache instances, they are
49 * otherwise lightweight, and should be constructed anew and discarded for each build request.
50 */
51public class ActionCacheChecker {
52 private final ActionCache actionCache;
53 private final Predicate<? super Action> executionFilter;
54 private final ArtifactResolver artifactResolver;
55 // True iff --verbose_explanations flag is set.
56 private final boolean verboseExplanations;
57
58 public ActionCacheChecker(ActionCache actionCache, ArtifactResolver artifactResolver,
59 Predicate<? super Action> executionFilter, boolean verboseExplanations) {
60 this.actionCache = actionCache;
61 this.executionFilter = executionFilter;
62 this.artifactResolver = artifactResolver;
63 this.verboseExplanations = verboseExplanations;
64 }
65
66 public boolean isActionExecutionProhibited(Action action) {
67 return !executionFilter.apply(action);
68 }
69
70 /**
71 * Checks whether one of existing output paths is already used as a key.
72 * If yes, returns it - otherwise uses first output file as a key
73 */
74 private ActionCache.Entry getCacheEntry(Action action) {
75 for (Artifact output : action.getOutputs()) {
76 ActionCache.Entry entry = actionCache.get(output.getExecPathString());
77 if (entry != null) {
78 return entry;
79 }
80 }
81 return null;
82 }
83
Janak Ramakrishnane3f04b82015-03-27 16:45:34 +000084 private void removeCacheEntry(Action action) {
85 for (Artifact output : action.getOutputs()) {
86 actionCache.remove(output.getExecPathString());
87 }
88 }
89
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010090 /**
91 * Validate metadata state for action input or output artifacts.
92 *
93 * @param entry cached action information.
94 * @param action action to be validated.
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +000095 * @param actionInputs the inputs of the action. Normally just the result of action.getInputs(),
96 * but if this action doesn't yet know its inputs, we check the inputs from the cache.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +010097 * @param metadataHandler provider of metadata for the artifacts this action interacts with.
98 * @param checkOutput true to validate output artifacts, Otherwise, just
99 * validate inputs.
100 *
101 * @return true if at least one artifact has changed, false - otherwise.
102 */
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +0000103 private boolean validateArtifacts(Entry entry, Action action,
104 Iterable<Artifact> actionInputs, MetadataHandler metadataHandler, boolean checkOutput) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100105 Iterable<Artifact> artifacts = checkOutput
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +0000106 ? Iterables.concat(action.getOutputs(), actionInputs)
107 : actionInputs;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100108 Map<String, Metadata> mdMap = new HashMap<>();
109 for (Artifact artifact : artifacts) {
110 mdMap.put(artifact.getExecPathString(), metadataHandler.getMetadataMaybe(artifact));
111 }
Shreya Bhattarai141b6c22016-08-22 22:00:24 +0000112 return !DigestUtils.fromMetadata(mdMap).equals(entry.getFileDigest());
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100113 }
114
115 private void reportCommand(EventHandler handler, Action action) {
116 if (handler != null) {
117 if (verboseExplanations) {
118 String keyDescription = action.describeKey();
Laszlo Csomorf04efcc2015-02-12 17:08:06 +0000119 reportRebuild(handler, action, keyDescription == null
120 ? "action command has changed"
121 : "action command has changed.\nNew action: " + keyDescription);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100122 } else {
123 reportRebuild(handler, action,
124 "action command has changed (try --verbose_explanations for more info)");
125 }
126 }
127 }
128
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +0000129 private void reportClientEnv(EventHandler handler, Action action, Map<String, String> used) {
130 if (handler != null) {
131 if (verboseExplanations) {
132 StringBuilder message = new StringBuilder();
133 message.append("Effective client environment has changed. Now using\n");
134 for (Map.Entry<String, String> entry : used.entrySet()) {
135 message.append(" ").append(entry.getKey()).append("=").append(entry.getValue())
136 .append("\n");
137 }
138 reportRebuild(handler, action, message.toString());
139 } else {
140 reportRebuild(
141 handler,
142 action,
143 "Effective client environment has changed (try --verbose_explanations for more info)");
144 }
145 }
146 }
147
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100148 protected boolean unconditionalExecution(Action action) {
149 return !isActionExecutionProhibited(action) && action.executeUnconditionally();
150 }
151
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +0000152 private static Map<String, String> computeUsedClientEnv(
153 Action action, Map<String, String> clientEnv) {
154 Map<String, String> used = new HashMap<>();
155 for (String var : action.getClientEnvironmentVariables()) {
156 String value = clientEnv.get(var);
157 if (value != null) {
158 used.put(var, value);
159 }
160 }
161 return used;
162 }
163
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100164 /**
165 * Checks whether {@code action} needs to be executed and returns a non-null Token if so.
166 *
167 * <p>The method checks if any of the action's inputs or outputs have changed. Returns a non-null
168 * {@link Token} if the action needs to be executed, and null otherwise.
169 *
170 * <p>If this method returns non-null, indicating that the action will be executed, the
Janak Ramakrishnan73055be2015-04-13 18:32:49 +0000171 * metadataHandler's {@link MetadataHandler#discardOutputMetadata} method must be called, so that
172 * it does not serve stale metadata for the action's outputs after the action is executed.
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100173 */
174 // Note: the handler should only be used for DEPCHECKER events; there's no
175 // guarantee it will be available for other events.
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +0000176 public Token getTokenIfNeedToExecute(
177 Action action,
178 Iterable<Artifact> resolvedCacheArtifacts,
179 Map<String, String> clientEnv,
180 EventHandler handler,
181 MetadataHandler metadataHandler) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100182 // TODO(bazel-team): (2010) For RunfilesAction/SymlinkAction and similar actions that
183 // produce only symlinks we should not check whether inputs are valid at all - all that matters
184 // that inputs and outputs are still exist (and new inputs have not appeared). All other checks
185 // are unnecessary. In other words, the only metadata we should check for them is file existence
186 // itself.
187
188 MiddlemanType middlemanType = action.getActionType();
189 if (middlemanType.isMiddleman()) {
190 // Some types of middlemen are not checked because they should not
191 // propagate invalidation of their inputs.
192 if (middlemanType != MiddlemanType.ERROR_PROPAGATING_MIDDLEMAN) {
193 checkMiddlemanAction(action, handler, metadataHandler);
194 }
195 return null;
196 }
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +0000197 Iterable<Artifact> actionInputs = action.getInputs();
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +0000198 // Resolve action inputs from cache, if necessary.
Lukacs Berki5ea2b142017-02-28 10:46:53 +0000199 boolean inputsDiscovered = action.inputsDiscovered();
200 if (!inputsDiscovered && resolvedCacheArtifacts != null) {
Janak Ramakrishnan90f3d342015-03-27 19:45:18 +0000201 // The action doesn't know its inputs, but the caller has a good idea of what they are.
202 Preconditions.checkState(action.discoversInputs(),
203 "Actions that don't know their inputs must discover them: %s", action);
204 actionInputs = resolvedCacheArtifacts;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100205 }
Janak Ramakrishnan90f3d342015-03-27 19:45:18 +0000206 ActionCache.Entry entry = getCacheEntry(action);
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +0000207 if (mustExecute(action, entry, handler, metadataHandler, actionInputs, clientEnv)) {
Janak Ramakrishnane3f04b82015-03-27 16:45:34 +0000208 if (entry != null) {
209 removeCacheEntry(action);
210 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100211 return new Token(getKeyString(action));
212 }
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +0000213
Lukacs Berki5ea2b142017-02-28 10:46:53 +0000214 if (!inputsDiscovered) {
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +0000215 action.updateInputs(actionInputs);
216 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100217 return null;
218 }
219
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +0000220 protected boolean mustExecute(
221 Action action,
222 @Nullable ActionCache.Entry entry,
223 EventHandler handler,
224 MetadataHandler metadataHandler,
225 Iterable<Artifact> actionInputs,
226 Map<String, String> clientEnv) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100227 // Unconditional execution can be applied only for actions that are allowed to be executed.
228 if (unconditionalExecution(action)) {
229 Preconditions.checkState(action.isVolatile());
230 reportUnconditionalExecution(handler, action);
231 return true; // must execute - unconditional execution is requested.
232 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100233 if (entry == null) {
234 reportNewAction(handler, action);
235 return true; // must execute -- no cache entry (e.g. first build)
236 }
237
238 if (entry.isCorrupted()) {
239 reportCorruptedCacheEntry(handler, action);
240 return true; // cache entry is corrupted - must execute
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +0000241 } else if (validateArtifacts(entry, action, actionInputs, metadataHandler, true)) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100242 reportChanged(handler, action);
243 return true; // files have changed
Eric Fellheimer1a34a242016-01-21 19:17:19 +0000244 } else if (!entry.getActionKey().equals(action.getKey())) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100245 reportCommand(handler, action);
246 return true; // must execute -- action key is different
247 }
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +0000248 Map<String, String> usedClientEnv = computeUsedClientEnv(action, clientEnv);
249 if (!entry.getUsedClientEnvDigest().equals(DigestUtils.fromEnv(usedClientEnv))) {
250 reportClientEnv(handler, action, usedClientEnv);
251 return true; // different values taken from the environment -- must execute
252 }
253
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100254
255 entry.getFileDigest();
256 return false; // cache hit
257 }
258
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +0000259 public void afterExecution(
260 Action action, Token token, MetadataHandler metadataHandler, Map<String, String> clientEnv)
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100261 throws IOException {
262 Preconditions.checkArgument(token != null);
263 String key = token.cacheKey;
Janak Ramakrishnane3f04b82015-03-27 16:45:34 +0000264 if (actionCache.get(key) != null) {
265 // This cache entry has already been updated by a shared action. We don't need to do it again.
266 return;
267 }
Klaus Aehlig6f33a1c2016-09-13 16:46:10 +0000268 Map<String, String> usedClientEnv = computeUsedClientEnv(action, clientEnv);
269 ActionCache.Entry entry =
Googlera2d61fa2017-02-16 16:14:32 +0000270 actionCache.newEntry(action.getKey(), usedClientEnv, action.discoversInputs());
271 if (entry == null) {
272 // Action cache is disabled, don't generate digests.
273 return;
274 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100275 for (Artifact output : action.getOutputs()) {
276 // Remove old records from the cache if they used different key.
277 String execPath = output.getExecPathString();
278 if (!key.equals(execPath)) {
Philipp Wollermanne03758b2015-06-15 08:10:47 +0000279 actionCache.remove(execPath);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100280 }
Michajlo Matijkiw13459b42015-03-12 19:43:20 +0000281 if (!metadataHandler.artifactOmitted(output)) {
282 // Output files *must* exist and be accessible after successful action execution.
283 Metadata metadata = metadataHandler.getMetadata(output);
284 Preconditions.checkState(metadata != null);
285 entry.addFile(output.getExecPath(), metadata);
286 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100287 }
288 for (Artifact input : action.getInputs()) {
289 entry.addFile(input.getExecPath(), metadataHandler.getMetadataMaybe(input));
290 }
291 entry.getFileDigest();
292 actionCache.put(key, entry);
293 }
294
Janak Ramakrishnan90f3d342015-03-27 19:45:18 +0000295 @Nullable
Michajlo Matijkiw331633c2015-06-09 22:09:03 +0000296 public Iterable<Artifact> getCachedInputs(Action action, PackageRootResolver resolver)
Janak Ramakrishnan3c0adb22016-08-15 21:54:55 +0000297 throws PackageRootResolutionException, InterruptedException {
Janak Ramakrishnan90f3d342015-03-27 19:45:18 +0000298 ActionCache.Entry entry = getCacheEntry(action);
299 if (entry == null || entry.isCorrupted()) {
300 return ImmutableList.of();
301 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100302
303 List<PathFragment> outputs = new ArrayList<>();
304 for (Artifact output : action.getOutputs()) {
305 outputs.add(output.getExecPath());
306 }
Lukacs Berki1f2caa52017-02-02 10:47:07 +0000307 List<PathFragment> inputExecPaths = new ArrayList<>();
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100308 for (String path : entry.getPaths()) {
309 PathFragment execPath = new PathFragment(path);
310 // Code assumes that action has only 1-2 outputs and ArrayList.contains() will be
311 // most efficient.
312 if (!outputs.contains(execPath)) {
Lukacs Berki1f2caa52017-02-02 10:47:07 +0000313 inputExecPaths.add(execPath);
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100314 }
315 }
Lukacs Berki1f2caa52017-02-02 10:47:07 +0000316
317 // Note that this method may trigger a violation of the desirable invariant that getInputs()
318 // is a superset of getMandatoryInputs(). See bug about an "action not in canonical form"
319 // error message and the integration test test_crosstool_change_and_failure().
320 Map<PathFragment, Artifact> allowedDerivedInputsMap = new HashMap<>();
321 for (Artifact derivedInput : action.getAllowedDerivedInputs()) {
322 if (!derivedInput.isSourceArtifact()) {
323 allowedDerivedInputsMap.put(derivedInput.getExecPath(), derivedInput);
324 }
325 }
326
327 List<Artifact> inputArtifacts = new ArrayList<>();
328 List<PathFragment> unresolvedPaths = new ArrayList<>();
329 for (PathFragment execPath : inputExecPaths) {
330 Artifact artifact = allowedDerivedInputsMap.get(execPath);
331 if (artifact != null) {
332 inputArtifacts.add(artifact);
333 } else {
334 // Remember this execPath, we will try to resolve it as a source artifact.
335 unresolvedPaths.add(execPath);
336 }
337 }
338
339 Map<PathFragment, Artifact> resolvedArtifacts =
340 artifactResolver.resolveSourceArtifacts(unresolvedPaths, resolver);
341 if (resolvedArtifacts == null) {
342 // We are missing some dependencies. We need to rerun this update later.
343 return null;
344 }
345
346 for (PathFragment execPath : unresolvedPaths) {
347 Artifact artifact = resolvedArtifacts.get(execPath);
348 // If PathFragment cannot be resolved into the artifact, ignore it. This could happen if the
349 // rule has changed and the action no longer depends on, e.g., an additional source file in a
350 // separate package and that package is no longer referenced anywhere else. It is safe to
351 // ignore such paths because dependency checker would identify changes in inputs (ignored path
352 // was used before) and will force action execution.
353 if (artifact != null) {
354 inputArtifacts.add(artifact);
355 }
356 }
357 return inputArtifacts;
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100358 }
359
360 /**
361 * Special handling for the MiddlemanAction. Since MiddlemanAction output
362 * artifacts are purely fictional and used only to stay within dependency
363 * graph model limitations (action has to depend on artifacts, not on other
364 * actions), we do not need to validate metadata for the outputs - only for
365 * inputs. We also do not need to validate MiddlemanAction key, since action
366 * cache entry key already incorporates that information for the middlemen
367 * and we will experience a cache miss when it is different. Whenever it
368 * encounters middleman artifacts as input artifacts for other actions, it
369 * consults with the aggregated middleman digest computed here.
370 */
371 protected void checkMiddlemanAction(Action action, EventHandler handler,
372 MetadataHandler metadataHandler) {
373 Artifact middleman = action.getPrimaryOutput();
374 String cacheKey = middleman.getExecPathString();
375 ActionCache.Entry entry = actionCache.get(cacheKey);
376 boolean changed = false;
377 if (entry != null) {
378 if (entry.isCorrupted()) {
379 reportCorruptedCacheEntry(handler, action);
380 changed = true;
Janak Ramakrishnana7c84b52015-03-18 21:59:16 +0000381 } else if (validateArtifacts(entry, action, action.getInputs(), metadataHandler, false)) {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100382 reportChanged(handler, action);
383 changed = true;
384 }
385 } else {
386 reportChangedDeps(handler, action);
387 changed = true;
388 }
389 if (changed) {
390 // Compute the aggregated middleman digest.
391 // Since we never validate action key for middlemen, we should not store
392 // it in the cache entry and just use empty string instead.
Googlera2d61fa2017-02-16 16:14:32 +0000393 entry = actionCache.newEntry("", ImmutableMap.<String, String>of(), false);
394 if (entry != null) {
395 for (Artifact input : action.getInputs()) {
396 entry.addFile(input.getExecPath(), metadataHandler.getMetadataMaybe(input));
397 }
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100398 }
399 }
400
Googlera2d61fa2017-02-16 16:14:32 +0000401 // Action cache is disabled, skip the digest.
402 if (entry == null) {
403 return;
404 }
405
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100406 metadataHandler.setDigestForVirtualArtifact(middleman, entry.getFileDigest());
407 if (changed) {
408 actionCache.put(cacheKey, entry);
409 }
410 }
411
412 /**
413 * Returns an action key. It is always set to the first output exec path string.
414 */
415 private static String getKeyString(Action action) {
416 Preconditions.checkState(!action.getOutputs().isEmpty());
417 return action.getOutputs().iterator().next().getExecPathString();
418 }
419
420
421 /**
422 * In most cases, this method should not be called directly - reportXXX() methods
423 * should be used instead. This is done to avoid cost associated with building
424 * the message.
425 */
426 private static void reportRebuild(@Nullable EventHandler handler, Action action, String message) {
427 // For MiddlemanAction, do not report rebuild.
428 if (handler != null && !action.getActionType().isMiddleman()) {
Ulf Adams760e7092016-04-21 08:09:51 +0000429 handler.handle(Event.of(
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100430 EventKind.DEPCHECKER, null, "Executing " + action.prettyPrint() + ": " + message + "."));
431 }
432 }
433
434 // Called by IncrementalDependencyChecker.
435 protected static void reportUnconditionalExecution(
436 @Nullable EventHandler handler, Action action) {
437 reportRebuild(handler, action, "unconditional execution is requested");
438 }
439
440 private static void reportChanged(@Nullable EventHandler handler, Action action) {
441 reportRebuild(handler, action, "One of the files has changed");
442 }
443
444 private static void reportChangedDeps(@Nullable EventHandler handler, Action action) {
445 reportRebuild(handler, action, "the set of files on which this action depends has changed");
446 }
447
448 private static void reportNewAction(@Nullable EventHandler handler, Action action) {
449 reportRebuild(handler, action, "no entry in the cache (action is new)");
450 }
451
452 private static void reportCorruptedCacheEntry(@Nullable EventHandler handler, Action action) {
453 reportRebuild(handler, action, "cache entry is corrupted");
454 }
455
456 /** Wrapper for all context needed by the ActionCacheChecker to handle a single action. */
457 public static final class Token {
Han-Wen Nienhuysd08b27f2015-02-25 16:45:20 +0100458 private final String cacheKey;
459
460 private Token(String cacheKey) {
461 this.cacheKey = Preconditions.checkNotNull(cacheKey);
462 }
463 }
464}