-
Notifications
You must be signed in to change notification settings - Fork 173
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
Gzip compress gRPC responses #3387
Changes from all commits
617cd48
53da48f
33f0ce7
f7e81e8
a742221
c8d2610
f429ae2
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change | ||
---|---|---|---|---|
|
@@ -12,11 +12,14 @@ import okio.BufferedSink | |||
* https://github.com/square/wire/search?q=GrpcMessageSink&type=Code | ||||
* | ||||
* @param sink the HTTP/2 stream body. | ||||
* @param minMessageToCompress the minimum message size for compression | ||||
* when [grpcEncoding] is not "identity". | ||||
* @param messageAdapter a proto adapter for each message. | ||||
* @param grpcEncoding the content coding for the stream body. | ||||
*/ | ||||
internal class GrpcMessageSink<T : Any> constructor( | ||||
internal class GrpcMessageSink<T : Any>( | ||||
private val sink: BufferedSink, | ||||
private val minMessageToCompress: Long, | ||||
private val messageAdapter: ProtoAdapter<T>, | ||||
private val grpcEncoding: String | ||||
) : MessageSink<T> { | ||||
|
@@ -25,15 +28,23 @@ internal class GrpcMessageSink<T : Any> constructor( | |||
check(!closed) { "closed" } | ||||
|
||||
val encodedMessage = Buffer() | ||||
grpcEncoding.toGrpcEncoder().encode(encodedMessage).use { encodingSink -> | ||||
messageAdapter.encode(encodingSink, message) | ||||
messageAdapter.encode(encodedMessage, message) | ||||
|
||||
if (grpcEncoding == "identity" || encodedMessage.size < minMessageToCompress) { | ||||
sink.writeByte(0) // 0 = Not encoded. | ||||
sink.writeInt(encodedMessage.size.toInt()) | ||||
sink.writeAll(encodedMessage) | ||||
} else { | ||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Should we explicitly check that gzip type is set instead of this being the else clause? Maybe else should throw unsupported operation exception or something like that. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't think that's needed. The
|
||||
val compressedMessage = Buffer() | ||||
grpcEncoding.toGrpcEncoder().encode(compressedMessage).use { sink -> | ||||
sink.writeAll(encodedMessage) | ||||
} | ||||
sink.writeByte(1) // 1 = Compressed. | ||||
sink.writeInt(compressedMessage.size.toInt()) | ||||
sink.writeAll(compressedMessage) | ||||
} | ||||
|
||||
val compressedFlag = if (grpcEncoding == "identity") 0 else 1 | ||||
sink.writeByte(compressedFlag) | ||||
// TODO: fail if the message size is more than MAX_INT | ||||
sink.writeInt(encodedMessage.size.toInt()) | ||||
sink.writeAll(encodedMessage) | ||||
sink.flush() | ||||
} | ||||
|
||||
|
Original file line number | Diff line number | Diff line change | ||||||
---|---|---|---|---|---|---|---|---|
|
@@ -114,10 +114,15 @@ data class WebConfig @JvmOverloads constructor( | |||||||
val close_connection_percent: Double = 0.0, | ||||||||
|
||||||||
/** | ||||||||
* If true responses which are larger than the minGzipSize will be compressed. | ||||||||
* If true non-gRPC responses which are larger than the minGzipSize will be compressed. | ||||||||
*/ | ||||||||
val gzip: Boolean = true, | ||||||||
|
||||||||
/** | ||||||||
* If true gRPC responses which are larger than the minGzipSize will be compressed. | ||||||||
*/ | ||||||||
val grpcGzip: Boolean = false, | ||||||||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Should this instead be an enum to allow for the other wire-supported encoding?
Suggested change
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes, I'd lean towards the encoding enum, not just a boolean There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Not sure if it's actually worth it:
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I think the current implementation makes more sense to the user:
cc @Hexcles There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Personally, I'd say @zpingcai 's current impl is better in terms of consistency.
|
||||||||
|
||||||||
/** The minimum size in bytes before the response body will be compressed. */ | ||||||||
val minGzipSize: Int = 1024, | ||||||||
|
||||||||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -100,10 +100,14 @@ class ExceptionHandlingInterceptor( | |
httpCall.setResponseTrailer("grpc-status-details-bin", response.toEncodedStatusProto) | ||
httpCall.setResponseTrailer("grpc-message", response.message ?: response.status.name) | ||
httpCall.takeResponseBody()?.use { responseBody: BufferedSink -> | ||
GrpcMessageSink(responseBody, ProtoAdapter.BYTES, grpcEncoding = "identity") | ||
.use { messageSink -> | ||
messageSink.write(ByteString.EMPTY) | ||
} | ||
GrpcMessageSink( | ||
sink = responseBody, | ||
minMessageToCompress = 0, | ||
messageAdapter = ProtoAdapter.BYTES, | ||
grpcEncoding = "identity" | ||
).use { messageSink -> | ||
messageSink.write(ByteString.EMPTY) | ||
} | ||
} | ||
} | ||
|
||
|
@@ -124,7 +128,11 @@ class ExceptionHandlingInterceptor( | |
val mapper = mapperResolver.mapperFor(th) | ||
if (mapper != null) { | ||
if (!suppressLog) { | ||
log.log(mapper.loggingLevel(th), th, *mdcTags.toTypedArray()) { "exception dispatching to $actionName" } | ||
log.log( | ||
level = mapper.loggingLevel(th), | ||
th = th, | ||
tags = *mdcTags.toTypedArray(), | ||
) { "exception dispatching to $actionName" } | ||
Comment on lines
+131
to
+135
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. IDE auto formatted this because the line is too long. |
||
} | ||
return mapper.toResponse(th) | ||
} | ||
|
@@ -139,7 +147,11 @@ class ExceptionHandlingInterceptor( | |
is InvocationTargetException -> toGrpcResponse(th.targetException, mdcTags) | ||
is UncheckedExecutionException -> toGrpcResponse(th.cause!!, mdcTags) | ||
else -> mapperResolver.mapperFor(th)?.let { | ||
log.log(it.loggingLevel(th), th, *mdcTags.toTypedArray()) { "exception dispatching to $actionName" } | ||
log.log( | ||
level = it.loggingLevel(th), | ||
th = th, | ||
tags = *mdcTags.toTypedArray(), | ||
) { "exception dispatching to $actionName" } | ||
val grpcResponse = it.toGrpcResponse(th) | ||
if (grpcResponse == null) { | ||
val httpResponse = toResponse(th, suppressLog = true, mdcTags) | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -71,7 +71,12 @@ class GrpcConnectivityTest { | |
} | ||
|
||
override fun writeTo(sink: BufferedSink) { | ||
val writer = GrpcMessageSink(sink, HelloRequest.ADAPTER, "gzip") | ||
val writer = GrpcMessageSink( | ||
sink = sink, | ||
minMessageToCompress = 0, | ||
messageAdapter = HelloRequest.ADAPTER, | ||
grpcEncoding = "gzip" | ||
) | ||
writer.write(HelloRequest("jesse!")) | ||
} | ||
}) | ||
|
@@ -82,7 +87,7 @@ class GrpcConnectivityTest { | |
response.use { | ||
assertThat(response.code).isEqualTo(200) | ||
assertThat(response.headers["grpc-status"]).isNull() // Sent in the trailers! | ||
assertThat(response.headers["grpc-encoding"]).isEqualTo("identity") | ||
assertThat(response.headers["grpc-encoding"]).isEqualTo("gzip") | ||
assertThat(response.body!!.contentType()).isEqualTo("application/grpc".toMediaType()) | ||
|
||
val reader = GrpcMessageSource( | ||
|
@@ -110,7 +115,12 @@ class GrpcConnectivityTest { | |
} | ||
|
||
override fun writeTo(sink: BufferedSink) { | ||
val writer = GrpcMessageSink(sink, HelloRequest.ADAPTER, "gzip") | ||
val writer = GrpcMessageSink( | ||
sink = sink, | ||
minMessageToCompress = 0, | ||
messageAdapter = HelloRequest.ADAPTER, | ||
grpcEncoding = "gzip" | ||
) | ||
writer.write(HelloRequest("jesse!")) | ||
} | ||
}) | ||
|
@@ -142,7 +152,12 @@ class GrpcConnectivityTest { | |
} | ||
|
||
override fun writeTo(sink: BufferedSink) { | ||
val writer = GrpcMessageSink(sink, HelloRequest.ADAPTER, "gzip") | ||
val writer = GrpcMessageSink( | ||
sink = sink, | ||
minMessageToCompress = 0, | ||
messageAdapter = HelloRequest.ADAPTER, | ||
grpcEncoding = "gzip" | ||
) | ||
writer.write(HelloRequest("jp!")) | ||
} | ||
}) | ||
|
@@ -169,8 +184,8 @@ class GrpcConnectivityTest { | |
interface GreeterSayHello : Service { | ||
@WireRpc( | ||
path = "/helloworld.Greeter/SayHello", | ||
requestAdapter = "com.squareup.protos.test.grpc.HelloRequest.ADAPTER", | ||
responseAdapter = "com.squareup.protos.test.grpc.HelloReply.ADAPTER" | ||
requestAdapter = "com.squareup.protos.test.grpc.HelloRequest#ADAPTER", | ||
responseAdapter = "com.squareup.protos.test.grpc.HelloReply#ADAPTER" | ||
Comment on lines
+187
to
+188
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. drive-by: fix the test setup |
||
) | ||
fun sayHello(request: HelloRequest): HelloReply | ||
} | ||
|
@@ -179,7 +194,7 @@ class GrpcConnectivityTest { | |
override fun configure() { | ||
install( | ||
WebServerTestingModule( | ||
webConfig = WebServerTestingModule.TESTING_WEB_CONFIG | ||
webConfig = WebServerTestingModule.TESTING_WEB_CONFIG.copy(grpcGzip = true) | ||
) | ||
) | ||
install(MiskTestingServiceModule()) | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -29,7 +29,12 @@ class GrpcSourceSinkTest { | |
@Test | ||
fun grpcMessageSinkHelloRequest() { | ||
val buffer = Buffer() | ||
val writer = GrpcMessageSink(buffer, HelloRequest.ADAPTER, "identity") | ||
val writer = GrpcMessageSink( | ||
sink = buffer, | ||
minMessageToCompress = 0, | ||
messageAdapter = HelloRequest.ADAPTER, | ||
grpcEncoding = "identity" | ||
) | ||
writer.write(HelloRequest("localhost")) | ||
writer.close() | ||
|
||
|
@@ -39,10 +44,95 @@ class GrpcSourceSinkTest { | |
@Test | ||
fun grpcMessageSinkHelloReply() { | ||
val buffer = Buffer() | ||
val writer = GrpcMessageSink(buffer, HelloReply.ADAPTER, "identity") | ||
val writer = GrpcMessageSink( | ||
sink = buffer, | ||
minMessageToCompress = 0, | ||
messageAdapter = HelloReply.ADAPTER, | ||
grpcEncoding = "identity" | ||
) | ||
writer.write(HelloReply("Hello localhost")) | ||
writer.close() | ||
|
||
assertEquals("00000000110a0f48656c6c6f206c6f63616c686f7374".decodeHex(), buffer.readByteString()) | ||
} | ||
|
||
@Test | ||
fun grpcMessageSourceCompressedHelloRequest() { | ||
val buffer = Buffer() | ||
buffer.write( | ||
"010000001f1f8b0800000000000000e3e2ccc94f4eccc9c82f2e01002fdef60d0b000000".decodeHex() | ||
) | ||
val reader = GrpcMessageSource(buffer, HelloRequest.ADAPTER, "gzip") | ||
|
||
assertEquals(HelloRequest("localhost"), reader.read()) | ||
} | ||
|
||
@Test | ||
fun messageLargerThanMinimumSizeIsCompressed() { | ||
val message = HelloRequest("localhost") | ||
val encodedMessage = HelloRequest.ADAPTER.encode(message) | ||
assertEquals(encodedMessage.size, 11) | ||
|
||
val buffer = Buffer() | ||
val writer = GrpcMessageSink( | ||
sink = buffer, | ||
minMessageToCompress = 10, | ||
messageAdapter = HelloRequest.ADAPTER, | ||
grpcEncoding = "gzip" | ||
) | ||
|
||
writer.write(message) | ||
writer.close() | ||
|
||
assertEquals( | ||
"010000001f1f8b0800000000000000e3e2ccc94f4eccc9c82f2e01002fdef60d0b000000".decodeHex(), | ||
buffer.readByteString() | ||
) | ||
} | ||
|
||
@Test | ||
fun messageEqualToMinimumSizeIsCompressed() { | ||
val message = HelloRequest("localhost") | ||
val encodedMessage = HelloRequest.ADAPTER.encode(message) | ||
assertEquals(encodedMessage.size, 11) | ||
|
||
val buffer = Buffer() | ||
val writer = GrpcMessageSink( | ||
sink = buffer, | ||
minMessageToCompress = 11, | ||
messageAdapter = HelloRequest.ADAPTER, | ||
grpcEncoding = "gzip" | ||
) | ||
|
||
writer.write(message) | ||
writer.close() | ||
|
||
assertEquals( | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Can you add an assertion on the min size and the actual size? Not just a snapshot assertion on expected output (still keep that but add the extra assertion). There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. The actual size is asserted in There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Added. I simply repeated the same assertion multiple times:
Can definitely create a helper function if needed. |
||
"010000001f1f8b0800000000000000e3e2ccc94f4eccc9c82f2e01002fdef60d0b000000".decodeHex(), | ||
buffer.readByteString() | ||
) | ||
} | ||
|
||
@Test | ||
fun messageSmallerThanMinimumSizeIsNotCompressed() { | ||
zpingcai marked this conversation as resolved.
Show resolved
Hide resolved
|
||
val message = HelloRequest("localhost") | ||
val encodedMessage = HelloRequest.ADAPTER.encode(message) | ||
assertEquals(encodedMessage.size, 11) | ||
|
||
val buffer = Buffer() | ||
val writer = GrpcMessageSink( | ||
sink = buffer, | ||
minMessageToCompress = 12, | ||
messageAdapter = HelloRequest.ADAPTER, | ||
grpcEncoding = "gzip" | ||
) | ||
|
||
writer.write(message) | ||
writer.close() | ||
|
||
assertEquals( | ||
"000000000b0a096c6f63616c686f7374".decodeHex(), | ||
buffer.readByteString() | ||
) | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Simply copies the changes from square/wire@17a3d9e