| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879 |
- /*
- * Copyright 2022, gRPC Authors All rights reserved.
- *
- * Licensed under the Apache License, Version 2.0 (the "License");
- * you may not use this file except in compliance with the License.
- * You may obtain a copy of the License at
- *
- * http://www.apache.org/licenses/LICENSE-2.0
- *
- * Unless required by applicable law or agreed to in writing, software
- * distributed under the License is distributed on an "AS IS" BASIS,
- * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
- * See the License for the specific language governing permissions and
- * limitations under the License.
- */
- #if compiler(>=5.6)
- import NIOHPACK
- @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
- extension ServerHandlerStateMachine {
- /// In the 'Draining' state the user handler has been invoked and the request stream has been
- /// closed (i.e. we have seen 'end' but it has not necessarily been consumed by the user handler).
- /// We can transition to a new state either by sending the end of the response stream or by
- /// cancelling.
- internal struct Draining {
- typealias NextStateAndOutput<Output> =
- ServerHandlerStateMachine.NextStateAndOutput<
- ServerHandlerStateMachine.Draining.NextState,
- Output
- >
- /// Whether the response headers have been written yet.
- private var headersWritten: Bool
- internal let context: GRPCAsyncServerCallContext
- init(from state: ServerHandlerStateMachine.Handling) {
- self.headersWritten = state.headersWritten
- self.context = state.context
- }
- mutating func handleMetadata() -> Self.NextStateAndOutput<HandleMetadataAction> {
- // We're already draining, i.e. the inbound stream is closed, cancel the RPC.
- return .init(nextState: .draining(self), output: .cancel)
- }
- mutating func handleMessage() -> Self.NextStateAndOutput<HandleMessageAction> {
- // We're already draining, i.e. the inbound stream is closed, cancel the RPC.
- return .init(nextState: .draining(self), output: .cancel)
- }
- mutating func handleEnd() -> Self.NextStateAndOutput<HandleEndAction> {
- // We're already draining, i.e. the inbound stream is closed, cancel the RPC.
- return .init(nextState: .draining(self), output: .cancel)
- }
- mutating func sendMessage() -> Self.NextStateAndOutput<SendMessageAction> {
- let headers: HPACKHeaders?
- if self.headersWritten {
- headers = nil
- } else {
- self.headersWritten = true
- headers = self.context.initialResponseMetadata
- }
- return .init(nextState: .draining(self), output: .intercept(headers: headers))
- }
- mutating func sendStatus() -> Self.NextStateAndOutput<SendStatusAction> {
- let trailers = self.context.trailingResponseMetadata
- return .init(nextState: .finished(from: self), output: .intercept(trailers: trailers))
- }
- mutating func cancel() -> Self.NextStateAndOutput<CancelAction> {
- return .init(nextState: .finished(from: self), output: .cancelAndNilOutHandlerComponents)
- }
- }
- }
- #endif // compiler(>=5.6)
|