Adds a CODECs for Package and PackageValue.
PiperOrigin-RevId: 181624201
diff --git a/src/main/java/com/google/devtools/build/lib/packages/Package.java b/src/main/java/com/google/devtools/build/lib/packages/Package.java
index 1bceb6b..985dedf 100644
--- a/src/main/java/com/google/devtools/build/lib/packages/Package.java
+++ b/src/main/java/com/google/devtools/build/lib/packages/Package.java
@@ -35,11 +35,15 @@
import com.google.devtools.build.lib.events.Location;
import com.google.devtools.build.lib.packages.AttributeMap.AcceptsLabelAttribute;
import com.google.devtools.build.lib.packages.License.DistributionType;
+import com.google.devtools.build.lib.skyframe.serialization.InjectingObjectCodec;
+import com.google.devtools.build.lib.skyframe.serialization.SerializationException;
import com.google.devtools.build.lib.syntax.SkylarkSemantics;
import com.google.devtools.build.lib.util.SpellChecker;
import com.google.devtools.build.lib.vfs.Canonicalizer;
import com.google.devtools.build.lib.vfs.Path;
import com.google.devtools.build.lib.vfs.PathFragment;
+import com.google.protobuf.CodedInputStream;
+import com.google.protobuf.CodedOutputStream;
import java.io.IOException;
import java.io.PrintStream;
import java.util.ArrayList;
@@ -53,16 +57,18 @@
import javax.annotation.Nullable;
/**
- * A package, which is a container of {@link Rule}s, each of
- * which contains a dictionary of named attributes.
+ * A package, which is a container of {@link Rule}s, each of which contains a dictionary of named
+ * attributes.
*
- * <p>Package instances are intended to be immutable and for all practical
- * purposes can be treated as such. Note, however, that some member variables
- * exposed via the public interface are not strictly immutable, so until their
- * types are guaranteed immutable we're not applying the {@code @Immutable}
- * annotation here.
+ * <p>Package instances are intended to be immutable and for all practical purposes can be treated
+ * as such. Note, however, that some member variables exposed via the public interface are not
+ * strictly immutable, so until their types are guaranteed immutable we're not applying the
+ * {@code @Immutable} annotation here.
*/
+@SuppressWarnings("JavaLangClash")
public class Package {
+ public static final InjectingObjectCodec<Package, PackageCodecDependencies> CODEC =
+ new PackageCodec();
/**
* Common superclass for all name-conflict exceptions.
@@ -1552,4 +1558,33 @@
return message + ", defined at " + target.getLocation();
}
}
+
+ /** Package codec implementation. */
+ private static final class PackageCodec
+ implements InjectingObjectCodec<Package, PackageCodecDependencies> {
+ @Override
+ public Class<Package> getEncodedClass() {
+ return Package.class;
+ }
+
+ @Override
+ public void serialize(
+ PackageCodecDependencies codecDeps, Package input, CodedOutputStream codedOut)
+ throws IOException {
+ codecDeps.getPackageSerializer().serialize(input, codedOut);
+ }
+
+ @Override
+ public Package deserialize(PackageCodecDependencies codecDeps, CodedInputStream codedIn)
+ throws SerializationException, IOException {
+ try {
+ return codecDeps.getPackageDeserializer().deserialize(codedIn);
+ } catch (PackageDeserializationException e) {
+ throw new SerializationException("Failed to deserialize Package", e);
+ } catch (InterruptedException e) {
+ throw new IllegalStateException(
+ "Unexpected InterruptedException during Package deserialization", e);
+ }
+ }
+ }
}
diff --git a/src/main/java/com/google/devtools/build/lib/packages/PackageCodecDependencies.java b/src/main/java/com/google/devtools/build/lib/packages/PackageCodecDependencies.java
new file mode 100644
index 0000000..52dc91b
--- /dev/null
+++ b/src/main/java/com/google/devtools/build/lib/packages/PackageCodecDependencies.java
@@ -0,0 +1,46 @@
+// Copyright 2018 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.packages;
+
+/** Dependencies of {@value Package#Codec}. */
+public interface PackageCodecDependencies {
+
+ PackageSerializerInterface getPackageSerializer();
+
+ PackageDeserializerInterface getPackageDeserializer();
+
+ /** Simplest implementation of PackageCodecDependencies. */
+ public static class SimplePackageCodecDependencies implements PackageCodecDependencies {
+ private final PackageSerializerInterface packageSerializer;
+ private final PackageDeserializerInterface packageDeserializer;
+
+ public SimplePackageCodecDependencies(
+ PackageSerializerInterface packageSerializer,
+ PackageDeserializerInterface packageDeserializer) {
+ this.packageSerializer = packageSerializer;
+ this.packageDeserializer = packageDeserializer;
+ }
+
+ @Override
+ public PackageSerializerInterface getPackageSerializer() {
+ return packageSerializer;
+ }
+
+ @Override
+ public PackageDeserializerInterface getPackageDeserializer() {
+ return packageDeserializer;
+ }
+ }
+}
diff --git a/src/main/java/com/google/devtools/build/lib/packages/PackageDeserializationException.java b/src/main/java/com/google/devtools/build/lib/packages/PackageDeserializationException.java
new file mode 100644
index 0000000..992c2ab
--- /dev/null
+++ b/src/main/java/com/google/devtools/build/lib/packages/PackageDeserializationException.java
@@ -0,0 +1,30 @@
+// Copyright 2018 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.packages;
+
+/** Exception thrown when something goes wrong during package deserialization. */
+public class PackageDeserializationException extends Exception {
+ PackageDeserializationException(String message) {
+ super(message);
+ }
+
+ PackageDeserializationException(String message, Exception reason) {
+ super(message, reason);
+ }
+
+ PackageDeserializationException(Exception reason) {
+ super(reason);
+ }
+}
diff --git a/src/main/java/com/google/devtools/build/lib/packages/PackageDeserializerInterface.java b/src/main/java/com/google/devtools/build/lib/packages/PackageDeserializerInterface.java
new file mode 100644
index 0000000..ea74a4d
--- /dev/null
+++ b/src/main/java/com/google/devtools/build/lib/packages/PackageDeserializerInterface.java
@@ -0,0 +1,39 @@
+// Copyright 2018 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.packages;
+
+import com.google.protobuf.CodedInputStream;
+import java.io.IOException;
+
+/**
+ * Interface for Package deserialization.
+ *
+ * <p>Provides a layer of indirection for breaking circular dependencies.
+ */
+public interface PackageDeserializerInterface {
+
+ /**
+ * Deserializes a {@link Package} from {@code codedIn}. The inverse of {@link
+ * PackageSerializer#serialize}.
+ *
+ * @param codedIn stream to read from
+ * @return a new {@link Package} as read from {@code codedIn}
+ * @throws PackageDeserializationException on failures deserializing the input
+ * @throws IOException on failures reading from {@code codedIn}
+ * @throws InterruptedException
+ */
+ Package deserialize(CodedInputStream codedIn)
+ throws PackageDeserializationException, IOException, InterruptedException;
+}
diff --git a/src/main/java/com/google/devtools/build/lib/packages/PackageSerializerInterface.java b/src/main/java/com/google/devtools/build/lib/packages/PackageSerializerInterface.java
new file mode 100644
index 0000000..af25075
--- /dev/null
+++ b/src/main/java/com/google/devtools/build/lib/packages/PackageSerializerInterface.java
@@ -0,0 +1,35 @@
+// Copyright 2018 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.packages;
+
+import com.google.protobuf.CodedOutputStream;
+import java.io.IOException;
+
+/**
+ * Abstraction layer for Package serialization.
+ *
+ * <p>Provides a layer of indirection for breaking circular dependencies.
+ */
+public interface PackageSerializerInterface {
+ /**
+ * Serialize a package to {@code codedOut}. The inverse of {@link
+ * PackageDeserializer#deserialize}.
+ *
+ * @param pkg the {@link Package} to be serialized
+ * @param codedOut the stream to write {@code pkg}'s serialized representation to
+ * @throws IOException on failure writing to {@code codedOut}
+ */
+ void serialize(Package pkg, CodedOutputStream codedOut) throws IOException;
+}
diff --git a/src/main/java/com/google/devtools/build/lib/skyframe/PackageValue.java b/src/main/java/com/google/devtools/build/lib/skyframe/PackageValue.java
index c5d9094..f31bf09 100644
--- a/src/main/java/com/google/devtools/build/lib/skyframe/PackageValue.java
+++ b/src/main/java/com/google/devtools/build/lib/skyframe/PackageValue.java
@@ -19,18 +19,22 @@
import com.google.devtools.build.lib.concurrent.ThreadSafety.ThreadSafe;
import com.google.devtools.build.lib.packages.BuildFileContainsErrorsException;
import com.google.devtools.build.lib.packages.Package;
+import com.google.devtools.build.lib.packages.PackageCodecDependencies;
+import com.google.devtools.build.lib.skyframe.serialization.InjectingObjectCodec;
+import com.google.devtools.build.lib.skyframe.serialization.autocodec.AutoCodec;
import com.google.devtools.build.skyframe.LegacySkyKey;
import com.google.devtools.build.skyframe.NotComparableSkyValue;
import com.google.devtools.build.skyframe.SkyKey;
import java.util.ArrayList;
import java.util.List;
-/**
- * A Skyframe value representing a package.
- */
+/** A Skyframe value representing a package. */
+@AutoCodec(dependency = PackageCodecDependencies.class)
@Immutable
@ThreadSafe
public class PackageValue implements NotComparableSkyValue {
+ public static final InjectingObjectCodec<PackageValue, PackageCodecDependencies> CODEC =
+ new PackageValue_AutoCodec();
private final Package pkg;
@@ -65,5 +69,4 @@
}
return keys;
}
-
}
diff --git a/src/test/java/com/google/devtools/build/lib/skyframe/BUILD b/src/test/java/com/google/devtools/build/lib/skyframe/BUILD
index a4b01da..e9cfbfd 100644
--- a/src/test/java/com/google/devtools/build/lib/skyframe/BUILD
+++ b/src/test/java/com/google/devtools/build/lib/skyframe/BUILD
@@ -78,6 +78,7 @@
"//src/main/java/com/google/devtools/build/lib/collect/nestedset",
"//src/main/java/com/google/devtools/build/lib/concurrent",
"//src/main/java/com/google/devtools/build/lib/rules/cpp",
+ "//src/main/java/com/google/devtools/build/lib/skyframe/serialization",
"//src/main/java/com/google/devtools/build/lib/skyframe/serialization/testutils",
"//src/main/java/com/google/devtools/build/lib/vfs",
"//src/main/java/com/google/devtools/build/lib/vfs/inmemoryfs",
@@ -97,6 +98,7 @@
"//third_party:junit4",
"//third_party:mockito",
"//third_party:truth",
+ "//third_party/protobuf:protobuf_java",
],
)
diff --git a/src/test/java/com/google/devtools/build/lib/skyframe/PackageValueTest.java b/src/test/java/com/google/devtools/build/lib/skyframe/PackageValueTest.java
new file mode 100644
index 0000000..2947249
--- /dev/null
+++ b/src/test/java/com/google/devtools/build/lib/skyframe/PackageValueTest.java
@@ -0,0 +1,85 @@
+// Copyright 2018 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.skyframe;
+
+import static com.google.common.truth.Truth.assertThat;
+import static org.junit.Assert.fail;
+import static org.mockito.Matchers.any;
+import static org.mockito.Mockito.doThrow;
+import static org.mockito.Mockito.mock;
+import static org.mockito.Mockito.when;
+
+import com.google.devtools.build.lib.packages.Package;
+import com.google.devtools.build.lib.packages.PackageCodecDependencies.SimplePackageCodecDependencies;
+import com.google.devtools.build.lib.packages.PackageDeserializationException;
+import com.google.devtools.build.lib.packages.PackageDeserializerInterface;
+import com.google.devtools.build.lib.skyframe.serialization.InjectingObjectCodecAdapter;
+import com.google.devtools.build.lib.skyframe.serialization.ObjectCodec;
+import com.google.devtools.build.lib.skyframe.serialization.SerializationException;
+import com.google.protobuf.CodedInputStream;
+import java.io.IOException;
+import org.junit.Before;
+import org.junit.Test;
+import org.junit.runner.RunWith;
+import org.junit.runners.JUnit4;
+import org.mockito.ArgumentCaptor;
+
+/** Basic tests for {@link PackageValue}. */
+@RunWith(JUnit4.class)
+public class PackageValueTest {
+
+ private PackageDeserializerInterface mockDeserializer;
+ private ObjectCodec<PackageValue> underTest;
+
+ @Before
+ public void setUp() {
+ this.mockDeserializer = mock(PackageDeserializerInterface.class);
+ this.underTest =
+ new InjectingObjectCodecAdapter<>(
+ PackageValue.CODEC, new SimplePackageCodecDependencies(null, mockDeserializer));
+ }
+
+ @Test
+ public void testDeserializationIsDelegatedToPackageDeserializer()
+ throws SerializationException, IOException, PackageDeserializationException,
+ InterruptedException {
+ // Mock because all we need is to verify that we're properly delegating to Package deserializer.
+ Package mockPackage = mock(Package.class);
+
+ when(mockDeserializer.deserialize(ArgumentCaptor.forClass(CodedInputStream.class).capture()))
+ .thenReturn(mockPackage);
+
+ CodedInputStream codedIn = CodedInputStream.newInstance(new byte[] {1, 2, 3, 4});
+ PackageValue result = underTest.deserialize(codedIn);
+
+ assertThat(result.getPackage()).isSameAs(mockPackage);
+ }
+
+ @Test
+ public void testInterruptedExceptionRaisesIllegalStateException() throws Exception {
+ InterruptedException staged = new InterruptedException("Stop that!");
+ doThrow(staged).when(mockDeserializer).deserialize(any(CodedInputStream.class));
+
+ try {
+ underTest.deserialize(CodedInputStream.newInstance(new byte[] {1, 2, 3, 4}));
+ fail("Expected exception");
+ } catch (IllegalStateException e) {
+ assertThat(e)
+ .hasMessageThat()
+ .isEqualTo("Unexpected InterruptedException during Package deserialization");
+ assertThat(e).hasCauseThat().isSameAs(staged);
+ }
+ }
+}