Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

fix: protobuf version not always getting set in headers #3322

Merged
merged 12 commits into from
Nov 7, 2024
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,8 @@
package com.google.api.gax.core;

import com.google.api.core.InternalApi;
import com.google.api.gax.util.ClassLoaderWrapper;
import com.google.api.gax.util.IClassLoaderWrapper;
import com.google.common.annotations.VisibleForTesting;
import com.google.common.base.Strings;
import com.google.protobuf.Any;
Expand All @@ -49,7 +51,7 @@ public class GaxProperties {
private static final String GAX_VERSION = getLibraryVersion(GaxProperties.class, "version.gax");
private static final String JAVA_VERSION = getRuntimeVersion();
private static final String PROTOBUF_VERSION =
getBundleVersion(Any.class).orElse(DEFAULT_VERSION);
getProtobufVersion(new ClassLoaderWrapper(), Any.class);

private GaxProperties() {}

Expand Down Expand Up @@ -148,4 +150,31 @@ static Optional<String> getBundleVersion(Class<?> clazz) {
return Optional.empty();
}
}

/**
* Returns the Protobuf runtime version as reported by com.google.protobuf.RuntimeVersion,
* if class is available, otherwise by reading from MANIFEST file. If niether option is available defaults to
* protobuf version 3 as RuntimeVersion class is available in protobuf version 4+
*/
@VisibleForTesting
static String getProtobufVersion(IClassLoaderWrapper classLoader, Class clazz) {
ldetmer marked this conversation as resolved.
Show resolved Hide resolved
try {
Class<?> protobufRuntimeVersionClass =
classLoader.loadClass("com.google.protobuf.RuntimeVersion");
return classLoader.getFieldValue(protobufRuntimeVersionClass, "MAJOR")
+ "."
+ classLoader.getFieldValue(protobufRuntimeVersionClass, "MINOR")
+ "."
+ classLoader.getFieldValue(protobufRuntimeVersionClass, "PATCH");
} catch (ClassNotFoundException | NoSuchFieldException | IllegalAccessException e) {
ldetmer marked this conversation as resolved.
Show resolved Hide resolved
Optional<String> protobufVersionFromManifest = getBundleVersion(clazz);
if (protobufVersionFromManifest.isPresent()) {
return protobufVersionFromManifest.get();
} else {
// If manifest file is not available default to protobuf generic version 3 as we know RuntimeVersion class is
// available in protobuf jar 4+.
return "3";
}
ldetmer marked this conversation as resolved.
Show resolved Hide resolved
}
}
}
Original file line number Diff line number Diff line change
Expand Up @@ -88,7 +88,7 @@ private static String checkAndAppendProtobufVersionIfNecessary(
// TODO(b/366417603): appending protobuf version to existing client library token until resolved
Pattern pattern = Pattern.compile("(gccl|gapic)\\S*");
Matcher matcher = pattern.matcher(apiClientHeaderValue);
if (matcher.find()) {
if (matcher.find() && GaxProperties.getProtobufVersion() != null) {
return apiClientHeaderValue.substring(0, matcher.end())
+ "--"
+ PROTOBUF_HEADER_VERSION_KEY
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,44 @@
/*
* Copyright 2024 Google LLC
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google LLC nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package com.google.api.gax.util;

/* Wrapper class for reflection Class methods to enable unit testing. */
public class ClassLoaderWrapper implements IClassLoaderWrapper {
ldetmer marked this conversation as resolved.
Show resolved Hide resolved
@Override
public Class<?> loadClass(String name) throws ClassNotFoundException {
return Class.forName(name);
}

@Override
public Object getFieldValue(Class<?> clazz, String fieldName)
throws NoSuchFieldException, IllegalAccessException {
return clazz.getField(fieldName).get(null);
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
/*
* Copyright 2024 Google LLC
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * Redistributions in binary form must reproduce the above
* copyright notice, this list of conditions and the following disclaimer
* in the documentation and/or other materials provided with the
* distribution.
* * Neither the name of Google LLC nor the names of its
* contributors may be used to endorse or promote products derived from
* this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS
* "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT
* LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR
* A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT
* OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL,
* SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT
* LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE,
* DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY
* THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE
* OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
package com.google.api.gax.util;

/* Interface that allows for unit testing reflection logic. */
public interface IClassLoaderWrapper {
/* Wraps {@link java.lang.Class#loadClass} method */
ldetmer marked this conversation as resolved.
Show resolved Hide resolved
Class<?> loadClass(String name) throws ClassNotFoundException;

/* Consolidates retrieving a field on a Class object via reflection and retrieving the value of that field */
Object getFieldValue(Class<?> clazz, String fieldName)
throws NoSuchFieldException, IllegalAccessException;
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,10 @@
[
{
"name": "com.google.protobuf.RuntimeVersion",
"fields" : [
{ "name" : "MAJOR" },
{ "name" : "MINOR" },
{ "name" : "PATCH" }
]
}
]
Original file line number Diff line number Diff line change
Expand Up @@ -30,11 +30,14 @@
package com.google.api.gax.core;

import static com.google.api.gax.core.GaxProperties.getBundleVersion;
import static org.junit.jupiter.api.Assertions.assertEquals;
import static org.junit.jupiter.api.Assertions.assertFalse;
import static org.junit.jupiter.api.Assertions.assertTrue;
import static org.junit.jupiter.api.Assertions.*;
import static org.mockito.ArgumentMatchers.*;
import static org.mockito.Mockito.mock;
import static org.mockito.Mockito.when;

import com.google.api.gax.util.*;
import com.google.common.base.Strings;
import com.google.protobuf.*;
import java.io.IOException;
import java.util.Optional;
import java.util.regex.Pattern;
Expand Down Expand Up @@ -160,12 +163,8 @@ void testGetJavaRuntimeInfo_nullJavaVersion() {

@Test
public void testGetProtobufVersion() throws IOException {
Version version = readVersion(GaxProperties.getProtobufVersion());

assertTrue(version.major >= 3);
if (version.major == 3) {
assertTrue(version.minor >= 25);
}
assertTrue(
Pattern.compile("^\\d+\\.\\d+\\.\\d+").matcher(GaxProperties.getProtobufVersion()).find());
}

@Test
Expand All @@ -175,6 +174,52 @@ public void testGetBundleVersion_noManifestFile() throws IOException {
assertFalse(version.isPresent());
}

@Test
void testGetProtobufVersion_success() throws Exception {
IClassLoaderWrapper mockClassLoader = mock(IClassLoaderWrapper.class);
ldetmer marked this conversation as resolved.
Show resolved Hide resolved
when(mockClassLoader.loadClass("com.google.protobuf.RuntimeVersion"))
.thenAnswer(invocationOnMock -> Class.class);
when(mockClassLoader.getFieldValue(Class.class, "MAJOR")).thenReturn("2");
when(mockClassLoader.getFieldValue(Class.class, "MINOR")).thenReturn("3");
when(mockClassLoader.getFieldValue(Class.class, "PATCH")).thenReturn("4");

String version = GaxProperties.getProtobufVersion(mockClassLoader, Any.class);

assertEquals("2.3.4", version);
}

@Test
void testGetProtobufVersion_classNotFoundException() throws Exception {
IClassLoaderWrapper mockClassLoader = mock(IClassLoaderWrapper.class);
when(mockClassLoader.loadClass("com.google.protobuf.RuntimeVersion"))
.thenThrow(new ClassNotFoundException(""));

String version = GaxProperties.getProtobufVersion(mockClassLoader, Any.class);

assertTrue(Pattern.compile("^\\d+\\.\\d+\\.\\d+").matcher(version).find());
}

@Test
void testgetProtobufVersion_noSuchFieldException() throws Exception {
IClassLoaderWrapper mockClassLoader = mock(IClassLoaderWrapper.class);
when(mockClassLoader.getFieldValue(any(), any())).thenThrow(NoSuchFieldException.class);

String version = GaxProperties.getProtobufVersion(mockClassLoader, Any.class);

assertTrue(Pattern.compile("^\\d+\\.\\d+\\.\\d+").matcher(version).find());
}

@Test
void testGetProtobufVersion_noManifest() throws Exception {
IClassLoaderWrapper mockClassLoader = mock(IClassLoaderWrapper.class);
when(mockClassLoader.loadClass("com.google.protobuf.RuntimeVersion"))
.thenThrow(new ClassNotFoundException(""));

String version = GaxProperties.getProtobufVersion(mockClassLoader, GaxProperties.class);

assertEquals("3", version);
}

private Version readVersion(String version) {
assertTrue(Pattern.compile("^\\d+\\.\\d+\\.\\d+").matcher(version).find());
String[] versionComponents = version.split("\\.");
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -241,7 +241,7 @@ void testHttpJsonCompliance_userApiVersionSetSuccess() throws IOException {
@Test
void testGrpcCall_sendsCorrectApiClientHeader() {
Pattern defautlGrpcHeaderPattern =
Pattern.compile("gl-java/.* gapic/.*?--protobuf-.* gax/.* grpc/.* protobuf/.*");
Pattern.compile("gl-java/.* gapic/.*?--protobuf-\\d.* gax/.* grpc/.* protobuf/\\d.*");
grpcClient.echo(EchoRequest.newBuilder().build());
String headerValue = grpcInterceptor.metadata.get(API_CLIENT_HEADER_KEY);
assertTrue(defautlGrpcHeaderPattern.matcher(headerValue).matches());
Expand All @@ -250,7 +250,7 @@ void testGrpcCall_sendsCorrectApiClientHeader() {
@Test
void testHttpJson_sendsCorrectApiClientHeader() {
Pattern defautlHttpHeaderPattern =
Pattern.compile("gl-java/.* gapic/.*?--protobuf-.* gax/.* rest/ protobuf/.*");
Pattern.compile("gl-java/.* gapic/.*?--protobuf-\\d.* gax/.* rest/ protobuf/\\d.*");
httpJsonClient.echo(EchoRequest.newBuilder().build());
ArrayList<String> headerValues =
(ArrayList<String>)
Expand Down
Loading