Skip to content

Commit

Permalink
netty (fix): Fixes #3313 'Implementation is missing' error
Browse files Browse the repository at this point in the history
  • Loading branch information
xerial committed Jan 29, 2024
1 parent 7d1ded4 commit 5fb6501
Show file tree
Hide file tree
Showing 3 changed files with 101 additions and 48 deletions.
Original file line number Diff line number Diff line change
Expand Up @@ -49,51 +49,58 @@ class NettyRequestHandler(config: NettyServerConfig, dispatcher: NettyBackend.Fi
}

override def channelRead0(ctx: ChannelHandlerContext, msg: FullHttpRequest): Unit = {
var req: wvlet.airframe.http.HttpMessage.Request = msg.method().name() match {
case HttpMethod.GET => Http.GET(msg.uri())
case HttpMethod.POST => Http.POST(msg.uri())
case HttpMethod.PUT => Http.PUT(msg.uri())
case HttpMethod.DELETE => Http.DELETE(msg.uri())
case HttpMethod.PATCH => Http.PATCH(msg.uri())
case HttpMethod.TRACE => Http.request(wvlet.airframe.http.HttpMethod.TRACE, msg.uri())
case HttpMethod.OPTIONS => Http.request(wvlet.airframe.http.HttpMethod.OPTIONS, msg.uri())
case _ => ???
}

ctx.channel().remoteAddress() match {
case x: InetSocketAddress =>
// TODO This address might be IPv6
req = req.withRemoteAddress(ServerAddress(s"${x.getHostString}:${x.getPort}"))
case _ =>
}
try {
var req: wvlet.airframe.http.HttpMessage.Request = msg.method().name().toUpperCase match {
case HttpMethod.GET => Http.GET(msg.uri())
case HttpMethod.POST => Http.POST(msg.uri())
case HttpMethod.PUT => Http.PUT(msg.uri())
case HttpMethod.DELETE => Http.DELETE(msg.uri())
case HttpMethod.PATCH => Http.PATCH(msg.uri())
case HttpMethod.TRACE => Http.request(wvlet.airframe.http.HttpMethod.TRACE, msg.uri())
case HttpMethod.OPTIONS => Http.request(wvlet.airframe.http.HttpMethod.OPTIONS, msg.uri())
case HttpMethod.HEAD => Http.request(wvlet.airframe.http.HttpMethod.HEAD, msg.uri())
case _ =>
throw RPCStatus.INVALID_REQUEST_U1.newException(s"Unsupported HTTP method: ${msg.method()}")
}

msg.headers().names().asScala.map { x =>
req = req.withHeader(x, msg.headers().get(x))
}
val requestBody = msg.content()
val requestBodySize = requestBody.readableBytes()
if (requestBodySize > 0) {
val buf = new Array[Byte](requestBodySize)
requestBody.getBytes(requestBody.readerIndex(), buf)
req = req.withContent(buf)
}
ctx.channel().remoteAddress() match {
case x: InetSocketAddress =>
// TODO This address might be IPv6
req = req.withRemoteAddress(ServerAddress(s"${x.getHostString}:${x.getPort}"))
case _ =>
}

val rxResponse: Rx[Response] = dispatcher.apply(
req,
NettyBackend.newContext { (request: Request) =>
Rx.single(Http.response(HttpStatus.NotFound_404))
msg.headers().names().asScala.map { x =>
req = req.withHeader(x, msg.headers().get(x))
}
val requestBody = msg.content()
val requestBodySize = requestBody.readableBytes()
if (requestBodySize > 0) {
val buf = new Array[Byte](requestBodySize)
requestBody.getBytes(requestBody.readerIndex(), buf)
req = req.withContent(buf)
}
)

RxRunner.run(rxResponse) {
case OnNext(v) =>
val nettyResponse = toNettyResponse(v.asInstanceOf[Response])
writeResponse(msg, ctx, nettyResponse)
case OnError(ex) =>
val resp = RPCStatus.INTERNAL_ERROR_I0.newException(ex.getMessage, ex).toResponse
val nettyResponse = toNettyResponse(resp)
writeResponse(msg, ctx, nettyResponse)
case OnCompletion =>
val rxResponse: Rx[Response] = dispatcher.apply(
req,
NettyBackend.newContext { (request: Request) =>
Rx.single(Http.response(HttpStatus.NotFound_404))
}
)

RxRunner.run(rxResponse) {
case OnNext(v) =>
val nettyResponse = toNettyResponse(v.asInstanceOf[Response])
writeResponse(msg, ctx, nettyResponse)
case OnError(ex) =>
val resp = RPCStatus.INTERNAL_ERROR_I0.newException(ex.getMessage, ex).toResponse
val nettyResponse = toNettyResponse(resp)
writeResponse(msg, ctx, nettyResponse)
case OnCompletion =>
}
} catch {
case e: RPCException =>
writeResponse(msg, ctx, toNettyResponse(e.toResponse))
}
}

Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -13,30 +13,75 @@
*/
package wvlet.airframe.http.netty

import wvlet.airframe.http.HttpServer
import wvlet.airframe.control.Control
import wvlet.airframe.control.Control.withResource
import wvlet.airframe.http.client.SyncClient
import wvlet.airframe.http.{Http, HttpMethod, HttpServer, HttpStatus}
import wvlet.airspec.AirSpec

class NettyServerTest extends AirSpec {

override def design = {
Netty.server.designWithSyncClient
Netty.server.design
.bind[SyncClient].toProvider { (server: NettyServer) =>
Http.client.withRetryContext(_.noRetry).newSyncClient(server.localAddress)
}
}

test("NettyServer should be available") { (server: NettyServer) =>
test("double start should be ignored") {
server.start
}
}

test("can't start server after closing it") {
server.close()
intercept[IllegalStateException] {
server.start
}
test("can't start server after closing it") { (server: NettyServer) =>
server.close()
intercept[IllegalStateException] {
server.start
}
}

test("safely close multiple times") { (server: HttpServer) =>
server.close()
server.close()
}

test("Handle various http methods") { (client: SyncClient) =>
test("valid methods") {
for (
m <- Seq(
HttpMethod.GET,
HttpMethod.POST,
HttpMethod.PUT,
HttpMethod.DELETE,
HttpMethod.PATCH,
HttpMethod.TRACE,
HttpMethod.OPTIONS,
HttpMethod.HEAD
)
) {
test(s"${m}") {
val resp = client.sendSafe(Http.request(m, "/get"))
resp.status shouldBe HttpStatus.NotFound_404

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.GET

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.POST

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.PUT

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.DELETE

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.PATCH

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.GET

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.POST

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.PUT

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.DELETE

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.PATCH

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.GET

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.POST

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.PUT

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.DELETE

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 65 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.PATCH

[500: Internal Server Error] didn't match with [404: Not Found]
}
test(s"${m.toLowerCase} (lower case)") {
val resp = client.sendSafe(Http.request(m, "/get"))
resp.status shouldBe HttpStatus.NotFound_404

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.get (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.post (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.put (lower case)

[503: Service Unavailable] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.delete (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 2.12

NettyServerTest.patch (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.get (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.post (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.put (lower case)

[503: Service Unavailable] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.delete (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3

NettyServerTest.patch (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.get (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.post (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.put (lower case)

[503: Service Unavailable] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.delete (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]

Check failure on line 69 in airframe-http-netty/src/test/scala/wvlet/airframe/http/netty/NettyServerTest.scala

View workflow job for this annotation

GitHub Actions / Scala 3 + JDK21

NettyServerTest.patch (lower case)

[500: Internal Server Error] didn't match with [404: Not Found]
}
}
}

test("reject unsupported methods") {
for (m <- Seq("UNKNOWN_METHOD", HttpMethod.CONNECT)) {
test(m) {
if (m == HttpMethod.CONNECT) {
pending("Not sure how to support CONNECT in Netty")
}
val resp = client.sendSafe(Http.request(m, "/get"))
resp.status shouldBe HttpStatus.BadRequest_400
}
}
}
}

}
Original file line number Diff line number Diff line change
Expand Up @@ -24,4 +24,5 @@ object HttpMethod {
final val OPTIONS = "OPTIONS"
final val TRACE = "TRACE"
final val HEAD = "HEAD"
final val CONNECT = "CONNECT"
}

0 comments on commit 5fb6501

Please sign in to comment.