OnFinishAsyncSequence.swift 1.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657
  1. /*
  2. * Copyright 2024, 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. @available(macOS 10.15, iOS 13, tvOS 13, watchOS 6, *)
  17. struct OnFinishAsyncSequence<Element: Sendable>: AsyncSequence, Sendable {
  18. private let _makeAsyncIterator: @Sendable () -> AsyncIterator
  19. init<S: AsyncSequence>(
  20. wrapping other: S,
  21. onFinish: @escaping () -> Void
  22. ) where S.Element == Element {
  23. self._makeAsyncIterator = {
  24. AsyncIterator(wrapping: other.makeAsyncIterator(), onFinish: onFinish)
  25. }
  26. }
  27. func makeAsyncIterator() -> AsyncIterator {
  28. self._makeAsyncIterator()
  29. }
  30. struct AsyncIterator: AsyncIteratorProtocol {
  31. private var iterator: any AsyncIteratorProtocol
  32. private var onFinish: (() -> Void)?
  33. fileprivate init<Iterator>(
  34. wrapping other: Iterator,
  35. onFinish: @escaping () -> Void
  36. ) where Iterator: AsyncIteratorProtocol, Iterator.Element == Element {
  37. self.iterator = other
  38. self.onFinish = onFinish
  39. }
  40. mutating func next() async throws -> Element? {
  41. let elem = try await self.iterator.next()
  42. if elem == nil {
  43. self.onFinish?()
  44. self.onFinish = nil
  45. }
  46. return elem as? Element
  47. }
  48. }
  49. }