blob: d1f426d86381699abdce00d0eb5eeffce54bd152 [file] [log] [blame]
// Copyright 2020 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.worker;
import static com.google.common.truth.Truth.assertThat;
import static com.google.common.truth.Truth.assertWithMessage;
import static com.google.devtools.build.lib.worker.TestUtils.createWorkerKey;
import static org.mockito.ArgumentMatchers.any;
import static org.mockito.Mockito.doAnswer;
import static org.mockito.Mockito.times;
import static org.mockito.Mockito.verify;
import static org.mockito.Mockito.when;
import com.google.common.collect.ImmutableList;
import com.google.common.collect.ImmutableMap;
import com.google.common.collect.Lists;
import com.google.devtools.build.lib.clock.BlazeClock;
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.io.IOException;
import java.lang.Thread.State;
import org.apache.commons.pool2.impl.DefaultPooledObject;
import org.junit.Before;
import org.junit.Rule;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
import org.mockito.Mock;
import org.mockito.junit.MockitoJUnit;
import org.mockito.junit.MockitoRule;
/** Tests WorkerPool. */
@RunWith(JUnit4.class)
public class WorkerPoolTest {
@Rule public final MockitoRule mockito = MockitoJUnit.rule();
@Mock WorkerFactory factoryMock;
private FileSystem fileSystem;
private int workerIds = 1;
private static class TestWorker extends SingleplexWorker {
TestWorker(WorkerKey workerKey, int workerId, Path workDir, Path logFile) {
super(workerKey, workerId, workDir, logFile);
}
}
@Before
public void setUp() throws Exception {
fileSystem = new InMemoryFileSystem(BlazeClock.instance(), DigestHashFunction.SHA256);
doAnswer(
arg -> {
return new DefaultPooledObject<>(
new TestWorker(
arg.getArgument(0),
workerIds++,
fileSystem.getPath("/workDir"),
fileSystem.getPath("/logDir")));
})
.when(factoryMock)
.makeObject(any());
when(factoryMock.validateObject(any(), any())).thenReturn(true);
}
@Test
public void testBorrow_createsWhenNeeded() throws Exception {
WorkerPool workerPool =
new WorkerPool(
factoryMock,
ImmutableMap.of("mnem", 2, "", 1),
ImmutableMap.of(),
Lists.newArrayList());
WorkerKey workerKey = createWorkerKey(fileSystem, "mnem", false);
Worker worker1 = workerPool.borrowObject(workerKey);
Worker worker2 = workerPool.borrowObject(workerKey);
assertThat(worker1.getWorkerId()).isEqualTo(1);
assertThat(worker2.getWorkerId()).isEqualTo(2);
verify(factoryMock, times(2)).makeObject(workerKey);
}
@Test
public void testBorrow_reusesWhenPossible() throws Exception {
WorkerPool workerPool =
new WorkerPool(
factoryMock,
ImmutableMap.of("mnem", 2, "", 1),
ImmutableMap.of(),
Lists.newArrayList());
WorkerKey workerKey = createWorkerKey(fileSystem, "mnem", false);
Worker worker1 = workerPool.borrowObject(workerKey);
workerPool.returnObject(workerKey, worker1);
Worker worker2 = workerPool.borrowObject(workerKey);
assertThat(worker1).isSameInstanceAs(worker2);
verify(factoryMock, times(1)).makeObject(workerKey);
}
@Test
public void testBorrow_usesDefault() throws Exception {
WorkerPool workerPool =
new WorkerPool(
factoryMock,
ImmutableMap.of("mnem", 2, "", 1),
ImmutableMap.of(),
Lists.newArrayList());
WorkerKey workerKey1 = createWorkerKey(fileSystem, "mnem", false);
Worker worker1 = workerPool.borrowObject(workerKey1);
Worker worker1a = workerPool.borrowObject(workerKey1);
assertThat(worker1.getWorkerId()).isEqualTo(1);
assertThat(worker1a.getWorkerId()).isEqualTo(2);
WorkerKey workerKey2 = createWorkerKey(fileSystem, "other", false);
Worker worker2 = workerPool.borrowObject(workerKey2);
assertThat(worker2.getWorkerId()).isEqualTo(3);
verify(factoryMock, times(2)).makeObject(workerKey1);
verify(factoryMock, times(1)).makeObject(workerKey2);
}
@Test
public void testBorrow_pooledByKey() throws Exception {
WorkerPool workerPool =
new WorkerPool(
factoryMock,
ImmutableMap.of("mnem", 2, "", 1),
ImmutableMap.of(),
Lists.newArrayList());
WorkerKey workerKey1 = createWorkerKey(fileSystem, "mnem", false);
Worker worker1 = workerPool.borrowObject(workerKey1);
Worker worker1a = workerPool.borrowObject(workerKey1);
assertThat(worker1.getWorkerId()).isEqualTo(1);
assertThat(worker1a.getWorkerId()).isEqualTo(2);
WorkerKey workerKey2 = createWorkerKey(fileSystem, "mnem", false, "arg1");
Worker worker2 = workerPool.borrowObject(workerKey2);
assertThat(worker2.getWorkerId()).isEqualTo(3);
verify(factoryMock, times(2)).makeObject(workerKey1);
verify(factoryMock, times(1)).makeObject(workerKey2);
}
@Test
public void testBorrow_separateMultiplexWorkers() throws Exception {
WorkerPool workerPool =
new WorkerPool(
factoryMock,
ImmutableMap.of("mnem", 1, "", 1),
ImmutableMap.of("mnem", 2, "", 1),
Lists.newArrayList());
WorkerKey workerKey = createWorkerKey(fileSystem, "mnem", false);
Worker worker1 = workerPool.borrowObject(workerKey);
assertThat(worker1.getWorkerId()).isEqualTo(1);
workerPool.returnObject(workerKey, worker1);
WorkerKey multiplexKey = createWorkerKey(fileSystem, "mnem", true);
Worker multiplexWorker1 = workerPool.borrowObject(multiplexKey);
Worker multiplexWorker2 = workerPool.borrowObject(multiplexKey);
Worker worker1a = workerPool.borrowObject(workerKey);
assertThat(multiplexWorker1.getWorkerId()).isEqualTo(2);
assertThat(multiplexWorker2.getWorkerId()).isEqualTo(3);
assertThat(worker1a.getWorkerId()).isEqualTo(1);
verify(factoryMock, times(1)).makeObject(workerKey);
verify(factoryMock, times(2)).makeObject(multiplexKey);
}
@Test
public void testBorrow_allowsOneHiPrio() throws Exception {
WorkerPool workerPool =
new WorkerPool(
factoryMock,
ImmutableMap.of("loprio", 2, "hiprio", 2, "", 1),
ImmutableMap.of(),
ImmutableList.of("hiprio"));
WorkerKey workerKey1 = createWorkerKey(fileSystem, "hiprio", false);
Worker worker1 = workerPool.borrowObject(workerKey1);
assertThat(worker1.getWorkerId()).isEqualTo(1);
// A single hiprio worker should not block.
WorkerKey workerKey2 = createWorkerKey(fileSystem, "loprio", false);
Worker worker2 = workerPool.borrowObject(workerKey2);
assertThat(worker2.getWorkerId()).isEqualTo(2);
verify(factoryMock, times(1)).makeObject(workerKey1);
verify(factoryMock, times(1)).makeObject(workerKey2);
}
@Test
public void testBorrow_twoHiPrioBlocks() throws Exception {
WorkerPool workerPool =
new WorkerPool(
factoryMock,
ImmutableMap.of("loprio", 2, "hiprio", 2, "", 1),
ImmutableMap.of(),
ImmutableList.of("hiprio"));
WorkerKey workerKey1 = createWorkerKey(fileSystem, "hiprio", false);
Worker worker1 = workerPool.borrowObject(workerKey1);
Worker worker1a = workerPool.borrowObject(workerKey1);
assertThat(worker1.getWorkerId()).isEqualTo(1);
assertThat(worker1a.getWorkerId()).isEqualTo(2);
WorkerKey workerKey2 = createWorkerKey(fileSystem, "loprio", false);
Thread t =
new Thread(
() -> {
try {
workerPool.borrowObject(workerKey2);
} catch (IOException | InterruptedException e) {
// Ignorable
}
});
t.start();
boolean waited = false;
for (int tries = 0; tries < 1000; tries++) {
if (t.getState() == State.WAITING) {
waited = true;
break;
}
Thread.sleep(1);
}
assertWithMessage("Expected low-priority worker to wait").that(waited).isTrue();
workerPool.returnObject(workerKey1, worker1);
boolean continued = false;
for (int tries = 0; tries < 1000; tries++) {
if (t.getState() != State.WAITING) {
continued = true;
break;
}
Thread.sleep(1);
}
assertWithMessage("Expected low-priority worker to eventually continue")
.that(continued)
.isTrue();
verify(factoryMock, times(2)).makeObject(workerKey1);
verify(factoryMock, times(1)).makeObject(workerKey2);
}
@Test
public void testStopWork_activePoolsStopped() throws Exception {
WorkerPool pool =
new WorkerPool(
factoryMock,
// Have to declare the mnemonics, or they all fall into the default SimpleWorkerPool.
ImmutableMap.of("mnem1", 2, "mnem2", 2),
ImmutableMap.of("mnem2", 2, "mnem3", 2),
Lists.newArrayList());
WorkerKey singleKey1 = createWorkerKey(fileSystem, "mnem1", false);
// These workers get borrowed, then both get destroyed in stopWork because they share mnemonic
WorkerKey singleKey1a = createWorkerKey(fileSystem, "mnem1", false, "anArg");
pool.borrowObject(singleKey1);
Worker worker1a = pool.borrowObject(singleKey1a);
pool.returnObject(singleKey1a, worker1a);
WorkerKey singleKey2 = createWorkerKey(fileSystem, "mnem2", false);
// This worker gets borrowed, then returned, doesn't get destroyed in stopWork
Worker worker1 = pool.borrowObject(singleKey2);
pool.returnObject(singleKey2, worker1);
WorkerKey multiKey1 = createWorkerKey(fileSystem, "mnem2", true);
// This worker gets borrowed, then destroyed in stopWork, but separately from the singleplex
// worker2 even though they share a mnemonic.
pool.borrowObject(multiKey1);
WorkerKey multiKey2 = createWorkerKey(fileSystem, "mnem3", true);
// This worker gets borrowed, then returned, doesn't get destroyed during stopWork.
Worker worker2 = pool.borrowObject(multiKey2);
pool.returnObject(multiKey2, worker2);
verify(factoryMock, times(1)).makeObject(singleKey1);
verify(factoryMock, times(1)).makeObject(singleKey1a);
verify(factoryMock, times(1)).makeObject(singleKey2);
verify(factoryMock, times(1)).makeObject(multiKey1);
verify(factoryMock, times(1)).makeObject(multiKey2);
pool.stopWork();
pool.borrowObject(singleKey1);
pool.borrowObject(singleKey1a);
pool.borrowObject(singleKey2);
pool.borrowObject(multiKey1);
pool.borrowObject(multiKey2);
// After stopWork, we had to create new workers for the keys that got their pools destroyed.
verify(factoryMock, times(2)).makeObject(singleKey1);
verify(factoryMock, times(2)).makeObject(singleKey1a);
verify(factoryMock, times(1)).makeObject(singleKey2);
verify(factoryMock, times(2)).makeObject(multiKey1);
verify(factoryMock, times(1)).makeObject(multiKey2);
}
}