diff --git a/Sources/ComposableArchitecture/Effects/Throttle.swift b/Sources/ComposableArchitecture/Effects/Throttle.swift index d53636628076..e4170fedd4c9 100644 --- a/Sources/ComposableArchitecture/Effects/Throttle.swift +++ b/Sources/ComposableArchitecture/Effects/Throttle.swift @@ -1,8 +1,8 @@ -import Combine +@preconcurrency import Combine import Dispatch import Foundation -extension Effect { +extension Effect where Action: Sendable { /// Throttles an effect so that it only publishes one output per given interval. /// /// The throttling of an effect is with respect to actions being sent into the store. So, if @@ -23,12 +23,13 @@ extension Effect { /// `false`, the publisher emits the first element received during the interval. /// - Returns: An effect that emits either the most-recent or first element received during the /// specified interval. - public func throttle( + public func throttle( id: some Hashable & Sendable, for interval: S.SchedulerTimeType.Stride, scheduler: S, latest: Bool - ) -> Self { + ) -> Self + where S.SchedulerTimeType.Stride: Sendable { switch self.operation { case .none: return .none @@ -42,38 +43,37 @@ extension Effect { publisher .receive(on: scheduler) .flatMap { value -> AnyPublisher in - throttleLock.lock() - defer { throttleLock.unlock() } + throttleState.withValue { + guard let throttleTime = $0.times[id] as! S.SchedulerTimeType? else { + $0.times[id] = scheduler.now + $0.values[id] = nil + return Just(value).eraseToAnyPublisher() + } - guard let throttleTime = throttleTimes[id] as! S.SchedulerTimeType? else { - throttleTimes[id] = scheduler.now - throttleValues[id] = nil - return Just(value).eraseToAnyPublisher() - } - - let value = latest ? value : (throttleValues[id] as! Action? ?? value) - throttleValues[id] = value + let value = latest ? value : ($0.values[id] as! Action? ?? value) + $0.values[id] = value - guard throttleTime.distance(to: scheduler.now) < interval else { - throttleTimes[id] = scheduler.now - throttleValues[id] = nil - return Just(value).eraseToAnyPublisher() - } + guard throttleTime.distance(to: scheduler.now) < interval else { + $0.times[id] = scheduler.now + $0.values[id] = nil + return Just(value).eraseToAnyPublisher() + } - return Just(value) - .delay( - for: scheduler.now.distance(to: throttleTime.advanced(by: interval)), - scheduler: scheduler - ) - .handleEvents( - receiveOutput: { _ in - throttleLock.sync { - throttleTimes[id] = scheduler.now - throttleValues[id] = nil + return Just(value) + .delay( + for: scheduler.now.distance(to: throttleTime.advanced(by: interval)), + scheduler: scheduler + ) + .handleEvents( + receiveOutput: { _ in + throttleState.withValue { + $0.times[id] = scheduler.now + $0.values[id] = nil + } } - } - ) - .eraseToAnyPublisher() + ) + .eraseToAnyPublisher() + } } } .cancellable(id: id, cancelInFlight: true) @@ -81,6 +81,6 @@ extension Effect { } } -var throttleTimes: [AnyHashable: Any] = [:] -var throttleValues: [AnyHashable: Any] = [:] -let throttleLock = NSRecursiveLock() +private let throttleState = LockIsolated<(times: [AnyHashable: Any], values: [AnyHashable: Any])>( + (times: [:], values: [:]) +)