ServerErrorDelegateTests.swift 6.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170
  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. import Logging
  20. import NIOCore
  21. import NIOEmbedded
  22. import NIOHPACK
  23. import NIOHTTP2
  24. import XCTest
  25. @testable import GRPC
  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. {
  92. try self
  93. .testTransformLibraryError_whenTransformingErrorToStatusAndMetadata(uri: "/echo.Echo/Update")
  94. }
  95. private func testTransformLibraryError_whenTransformingErrorToStatusAndMetadata(
  96. uri: String,
  97. line: UInt = #line
  98. ) throws {
  99. self.setupChannelAndDelegate { _ in
  100. GRPCStatusAndTrailers(
  101. status: .init(code: .notFound, message: "some error"),
  102. trailers: ["some-metadata": "test"]
  103. )
  104. }
  105. let requestHeaders: HPACKHeaders = [
  106. ":method": "POST",
  107. ":path": uri,
  108. "content-type": "application/grpc",
  109. ]
  110. let headersPayload: HTTP2Frame.FramePayload = .headers(.init(headers: requestHeaders))
  111. XCTAssertNoThrow(try self.channel.writeInbound(headersPayload))
  112. self.channel.pipeline.fireErrorCaught(GRPCStatus(code: .aborted, message: nil))
  113. // Read out the response headers.
  114. XCTAssertNoThrow(try self.channel.readOutbound(as: HTTP2Frame.FramePayload.self))
  115. let end = try self.channel.readOutbound(as: HTTP2Frame.FramePayload.self)
  116. guard case let .some(.headers(trailers)) = end else {
  117. XCTFail("Expected headers but got \(end.debugDescription)")
  118. return
  119. }
  120. XCTAssertEqual(trailers.headers.first(name: "grpc-status"), "5", line: line)
  121. XCTAssertEqual(trailers.headers.first(name: "grpc-message"), "some error", line: line)
  122. XCTAssertEqual(trailers.headers.first(name: "some-metadata"), "test", line: line)
  123. XCTAssertTrue(trailers.endStream)
  124. }
  125. private func setupChannelAndDelegate(
  126. transformLibraryErrorHandler: @escaping (Error) -> GRPCStatusAndTrailers?
  127. ) {
  128. let provider = EchoProvider()
  129. self.errorDelegate = ServerErrorDelegateMock(
  130. transformLibraryErrorHandler: transformLibraryErrorHandler
  131. )
  132. let handler = HTTP2ToRawGRPCServerCodec(
  133. servicesByName: [provider.serviceName: provider],
  134. encoding: .disabled,
  135. errorDelegate: self.errorDelegate,
  136. normalizeHeaders: true,
  137. maximumReceiveMessageLength: .max,
  138. logger: self.logger
  139. )
  140. self.channel = EmbeddedChannel(handler: handler)
  141. }
  142. }