Skip to content

Commit bfa27cf

Browse files
committed
Upgrade to Wire 3.0.0-rc01
This rearranges several gRPC classes.
1 parent 7c0d985 commit bfa27cf

22 files changed

+258
-320
lines changed

dependencies.gradle

+3-3
Original file line numberDiff line numberDiff line change
@@ -84,9 +84,9 @@ ext.dep = [
8484
"tracingJaeger": "com.uber.jaeger:jaeger-core:0.24.0",
8585
"tracingZipkin": "io.opentracing.brave:brave-opentracing:0.29.0",
8686
"vitess": "io.vitess:vitess-jdbc:3.0.0",
87-
"wireGrpcClient": "com.squareup.wire:wire-grpc-client:3.0.0-alpha01",
88-
"wirePlugin": "com.squareup.wire:wire-gradle-plugin:3.0.0-alpha01",
89-
"wireRuntime": "com.squareup.wire:wire-runtime:3.0.0-alpha01",
87+
"wireRuntime": "com.squareup.wire:wire-runtime:3.0.0-rc01",
88+
"wireGrpcClient": "com.squareup.wire:wire-grpc-client:3.0.0-rc01",
89+
"wirePlugin": "com.squareup.wire:wire-gradle-plugin:3.0.0-rc01",
9090
"zipkinBrave": "io.zipkin.brave:brave:4.17.2",
9191
"zipkinReporter": "io.zipkin.reporter2:zipkin-sender-okhttp3:2.4.1",
9292
"zookeeper": "org.apache.zookeeper:zookeeper:3.5.4-beta",

misk-grpc-tests/build.gradle

+9
Original file line numberDiff line numberDiff line change
@@ -32,6 +32,14 @@ protobuf {
3232
wire {
3333
kotlin {
3434
}
35+
36+
// Generate service interfaces also.
37+
kotlin {
38+
includes = ['routeguide.RouteGuide']
39+
exclusive = false
40+
blockingServices = true
41+
singleMethodServices = true
42+
}
3543
}
3644

3745
sourceSets {
@@ -49,6 +57,7 @@ dependencies {
4957
compile dep.grpcProtobuf
5058
compile dep.grpcStub
5159
compile dep.wireGrpcClient
60+
compile dep.wireRuntime
5261
compile project(':misk')
5362
compile project(':misk-testing')
5463
}

misk-grpc-tests/src/main/java/routeguide/RouteGuide.kt

-41
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -1,16 +1,15 @@
11
package misk.grpc.miskserver
22

3-
import misk.web.Grpc
43
import misk.web.actions.WebAction
54
import misk.web.interceptors.LogRequestResponse
65
import routeguide.Feature
76
import routeguide.Point
7+
import routeguide.RouteGuideGetFeature
88
import javax.inject.Inject
99

10-
class GetFeatureGrpcAction @Inject constructor() : WebAction {
11-
@Grpc("/routeguide.RouteGuide/GetFeature")
10+
class GetFeatureGrpcAction @Inject constructor() : WebAction, RouteGuideGetFeature {
1211
@LogRequestResponse(sampling = 1.0, includeBody = true)
13-
fun sayHello(point: Point): Feature {
14-
return Feature(name = "maple tree", location = point)
12+
override fun GetFeature(request: Point): Feature {
13+
return Feature(name = "maple tree", location = request)
1514
}
1615
}

misk-grpc-tests/src/main/kotlin/misk/grpc/miskserver/RouteChatGrpcAction.kt

+8-9
Original file line numberDiff line numberDiff line change
@@ -1,24 +1,23 @@
11
package misk.grpc.miskserver
22

3-
import misk.grpc.GrpcReceiveChannel
4-
import misk.grpc.GrpcSendChannel
3+
import com.squareup.wire.MessageSink
4+
import com.squareup.wire.MessageSource
55
import misk.grpc.consumeEachAndClose
6-
import misk.web.Grpc
76
import misk.web.actions.WebAction
87
import misk.web.interceptors.LogRequestResponse
8+
import routeguide.RouteGuideRouteChat
99
import routeguide.RouteNote
1010
import javax.inject.Inject
1111

12-
class RouteChatGrpcAction @Inject constructor() : WebAction {
13-
@Grpc("/routeguide.RouteGuide/RouteChat")
12+
class RouteChatGrpcAction @Inject constructor() : WebAction, RouteGuideRouteChat {
1413
@LogRequestResponse(sampling = 1.0, includeBody = true)
15-
fun chat(
16-
request: GrpcReceiveChannel<RouteNote>,
17-
response: GrpcSendChannel<RouteNote>
14+
override fun RouteChat(
15+
request: MessageSource<RouteNote>,
16+
response: MessageSink<RouteNote>
1817
) {
1918
response.use {
2019
request.consumeEachAndClose { routeNote ->
21-
response.send(RouteNote(message = "ACK: ${routeNote.message}"))
20+
response.write(RouteNote(message = "ACK: ${routeNote.message}"))
2221
}
2322
}
2423
}

misk-grpc-tests/src/test/kotlin/misk/grpc/MiskClientMiskServerTest.kt

+1-1
Original file line numberDiff line numberDiff line change
@@ -69,7 +69,7 @@ class MiskClientMiskServerTest {
6969

7070
// Confirm interceptors were invoked.
7171
assertThat(logCollector.takeMessage(RequestLoggingInterceptor::class)).isEqualTo(
72-
"RouteChatGrpcAction principal=unknown request=[GrpcReceiveChannel, GrpcSendChannel]")
72+
"RouteChatGrpcAction principal=unknown request=[GrpcMessageSource, GrpcMessageSink]")
7373
assertThat(logCollector.takeMessage(RequestLoggingInterceptor::class)).isEqualTo(
7474
"RouteChatGrpcAction principal=unknown time=0.000 ns response=kotlin.Unit")
7575
}
+22-5
Original file line numberDiff line numberDiff line change
@@ -1,17 +1,34 @@
11
package misk.grpc
22

3+
import com.squareup.wire.MessageSink
4+
import com.squareup.wire.MessageSource
35
import java.lang.reflect.ParameterizedType
46
import java.lang.reflect.Type
7+
import java.lang.reflect.WildcardType
58
import kotlin.reflect.KType
69
import kotlin.reflect.jvm.javaType
710

811
/**
9-
* Returns the channel element type, like `MyRequest` if this is `Channel<MyRequest>`. Returns null
10-
* if this is not a channel.
12+
* Returns the stream element type, like `MyRequest` if this is `MessageSource<MyRequest>`.
13+
* Returns null if this is not a [MessageSource] or [MessageSink].
1114
*/
1215
internal fun KType.streamElementType(): Type? {
16+
// Unbox the type parameter.
1317
val parameterizedType = javaType as? ParameterizedType ?: return null
14-
if (parameterizedType.rawType != GrpcReceiveChannel::class.java &&
15-
parameterizedType.rawType != GrpcSendChannel::class.java) return null
16-
return parameterizedType.actualTypeArguments[0]
18+
if (parameterizedType.rawType != MessageSource::class.java &&
19+
parameterizedType.rawType != MessageSink::class.java) return null
20+
// Remove the wildcard, like 'out MessageSource' (Kotlin) or '? super MessageSource' (Java).
21+
return when (val typeArgument = parameterizedType.actualTypeArguments[0]) {
22+
is WildcardType -> typeArgument.lowerBounds[0]
23+
else -> typeArgument //
24+
}
25+
}
26+
27+
fun <T : Any> MessageSource<T>.consumeEachAndClose(block: (T) -> Unit) {
28+
use {
29+
while (true) {
30+
val message = read() ?: return
31+
block(message)
32+
}
33+
}
1734
}

misk/src/main/kotlin/misk/grpc/GrpcChannel.kt

-20
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
package misk.grpc
2+
3+
import okio.BufferedSource
4+
import okio.GzipSource
5+
import okio.Source
6+
import java.net.ProtocolException
7+
8+
/**
9+
* This is derived from Wire's GrpcDecoder.kt.
10+
* https://github.com/square/wire/blob/master/wire-grpc-client/src/main/java/com/squareup/wire/GrpcDecoder.kt
11+
*/
12+
internal sealed class GrpcDecoder(val name: String) {
13+
/** Returns a stream that decodes `source`. */
14+
abstract fun decode(source: BufferedSource): Source
15+
16+
internal object IdentityGrpcDecoder : GrpcDecoder("identity") {
17+
override fun decode(source: BufferedSource) = source
18+
}
19+
20+
internal object GzipGrpcDecoder : GrpcDecoder("gzip") {
21+
override fun decode(source: BufferedSource) = GzipSource(source)
22+
}
23+
}
24+
25+
internal fun String.toGrpcDecoding(): GrpcDecoder {
26+
return when (this) {
27+
"identity" -> GrpcDecoder.IdentityGrpcDecoder
28+
"gzip" -> GrpcDecoder.GzipGrpcDecoder
29+
"deflate" -> throw ProtocolException("deflate not yet supported")
30+
"snappy" -> throw ProtocolException("snappy not yet supported")
31+
else -> throw ProtocolException("unsupported grpc-encoding: $this")
32+
}
33+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,33 @@
1+
package misk.grpc
2+
3+
import okio.BufferedSink
4+
import okio.buffer
5+
import okio.gzip
6+
import java.net.ProtocolException
7+
8+
/**
9+
* This is derived from Wire's GrpcEncoder.kt.
10+
* https://github.com/square/wire/blob/master/wire-grpc-client/src/main/java/com/squareup/wire/GrpcEncoder.kt
11+
*/
12+
internal sealed class GrpcEncoder(val name: String) {
13+
/** Returns a stream that decodes `source`. */
14+
abstract fun encode(sink: BufferedSink): BufferedSink
15+
16+
internal object IdentityGrpcEncoder : GrpcEncoder("identity") {
17+
override fun encode(sink: BufferedSink) = sink
18+
}
19+
20+
internal object GzipGrpcEncoder : GrpcEncoder("gzip") {
21+
override fun encode(sink: BufferedSink) = sink.gzip().buffer()
22+
}
23+
}
24+
25+
internal fun String.toGrpcEncoder(): GrpcEncoder {
26+
return when (this) {
27+
"identity" -> GrpcEncoder.IdentityGrpcEncoder
28+
"gzip" -> GrpcEncoder.GzipGrpcEncoder
29+
"deflate" -> throw ProtocolException("deflate not yet supported")
30+
"snappy" -> throw ProtocolException("snappy not yet supported")
31+
else -> throw ProtocolException("unsupported grpc-encoding: $this")
32+
}
33+
}

misk/src/main/kotlin/misk/grpc/GrpcEncoding.kt

-29
This file was deleted.
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,42 @@
1+
package misk.grpc
2+
3+
import com.squareup.wire.MessageSink
4+
import com.squareup.wire.ProtoAdapter
5+
import okio.Buffer
6+
import okio.BufferedSink
7+
import java.io.Closeable
8+
9+
/**
10+
* Writes a sequence of gRPC messages as an HTTP/2 stream.
11+
*
12+
* This is derived from Wire's GrpcMessageSink.kt.
13+
* https://github.com/square/wire/blob/master/wire-grpc-client/src/main/java/com/squareup/wire/GrpcMessageSink.kt
14+
*
15+
* @param sink the HTTP/2 stream body.
16+
* @param messageAdapter a proto adapter for each message.
17+
* @param grpcEncoding the content coding for the stream body.
18+
*/
19+
internal class GrpcMessageSink<T : Any> constructor(
20+
private val sink: BufferedSink,
21+
private val messageAdapter: ProtoAdapter<T>,
22+
private val grpcEncoding: String = "identity"
23+
) : MessageSink<T>, Closeable by sink {
24+
override fun write(message: T) {
25+
val messageEncoding = grpcEncoding.toGrpcEncoder()
26+
val encodingSink = messageEncoding.encode(sink)
27+
28+
val compressedFlag = if (grpcEncoding == "identity") 0 else 1
29+
encodingSink.writeByte(compressedFlag)
30+
31+
val encodedMessage = Buffer()
32+
messageAdapter.encode(encodedMessage, message)
33+
34+
// TODO: fail if the message size is more than MAX_INT
35+
encodingSink.writeInt(encodedMessage.size.toInt())
36+
encodingSink.writeAll(encodedMessage)
37+
38+
sink.flush()
39+
}
40+
41+
override fun toString() = "GrpcMessageSink"
42+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,53 @@
1+
package misk.grpc
2+
3+
import com.squareup.wire.MessageSource
4+
import com.squareup.wire.ProtoAdapter
5+
import okio.Buffer
6+
import okio.BufferedSource
7+
import okio.buffer
8+
import java.io.Closeable
9+
import java.net.ProtocolException
10+
11+
/**
12+
* Reads an HTTP/2 stream as a sequence of gRPC messages.
13+
*
14+
* This is derived from Wire's GrpcMessageSource.kt.
15+
* https://github.com/square/wire/blob/master/wire-grpc-client/src/main/java/com/squareup/wire/GrpcMessageSource.kt
16+
*
17+
* @param source the HTTP/2 stream body.
18+
* @param messageAdapter a proto adapter for each message.
19+
* @param grpcEncoding the "grpc-encoding" header, or null if it is absent.
20+
*/
21+
internal class GrpcMessageSource<T : Any>(
22+
private val source: BufferedSource,
23+
private val messageAdapter: ProtoAdapter<T>,
24+
private val grpcEncoding: String? = null
25+
) : MessageSource<T>, Closeable by source {
26+
override fun read(): T? {
27+
if (source.exhausted()) return null
28+
29+
// Length-Prefixed-Message → Compressed-Flag Message-Length Message
30+
// Compressed-Flag → 0 / 1 # encoded as 1 byte unsigned integer
31+
// Message-Length → {length of Message} # encoded as 4 byte unsigned integer
32+
// Message → *{binary octet}
33+
34+
val compressedFlag = source.readByte()
35+
val messageDecoding: GrpcDecoder = when {
36+
compressedFlag.toInt() == 0 -> GrpcDecoder.IdentityGrpcDecoder
37+
compressedFlag.toInt() == 1 -> {
38+
grpcEncoding?.toGrpcDecoding() ?: throw ProtocolException(
39+
"message is encoded but message-encoding header was omitted")
40+
}
41+
else -> throw ProtocolException("unexpected compressed-flag: $compressedFlag")
42+
}
43+
44+
val encodedLength = source.readInt().toLong() and 0xffffffffL
45+
46+
val encodedMessage = Buffer()
47+
encodedMessage.write(source, encodedLength)
48+
49+
return messageAdapter.decode(messageDecoding.decode(encodedMessage).buffer())
50+
}
51+
52+
override fun toString() = "GrpcMessageSource"
53+
}

0 commit comments

Comments
 (0)