// Copyright 2016 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.buildeventstream.transports; import static com.google.common.truth.Truth.assertThat; import static org.junit.Assert.assertFalse; import static org.junit.Assert.assertTrue; import static org.mockito.Mockito.when; import com.google.devtools.build.lib.buildeventstream.ArtifactGroupNamer; import com.google.devtools.build.lib.buildeventstream.BuildEvent; import com.google.devtools.build.lib.buildeventstream.BuildEventConverters; import com.google.devtools.build.lib.buildeventstream.BuildEventStreamProtos; import com.google.devtools.build.lib.buildeventstream.BuildEventStreamProtos.BuildStarted; import com.google.devtools.build.lib.buildeventstream.BuildEventStreamProtos.Progress; import com.google.devtools.build.lib.buildeventstream.BuildEventStreamProtos.TargetComplete; import com.google.devtools.build.lib.buildeventstream.PathConverter; import java.io.File; import java.io.FileInputStream; import java.io.InputStream; import java.util.concurrent.Future; import org.junit.After; import org.junit.Before; import org.junit.Rule; import org.junit.Test; import org.junit.rules.TemporaryFolder; import org.junit.runner.RunWith; import org.junit.runners.JUnit4; import org.mockito.Matchers; import org.mockito.Mock; import org.mockito.Mockito; import org.mockito.MockitoAnnotations; /** Tests {@link BinaryFormatFileTransport}. **/ @RunWith(JUnit4.class) public class BinaryFormatFileTransportTest { @Rule public TemporaryFolder tmp = new TemporaryFolder(); @Mock public BuildEvent buildEvent; @Mock public PathConverter pathConverter; @Mock public ArtifactGroupNamer artifactGroupNamer; @Before public void initMocks() { MockitoAnnotations.initMocks(this); } @After public void validateMocks() { Mockito.validateMockitoUsage(); } @Test public void testCreatesFileAndWritesProtoBinaryFormat() throws Exception { File output = tmp.newFile(); BuildEventStreamProtos.BuildEvent started = BuildEventStreamProtos.BuildEvent.newBuilder() .setStarted(BuildStarted.newBuilder().setCommand("build")) .build(); when(buildEvent.asStreamProto(Matchers.<BuildEventConverters>any())).thenReturn(started); BinaryFormatFileTransport transport = new BinaryFormatFileTransport(output.getAbsolutePath(), pathConverter); transport.sendBuildEvent(buildEvent, artifactGroupNamer); BuildEventStreamProtos.BuildEvent progress = BuildEventStreamProtos.BuildEvent.newBuilder().setProgress(Progress.newBuilder()).build(); when(buildEvent.asStreamProto(Matchers.<BuildEventConverters>any())).thenReturn(progress); transport.sendBuildEvent(buildEvent, artifactGroupNamer); BuildEventStreamProtos.BuildEvent completed = BuildEventStreamProtos.BuildEvent.newBuilder() .setCompleted(TargetComplete.newBuilder().setSuccess(true)) .build(); when(buildEvent.asStreamProto(Matchers.<BuildEventConverters>any())).thenReturn(completed); transport.sendBuildEvent(buildEvent, artifactGroupNamer); transport.close().get(); try (InputStream in = new FileInputStream(output)) { assertThat(BuildEventStreamProtos.BuildEvent.parseDelimitedFrom(in)).isEqualTo(started); assertThat(BuildEventStreamProtos.BuildEvent.parseDelimitedFrom(in)).isEqualTo(progress); assertThat(BuildEventStreamProtos.BuildEvent.parseDelimitedFrom(in)).isEqualTo(completed); assertThat(in.available()).isEqualTo(0); } } @Test public void testFileDoesNotExist() throws Exception { // Get a file that doesn't exist by creating a new file and immediately deleting it. File output = tmp.newFile(); String path = output.getAbsolutePath(); assertTrue(output.delete()); BuildEventStreamProtos.BuildEvent started = BuildEventStreamProtos.BuildEvent.newBuilder() .setStarted(BuildStarted.newBuilder().setCommand("build")) .build(); when(buildEvent.asStreamProto(Matchers.<BuildEventConverters>any())).thenReturn(started); BinaryFormatFileTransport transport = new BinaryFormatFileTransport(path, pathConverter); transport.sendBuildEvent(buildEvent, artifactGroupNamer); transport.close().get(); try (InputStream in = new FileInputStream(output)) { assertThat(BuildEventStreamProtos.BuildEvent.parseDelimitedFrom(in)).isEqualTo(started); assertThat(in.available()).isEqualTo(0); } } @Test public void testWriteWhenFileClosed() throws Exception { File output = tmp.newFile(); BuildEventStreamProtos.BuildEvent started = BuildEventStreamProtos.BuildEvent.newBuilder() .setStarted(BuildStarted.newBuilder().setCommand("build")) .build(); when(buildEvent.asStreamProto(Matchers.<BuildEventConverters>any())).thenReturn(started); BinaryFormatFileTransport transport = new BinaryFormatFileTransport(output.getAbsolutePath(), pathConverter); // Close the file. transport.ch.close(); assertFalse(transport.ch.isOpen()); // This should not throw an exception. transport.sendBuildEvent(buildEvent, artifactGroupNamer); transport.close().get(); // Also, nothing should have been written to the file try (InputStream in = new FileInputStream(output)) { assertThat(in.available()).isEqualTo(0); } } @Test public void testWriteWhenTransportClosed() throws Exception { File output = tmp.newFile(); BuildEventStreamProtos.BuildEvent started = BuildEventStreamProtos.BuildEvent.newBuilder() .setStarted(BuildStarted.newBuilder().setCommand("build")) .build(); when(buildEvent.asStreamProto(Matchers.<BuildEventConverters>any())).thenReturn(started); BinaryFormatFileTransport transport = new BinaryFormatFileTransport(output.getAbsolutePath(), pathConverter); transport.sendBuildEvent(buildEvent, artifactGroupNamer); Future<Void> closeFuture = transport.close(); // This should not throw an exception, but also not perform any write. transport.sendBuildEvent(buildEvent, artifactGroupNamer); closeFuture.get(); assertFalse(transport.ch.isOpen()); // There should have only been one write. try (InputStream in = new FileInputStream(output)) { assertThat(BuildEventStreamProtos.BuildEvent.parseDelimitedFrom(in)).isEqualTo(started); assertThat(in.available()).isEqualTo(0); } } }