blob: a301bc6c25c523dbb6e1b922ee7ca78d69182880 [file] [log] [blame]
// 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.actions;
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.ArtifactRoot.RootType;
import com.google.devtools.build.lib.actions.MutableActionGraph.ActionConflictException;
import com.google.devtools.build.lib.actions.util.ActionsTestUtil;
import com.google.devtools.build.lib.actions.util.ActionsTestUtil.UncheckedActionConflictException;
import com.google.devtools.build.lib.actions.util.TestAction;
import com.google.devtools.build.lib.collect.nestedset.NestedSetBuilder;
import com.google.devtools.build.lib.collect.nestedset.Order;
import com.google.devtools.build.lib.concurrent.AbstractQueueVisitor;
import com.google.devtools.build.lib.concurrent.ErrorClassifier;
import com.google.devtools.build.lib.vfs.DigestHashFunction;
import com.google.devtools.build.lib.vfs.FileSystem;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.build.lib.vfs.inmemoryfs.InMemoryFileSystem;
import java.util.Set;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/** Tests for {@link MapBasedActionGraph}. */
@RunWith(JUnit4.class)
public final class MapBasedActionGraphTest {
private final FileSystem fileSystem = new InMemoryFileSystem(DigestHashFunction.SHA256);
private final ActionKeyContext actionKeyContext = new ActionKeyContext();
@Test
public void testSmoke() throws Exception {
MutableActionGraph actionGraph = new MapBasedActionGraph(actionKeyContext);
Path execRoot = fileSystem.getPath("/");
String outSegment = "root";
Path root = execRoot.getChild(outSegment);
Path path = root.getRelative("foo");
Artifact output =
ActionsTestUtil.createArtifact(
ArtifactRoot.asDerivedRoot(execRoot, RootType.Output, outSegment), path);
Action action =
new TestAction(
TestAction.NO_EFFECT,
NestedSetBuilder.emptySet(Order.STABLE_ORDER),
ImmutableSet.of(output));
actionGraph.registerAction(action);
path = root.getRelative("bar");
output =
ActionsTestUtil.createArtifact(
ArtifactRoot.asDerivedRoot(execRoot, RootType.Output, outSegment), path);
Action action2 =
new TestAction(
TestAction.NO_EFFECT,
NestedSetBuilder.emptySet(Order.STABLE_ORDER),
ImmutableSet.of(output));
actionGraph.registerAction(action);
actionGraph.registerAction(action2);
}
@Test
public void testNoActionConflictWhenUnregisteringSharedAction() throws Exception {
MutableActionGraph actionGraph = new MapBasedActionGraph(actionKeyContext);
Path execRoot = fileSystem.getPath("/");
Path root = fileSystem.getPath("/root");
Path path = root.getRelative("foo");
Artifact output =
ActionsTestUtil.createArtifact(
ArtifactRoot.asDerivedRoot(
execRoot, RootType.Output, root.relativeTo(execRoot).getPathString()),
path);
Action action =
new TestAction(
TestAction.NO_EFFECT,
NestedSetBuilder.emptySet(Order.STABLE_ORDER),
ImmutableSet.of(output));
actionGraph.registerAction(action);
Action otherAction =
new TestAction(
TestAction.NO_EFFECT,
NestedSetBuilder.emptySet(Order.STABLE_ORDER),
ImmutableSet.of(output));
actionGraph.registerAction(otherAction);
}
private class ActionRegisterer extends AbstractQueueVisitor {
private final MutableActionGraph graph = new MapBasedActionGraph(new ActionKeyContext());
private final Artifact output;
// Just to occasionally add actions that were already present.
private final Set<Action> allActions = Sets.newConcurrentHashSet();
private final AtomicInteger actionCount = new AtomicInteger(0);
private ActionRegisterer() {
super(
200,
1,
TimeUnit.SECONDS,
/*failFastOnException=*/ true,
"action-graph-test",
ErrorClassifier.DEFAULT);
Path execRoot = fileSystem.getPath("/");
String rootSegment = "root";
Path root = execRoot.getChild(rootSegment);
Path path = root.getChild("foo");
output =
ActionsTestUtil.createArtifact(
ArtifactRoot.asDerivedRoot(execRoot, RootType.Output, rootSegment), path);
allActions.add(
new TestAction(
TestAction.NO_EFFECT,
NestedSetBuilder.emptySet(Order.STABLE_ORDER),
ImmutableSet.of(output)));
}
private void registerAction(Action action) {
execute(
() -> {
try {
graph.registerAction(action);
} catch (ActionConflictException e) {
throw new UncheckedActionConflictException(e);
} catch (InterruptedException e) {
Thread.currentThread().interrupt();
throw new IllegalStateException("Interrupts not expected in this test");
}
doRandom();
});
}
private void doRandom() {
if (actionCount.incrementAndGet() > 10000) {
return;
}
Action action;
if (Math.random() < 0.5) {
action = Iterables.getFirst(allActions, null);
} else {
action =
new TestAction(
TestAction.NO_EFFECT,
NestedSetBuilder.emptySet(Order.STABLE_ORDER),
ImmutableSet.of(output));
allActions.add(action);
}
registerAction(action);
}
private void work() throws InterruptedException {
awaitQuiescence(/*interruptWorkers=*/ true);
}
}
@Test
public void testSharedActionStressTest() throws Exception {
ActionRegisterer actionRegisterer = new ActionRegisterer();
actionRegisterer.doRandom();
actionRegisterer.work();
}
}