ServerErrorDelegateTests.swift 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168
  1. /*
  2. * Copyright 2020, gRPC Authors All rights reserved.
  3. *
  4. * Licensed under the Apache License, Version 2.0 (the "License");
  5. * you may not use this file except in compliance with the License.
  6. * You may obtain a copy of the License at
  7. *
  8. * http://www.apache.org/licenses/LICENSE-2.0
  9. *
  10. * Unless required by applicable law or agreed to in writing, software
  11. * distributed under the License is distributed on an "AS IS" BASIS,
  12. * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
  13. * See the License for the specific language governing permissions and
  14. * limitations under the License.
  15. */
  16. import EchoImplementation
  17. import EchoModel
  18. import Foundation
  19. @testable import GRPC
  20. import Logging
  21. import NIOCore
  22. import NIOEmbedded
  23. import NIOHPACK
  24. import NIOHTTP2
  25. import XCTest
  26. private class ServerErrorDelegateMock: ServerErrorDelegate {
  27. private let transformLibraryErrorHandler: (Error) -> (GRPCStatusAndTrailers?)
  28. init(transformLibraryErrorHandler: @escaping ((Error) -> (GRPCStatusAndTrailers?))) {
  29. self.transformLibraryErrorHandler = transformLibraryErrorHandler
  30. }
  31. func transformLibraryError(_ error: Error) -> GRPCStatusAndTrailers? {
  32. return self.transformLibraryErrorHandler(error)
  33. }
  34. }
  35. class ServerErrorDelegateTests: GRPCTestCase {
  36. private var channel: EmbeddedChannel!
  37. private var errorDelegate: ServerErrorDelegate!
  38. override func tearDown() {
  39. XCTAssertNoThrow(try self.channel.finish(acceptAlreadyClosed: true))
  40. super.tearDown()
  41. }
  42. func testTransformLibraryError_whenTransformingErrorToStatus_unary() throws {
  43. try self.testTransformLibraryError_whenTransformingErrorToStatus(uri: "/echo.Echo/Get")
  44. }
  45. func testTransformLibraryError_whenTransformingErrorToStatus_clientStreaming() throws {
  46. try self.testTransformLibraryError_whenTransformingErrorToStatus(uri: "/echo.Echo/Collect")
  47. }
  48. func testTransformLibraryError_whenTransformingErrorToStatus_serverStreaming() throws {
  49. try self.testTransformLibraryError_whenTransformingErrorToStatus(uri: "/echo.Echo/Expand")
  50. }
  51. func testTransformLibraryError_whenTransformingErrorToStatus_bidirectionalStreaming() throws {
  52. try self.testTransformLibraryError_whenTransformingErrorToStatus(uri: "/echo.Echo/Update")
  53. }
  54. private func testTransformLibraryError_whenTransformingErrorToStatus(uri: String) throws {
  55. self.setupChannelAndDelegate { _ in
  56. GRPCStatusAndTrailers(status: .init(code: .notFound, message: "some error"))
  57. }
  58. let requestHeaders: HPACKHeaders = [
  59. ":method": "POST",
  60. ":path": uri,
  61. "content-type": "application/grpc",
  62. ]
  63. let headersPayload: HTTP2Frame.FramePayload = .headers(.init(headers: requestHeaders))
  64. XCTAssertNoThrow(try self.channel.writeInbound(headersPayload))
  65. self.channel.pipeline.fireErrorCaught(GRPCStatus(code: .aborted, message: nil))
  66. // Read out the response headers.
  67. XCTAssertNoThrow(try self.channel.readOutbound(as: HTTP2Frame.FramePayload.self))
  68. let end = try self.channel.readOutbound(as: HTTP2Frame.FramePayload.self)
  69. guard case let .some(.headers(trailers)) = end else {
  70. XCTFail("Expected headers but got \(end.debugDescription)")
  71. return
  72. }
  73. XCTAssertEqual(trailers.headers.first(name: "grpc-status"), "5")
  74. XCTAssertEqual(trailers.headers.first(name: "grpc-message"), "some error")
  75. XCTAssertTrue(trailers.endStream)
  76. }
  77. func testTransformLibraryError_whenTransformingErrorToStatusAndMetadata_unary() throws {
  78. try self
  79. .testTransformLibraryError_whenTransformingErrorToStatusAndMetadata(uri: "/echo.Echo/Get")
  80. }
  81. func testTransformLibraryError_whenTransformingErrorToStatusAndMetadata_clientStreaming() throws {
  82. try self
  83. .testTransformLibraryError_whenTransformingErrorToStatusAndMetadata(uri: "/echo.Echo/Collect")
  84. }
  85. func testTransformLibraryError_whenTransformingErrorToStatusAndMetadata_serverStreaming() throws {
  86. try self
  87. .testTransformLibraryError_whenTransformingErrorToStatusAndMetadata(uri: "/echo.Echo/Expand")
  88. }
  89. func testTransformLibraryError_whenTransformingErrorToStatusAndMetadata_bidirectionalStreaming(
  90. ) throws {
  91. try self
  92. .testTransformLibraryError_whenTransformingErrorToStatusAndMetadata(uri: "/echo.Echo/Update")
  93. }
  94. private func testTransformLibraryError_whenTransformingErrorToStatusAndMetadata(
  95. uri: String,
  96. line: UInt = #line
  97. ) throws {
  98. self.setupChannelAndDelegate { _ in
  99. GRPCStatusAndTrailers(
  100. status: .init(code: .notFound, message: "some error"),
  101. trailers: ["some-metadata": "test"]
  102. )
  103. }
  104. let requestHeaders: HPACKHeaders = [
  105. ":method": "POST",
  106. ":path": uri,
  107. "content-type": "application/grpc",
  108. ]
  109. let headersPayload: HTTP2Frame.FramePayload = .headers(.init(headers: requestHeaders))
  110. XCTAssertNoThrow(try self.channel.writeInbound(headersPayload))
  111. self.channel.pipeline.fireErrorCaught(GRPCStatus(code: .aborted, message: nil))
  112. // Read out the response headers.
  113. XCTAssertNoThrow(try self.channel.readOutbound(as: HTTP2Frame.FramePayload.self))
  114. let end = try self.channel.readOutbound(as: HTTP2Frame.FramePayload.self)
  115. guard case let .some(.headers(trailers)) = end else {
  116. XCTFail("Expected headers but got \(end.debugDescription)")
  117. return
  118. }
  119. XCTAssertEqual(trailers.headers.first(name: "grpc-status"), "5", line: line)
  120. XCTAssertEqual(trailers.headers.first(name: "grpc-message"), "some error", line: line)
  121. XCTAssertEqual(trailers.headers.first(name: "some-metadata"), "test", line: line)
  122. XCTAssertTrue(trailers.endStream)
  123. }
  124. private func setupChannelAndDelegate(
  125. transformLibraryErrorHandler: @escaping (Error) -> GRPCStatusAndTrailers?
  126. ) {
  127. let provider = EchoProvider()
  128. self.errorDelegate = ServerErrorDelegateMock(
  129. transformLibraryErrorHandler: transformLibraryErrorHandler
  130. )
  131. let handler = HTTP2ToRawGRPCServerCodec(
  132. servicesByName: [provider.serviceName: provider],
  133. encoding: .disabled,
  134. errorDelegate: self.errorDelegate,
  135. normalizeHeaders: true,
  136. maximumReceiveMessageLength: .max,
  137. logger: self.logger
  138. )
  139. self.channel = EmbeddedChannel(handler: handler)
  140. }
  141. }