blob: 3dfbf0eb968d22a175a768c12d7c2f7a5c2b84c2 [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.vfs;
import static com.google.common.truth.Truth.assertThat;
import com.google.devtools.build.lib.testutil.TestThread;
import com.google.devtools.build.lib.testutil.TestUtils;
import com.google.devtools.build.lib.vfs.inmemoryfs.InMemoryFileSystem;
import java.io.IOException;
import java.util.concurrent.CountDownLatch;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.atomic.AtomicInteger;
import org.junit.After;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.JUnit4;
/** Tests for {@link DigestUtils}. */
@RunWith(JUnit4.class)
public final class DigestUtilsTest {
@After
public void tearDown() {
DigestUtils.configureCache(/*maximumSize=*/ 0);
}
private static void assertDigestCalculationConcurrency(
boolean expectConcurrent,
boolean fastDigest,
int fileSize1,
int fileSize2,
DigestHashFunction hf)
throws Exception {
CountDownLatch barrierLatch = new CountDownLatch(2); // Used to block test threads.
CountDownLatch readyLatch = new CountDownLatch(1); // Used to block main thread.
FileSystem myfs =
new InMemoryFileSystem(hf) {
@Override
protected byte[] getDigest(PathFragment path) throws IOException {
try {
barrierLatch.countDown();
readyLatch.countDown();
// Either both threads will be inside getDigest at the same time or they
// both will be blocked.
barrierLatch.await();
} catch (Exception e) {
throw new IOException(e);
}
return super.getDigest(path);
}
@Override
protected byte[] getFastDigest(PathFragment path) throws IOException {
return fastDigest ? super.getDigest(path) : null;
}
};
Path myFile1 = myfs.getPath("/f1.dat");
Path myFile2 = myfs.getPath("/f2.dat");
FileSystemUtils.writeContentAsLatin1(myFile1, "a".repeat(fileSize1));
FileSystemUtils.writeContentAsLatin1(myFile2, "b".repeat(fileSize2));
TestThread thread1 =
new TestThread(
() ->
DigestUtils.getDigestWithManualFallback(myFile1, fileSize1, SyscallCache.NO_CACHE));
TestThread thread2 =
new TestThread(
() ->
DigestUtils.getDigestWithManualFallback(myFile2, fileSize2, SyscallCache.NO_CACHE));
thread1.start();
thread2.start();
if (!expectConcurrent) { // Synchronized case.
// Wait until at least one thread reached getDigest().
assertThat(readyLatch.await(TestUtils.WAIT_TIMEOUT_SECONDS, TimeUnit.SECONDS)).isTrue();
// Only 1 thread should be inside getDigest().
assertThat(barrierLatch.getCount()).isEqualTo(1);
barrierLatch.countDown(); // Release barrier latch, allowing both threads to proceed.
}
// Test successful execution within 5 seconds.
thread1.joinAndAssertState(TestUtils.WAIT_TIMEOUT_MILLISECONDS);
thread2.joinAndAssertState(TestUtils.WAIT_TIMEOUT_MILLISECONDS);
}
@Test
public void testCache() throws Exception {
AtomicInteger getFastDigestCounter = new AtomicInteger(0);
AtomicInteger getDigestCounter = new AtomicInteger(0);
FileSystem tracingFileSystem =
new InMemoryFileSystem(DigestHashFunction.SHA256) {
@Override
protected byte[] getFastDigest(PathFragment path) {
getFastDigestCounter.incrementAndGet();
return null;
}
@Override
protected byte[] getDigest(PathFragment path) throws IOException {
getDigestCounter.incrementAndGet();
return super.getDigest(path);
}
};
DigestUtils.configureCache(/*maximumSize=*/ 100);
Path file = tracingFileSystem.getPath("/file.txt");
FileSystemUtils.writeContentAsLatin1(file, "some contents");
byte[] digest1 =
DigestUtils.getDigestWithManualFallback(file, file.getFileSize(), SyscallCache.NO_CACHE);
assertThat(getFastDigestCounter.get()).isEqualTo(1);
assertThat(getDigestCounter.get()).isEqualTo(1);
assertThat(
DigestUtils.getDigestWithManualFallback(
file, file.getFileSize(), SyscallCache.NO_CACHE))
.isEqualTo(digest1);
assertThat(getFastDigestCounter.get()).isEqualTo(2);
assertThat(getDigestCounter.get()).isEqualTo(1); // Cached.
}
@Test
public void manuallyComputeDigest() throws Exception {
byte[] digest = {1, 2, 3};
FileSystem noDigestFileSystem =
new InMemoryFileSystem(DigestHashFunction.SHA256) {
@Override
protected byte[] getFastDigest(PathFragment path) {
throw new AssertionError("Unexpected call to getFastDigest");
}
@Override
protected byte[] getDigest(PathFragment path) {
return digest;
}
};
Path file = noDigestFileSystem.getPath("/f.txt");
FileSystemUtils.writeContentAsLatin1(file, "contents");
assertThat(DigestUtils.manuallyComputeDigest(file, /*fileSize=*/ 8)).isEqualTo(digest);
}
}