diff --git a/PostHog.xcodeproj/project.pbxproj b/PostHog.xcodeproj/project.pbxproj index 6d1a5da44..64149ce96 100644 --- a/PostHog.xcodeproj/project.pbxproj +++ b/PostHog.xcodeproj/project.pbxproj @@ -1252,6 +1252,7 @@ SUPPORTS_MACCATALYST = NO; SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_STRICT_CONCURRENCY = complete; SWIFT_VERSION = 5.0; TARGETED_DEVICE_FAMILY = "1,2"; TVOS_DEPLOYMENT_TARGET = 13.0; @@ -1287,6 +1288,7 @@ SUPPORTS_MACCATALYST = NO; SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = NO; SWIFT_EMIT_LOC_STRINGS = YES; + SWIFT_STRICT_CONCURRENCY = complete; SWIFT_VERSION = 5.0; TARGETED_DEVICE_FAMILY = "1,2"; TVOS_DEPLOYMENT_TARGET = 13.0; @@ -1354,6 +1356,7 @@ SDKROOT = iphoneos; SWIFT_ACTIVE_COMPILATION_CONDITIONS = DEBUG; SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_VERSION = 5.0; TVOS_DEPLOYMENT_TARGET = 13.0; VERSIONING_SYSTEM = "apple-generic"; VERSION_INFO_PREFIX = ""; @@ -1415,6 +1418,7 @@ SDKROOT = iphoneos; SWIFT_COMPILATION_MODE = wholemodule; SWIFT_OPTIMIZATION_LEVEL = "-O"; + SWIFT_VERSION = 5.0; TVOS_DEPLOYMENT_TARGET = 13.0; VALIDATE_PRODUCT = YES; VERSIONING_SYSTEM = "apple-generic"; @@ -1458,6 +1462,7 @@ SWIFT_EMIT_LOC_STRINGS = YES; SWIFT_OBJC_BRIDGING_HEADER = ""; SWIFT_OPTIMIZATION_LEVEL = "-Onone"; + SWIFT_STRICT_CONCURRENCY = complete; SWIFT_VERSION = 5.0; TARGETED_DEVICE_FAMILY = "1,2,3,4"; TVOS_DEPLOYMENT_TARGET = 13.0; @@ -1500,6 +1505,7 @@ SUPPORTS_MAC_DESIGNED_FOR_IPHONE_IPAD = YES; SWIFT_EMIT_LOC_STRINGS = YES; SWIFT_OBJC_BRIDGING_HEADER = ""; + SWIFT_STRICT_CONCURRENCY = complete; SWIFT_VERSION = 5.0; TARGETED_DEVICE_FAMILY = "1,2,3,4"; TVOS_DEPLOYMENT_TARGET = 13.0; diff --git a/PostHog/Models/PostHogEvent.swift b/PostHog/Models/PostHogEvent.swift index 3446cde87..2ee3a5b4d 100644 --- a/PostHog/Models/PostHogEvent.swift +++ b/PostHog/Models/PostHogEvent.swift @@ -7,7 +7,7 @@ import Foundation -public class PostHogEvent { +public class PostHogEvent: @unchecked Sendable { public var event: String public var distinctId: String public var properties: [String: Any] diff --git a/PostHog/PostHogApi.swift b/PostHog/PostHogApi.swift index 79d00c134..e75a47b61 100644 --- a/PostHog/PostHogApi.swift +++ b/PostHog/PostHogApi.swift @@ -7,7 +7,7 @@ import Foundation -class PostHogApi { +class PostHogApi: @unchecked Sendable { private let config: PostHogConfig // default is 60s but we do 10s @@ -35,7 +35,7 @@ class PostHogApi { return request } - func batch(events: [PostHogEvent], completion: @escaping (PostHogBatchUploadInfo) -> Void) { + func batch(events: [PostHogEvent], completion: @escaping @Sendable (PostHogBatchUploadInfo) -> Void) { guard let url = URL(string: "batch", relativeTo: config.host) else { hedgeLog("Malformed batch URL error.") return completion(PostHogBatchUploadInfo(statusCode: nil, error: nil)) @@ -92,7 +92,7 @@ class PostHogApi { }.resume() } - func snapshot(events: [PostHogEvent], completion: @escaping (PostHogBatchUploadInfo) -> Void) { + func snapshot(events: [PostHogEvent], completion: @escaping @Sendable (PostHogBatchUploadInfo) -> Void) { guard let url = URL(string: config.snapshotEndpoint, relativeTo: config.host) else { hedgeLog("Malformed snapshot URL error.") return completion(PostHogBatchUploadInfo(statusCode: nil, error: nil)) @@ -153,7 +153,7 @@ class PostHogApi { distinctId: String, anonymousId: String, groups: [String: String], - completion: @escaping ([String: Any]?, _ error: Error?) -> Void + completion: @escaping @Sendable ([String: Any]?, _ error: Error?) -> Void ) { var urlComps = URLComponents() urlComps.path = "/decide" diff --git a/PostHog/PostHogConsumerPayload.swift b/PostHog/PostHogConsumerPayload.swift index 923752fdf..fd12972f1 100644 --- a/PostHog/PostHogConsumerPayload.swift +++ b/PostHog/PostHogConsumerPayload.swift @@ -9,5 +9,5 @@ import Foundation struct PostHogConsumerPayload { let events: [PostHogEvent] - let completion: (Bool) -> Void + let completion: @Sendable (Bool) -> Void } diff --git a/PostHog/PostHogContext.swift b/PostHog/PostHogContext.swift index 4f0c9149d..73444336d 100644 --- a/PostHog/PostHogContext.swift +++ b/PostHog/PostHogContext.swift @@ -13,12 +13,21 @@ import Foundation import AppKit #endif -class PostHogContext { +class PostHogContext: @unchecked Sendable { #if !os(watchOS) private let reachability: Reachability? #endif - private lazy var theStaticContext: [String: Any] = { + #if !os(watchOS) + init(_ reachability: Reachability?) { + self.reachability = reachability + } + #else + init() {} + #endif + + // UIDevice is marked as @MainActor + @MainActor func staticContext() -> [String: Any] { // Properties that do not change over the lifecycle of an application var properties: [String: Any] = [:] @@ -80,18 +89,6 @@ class PostHogContext { #endif return properties - }() - - #if !os(watchOS) - init(_ reachability: Reachability?) { - self.reachability = reachability - } - #else - init() {} - #endif - - func staticContext() -> [String: Any] { - theStaticContext } private func platform() -> String { @@ -102,7 +99,8 @@ class PostHogContext { return String(cString: machine) } - func dynamicContext() -> [String: Any] { + // UIScreen is marked as @MainActor + @MainActor func dynamicContext() -> [String: Any] { var properties: [String: Any] = [:] #if os(iOS) || os(tvOS) diff --git a/PostHog/PostHogFeatureFlags.swift b/PostHog/PostHogFeatureFlags.swift index bbffb0bb0..66050a392 100644 --- a/PostHog/PostHogFeatureFlags.swift +++ b/PostHog/PostHogFeatureFlags.swift @@ -7,7 +7,7 @@ import Foundation -class PostHogFeatureFlags { +class PostHogFeatureFlags: @unchecked Sendable { private let config: PostHogConfig private let storage: PostHogStorage private let api: PostHogApi @@ -38,7 +38,7 @@ class PostHogFeatureFlags { distinctId: String, anonymousId: String, groups: [String: String], - callback: @escaping () -> Void + callback: @escaping @Sendable () -> Void ) { isLoadingLock.withLock { if self.isLoadingFeatureFlags { @@ -52,8 +52,9 @@ class PostHogFeatureFlags { groups: groups) { data, _ in self.dispatchQueue.async { - guard let featureFlags = data?["featureFlags"] as? [String: Any], - let featureFlagPayloads = data?["featureFlagPayloads"] as? [String: Any] + guard let data = data, + let featureFlags = data["featureFlags"] as? [String: Any], + let featureFlagPayloads = data["featureFlagPayloads"] as? [String: Any] else { hedgeLog("Error: Decide response missing correct featureFlags format") @@ -61,12 +62,12 @@ class PostHogFeatureFlags { return callback() } - let errorsWhileComputingFlags = data?["errorsWhileComputingFlags"] as? Bool ?? false + let errorsWhileComputingFlags = data["errorsWhileComputingFlags"] as? Bool ?? false #if os(iOS) - if let sessionRecording = data?["sessionRecording"] as? Bool { + if let sessionRecording = data["sessionRecording"] as? Bool { self.config.sessionReplay = self.config.sessionReplay && sessionRecording - } else if let sessionRecording = data?["sessionRecording"] as? [String: Any] { + } else if let sessionRecording = data["sessionRecording"] as? [String: Any] { // keeps the value from config.sessionReplay since having sessionRecording // means its enabled on the project settings, but its only enabled // when local config.sessionReplay is also enabled diff --git a/PostHog/PostHogQueue.swift b/PostHog/PostHogQueue.swift index add4b846d..2171fda03 100644 --- a/PostHog/PostHogQueue.swift +++ b/PostHog/PostHogQueue.swift @@ -17,7 +17,7 @@ import Foundation */ -class PostHogQueue { +class PostHogQueue: @unchecked Sendable { enum PostHogApiEndpoint: Int { case batch case snapshot @@ -221,7 +221,7 @@ class PostHogQueue { flushIfOverThreshold() } - private func take(_ count: Int, completion: @escaping (PostHogConsumerPayload) -> Void) { + private func take(_ count: Int, completion: @escaping @Sendable (PostHogConsumerPayload) -> Void) { dispatchQueue.async { self.isFlushingLock.withLock { if self.isFlushing { diff --git a/PostHog/PostHogSDK.swift b/PostHog/PostHogSDK.swift index b8398fab7..fa8a2382e 100644 --- a/PostHog/PostHogSDK.swift +++ b/PostHog/PostHogSDK.swift @@ -19,7 +19,8 @@ let maxRetryDelay = 30.0 private let sessionChangeThreshold: TimeInterval = 60 * 30 // renamed to PostHogSDK due to https://github.com/apple/swift/issues/56573 -@objc public class PostHogSDK: NSObject { +// @unchecked because its operations are manually locked +@objc public class PostHogSDK: NSObject, @unchecked Sendable { private var config: PostHogConfig private init(_ config: PostHogConfig) { @@ -45,7 +46,8 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 private var flagCallReported = Set() private var featureFlags: PostHogFeatureFlags? private var context: PostHogContext? - private static var apiKeys = Set() + // nonisolated because its manually locked with setupLock + private nonisolated(unsafe) static var apiKeys = Set() private var capturedAppInstalled = false private var appFromBackground = false private var sessionId: String? @@ -74,7 +76,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 toggleHedgeLog(enabled) } - @objc public func setup(_ config: PostHogConfig) { + @MainActor @objc public func setup(_ config: PostHogConfig) { setupLock.withLock { toggleHedgeLog(config.debug) if enabled { @@ -219,7 +221,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 return properties } - private func buildProperties(distinctId: String, + @MainActor private func buildProperties(distinctId: String, properties: [String: Any]?, userProperties: [String: Any]? = nil, userPropertiesSetOnce: [String: Any]? = nil, @@ -335,18 +337,18 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 } } - @objc public func identify(_ distinctId: String) { + @MainActor @objc public func identify(_ distinctId: String) { identify(distinctId, userProperties: nil, userPropertiesSetOnce: nil) } - @objc(identifyWithDistinctId:userProperties:) + @MainActor @objc(identifyWithDistinctId:userProperties:) public func identify(_ distinctId: String, userProperties: [String: Any]? = nil) { identify(distinctId, userProperties: userProperties, userPropertiesSetOnce: nil) } - @objc(identifyWithDistinctId:userProperties:userPropertiesSetOnce:) + @MainActor @objc(identifyWithDistinctId:userProperties:userPropertiesSetOnce:) public func identify(_ distinctId: String, userProperties: [String: Any]? = nil, userPropertiesSetOnce: [String: Any]? = nil) @@ -382,18 +384,18 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 } } - @objc public func capture(_ event: String) { + @MainActor @objc public func capture(_ event: String) { capture(event, properties: nil, userProperties: nil, userPropertiesSetOnce: nil, groupProperties: nil) } - @objc(captureWithEvent:properties:) + @MainActor @objc(captureWithEvent:properties:) public func capture(_ event: String, properties: [String: Any]? = nil) { capture(event, properties: properties, userProperties: nil, userPropertiesSetOnce: nil, groupProperties: nil) } - @objc(captureWithEvent:properties:userProperties:) + @MainActor @objc(captureWithEvent:properties:userProperties:) public func capture(_ event: String, properties: [String: Any]? = nil, userProperties: [String: Any]? = nil) @@ -401,7 +403,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 capture(event, properties: properties, userProperties: userProperties, userPropertiesSetOnce: nil, groupProperties: nil) } - @objc(captureWithEvent:properties:userProperties:userPropertiesSetOnce:) + @MainActor @objc(captureWithEvent:properties:userProperties:userPropertiesSetOnce:) public func capture(_ event: String, properties: [String: Any]? = nil, userProperties: [String: Any]? = nil, @@ -418,7 +420,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 return false } - @objc(captureWithEvent:properties:userProperties:userPropertiesSetOnce:groupProperties:) + @MainActor @objc(captureWithEvent:properties:userProperties:userPropertiesSetOnce:groupProperties:) public func capture(_ event: String, properties: [String: Any]? = nil, userProperties: [String: Any]? = nil, @@ -477,11 +479,11 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 queue.add(posthogEvent) } - @objc public func screen(_ screenTitle: String) { + @MainActor @objc public func screen(_ screenTitle: String) { screen(screenTitle, properties: nil) } - @objc(screenWithTitle:properties:) + @MainActor @objc(screenWithTitle:properties:) public func screen(_ screenTitle: String, properties: [String: Any]? = nil) { if !isEnabled() { return @@ -507,7 +509,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 )) } - @objc public func alias(_ alias: String) { + @MainActor @objc public func alias(_ alias: String) { if !isEnabled() { return } @@ -558,7 +560,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 return mergedGroups ?? [:] } - private func groupIdentify(type: String, key: String, groupProperties: [String: Any]? = nil) { + @MainActor private func groupIdentify(type: String, key: String, groupProperties: [String: Any]? = nil) { if !isEnabled() { return } @@ -589,12 +591,12 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 )) } - @objc(groupWithType:key:) + @MainActor @objc(groupWithType:key:) public func group(type: String, key: String) { group(type: type, key: key, groupProperties: nil) } - @objc(groupWithType:key:groupProperties:) + @MainActor @objc(groupWithType:key:groupProperties:) public func group(type: String, key: String, groupProperties: [String: Any]? = nil) { if !isEnabled() { return @@ -617,7 +619,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 } @objc(reloadFeatureFlagsWithCallback:) - public func reloadFeatureFlags(_ callback: @escaping () -> Void) { + public func reloadFeatureFlags(_ callback: @escaping @Sendable () -> Void) { if !isEnabled() { return } @@ -638,7 +640,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 ) } - @objc public func getFeatureFlag(_ key: String) -> Any? { + @MainActor @objc public func getFeatureFlag(_ key: String) -> Any? { if !isEnabled() { return nil } @@ -656,7 +658,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 return value } - @objc public func isFeatureEnabled(_ key: String) -> Bool { + @MainActor @objc public func isFeatureEnabled(_ key: String) -> Bool { if !isEnabled() { return false } @@ -686,7 +688,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 return featureFlags.getFeatureFlagPayload(key) } - private func reportFeatureFlagCalled(flagKey: String, flagValue: Any?) { + @MainActor private func reportFeatureFlagCalled(flagKey: String, flagValue: Any?) { if !flagCallReported.contains(flagKey) { let properties: [String: Any] = [ "$feature_flag": flagKey, @@ -757,7 +759,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 return config.optOut } - @objc public func close() { + @MainActor @objc public func close() { if !isEnabled() { return } @@ -799,13 +801,13 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 } } - @objc public static func with(_ config: PostHogConfig) -> PostHogSDK { + @MainActor @objc public static func with(_ config: PostHogConfig) -> PostHogSDK { let postHog = PostHogSDK(config) postHog.setup(config) return postHog } - private func unregisterNotifications() { + @MainActor private func unregisterNotifications() { let defaultCenter = NotificationCenter.default #if os(iOS) || os(tvOS) @@ -819,7 +821,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 #endif } - private func registerNotifications() { + @MainActor private func registerNotifications() { let defaultCenter = NotificationCenter.default #if os(iOS) || os(tvOS) @@ -852,7 +854,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 #endif } - private func captureScreenViews() { + @MainActor private func captureScreenViews() { if config.captureScreenViews { #if os(iOS) || os(tvOS) UIViewController.swizzleScreenView() @@ -860,11 +862,11 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 } } - @objc func handleAppDidFinishLaunching() { + @MainActor @objc func handleAppDidFinishLaunching() { captureAppInstallLifecycle() } - private func captureAppInstallLifecycle() { + @MainActor private func captureAppInstallLifecycle() { if !config.captureApplicationLifecycleEvents { return } @@ -923,14 +925,14 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 } } - @objc func handleAppDidBecomeActive() { + @MainActor @objc func handleAppDidBecomeActive() { rotateSessionIdIfRequired() isInBackground = false captureAppOpened() } - private func captureAppOpened() { + @MainActor private func captureAppOpened() { if !config.captureApplicationLifecycleEvents { return } @@ -957,7 +959,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 capture("Application Opened", properties: props) } - @objc func handleAppDidEnterBackground() { + @MainActor @objc func handleAppDidEnterBackground() { captureAppBackgrounded() sessionLock.withLock { @@ -967,7 +969,7 @@ private let sessionChangeThreshold: TimeInterval = 60 * 30 isInBackground = true } - private func captureAppBackgrounded() { + @MainActor private func captureAppBackgrounded() { if !config.captureApplicationLifecycleEvents { return } diff --git a/PostHog/PostHogVersion.swift b/PostHog/PostHogVersion.swift index f33266e90..e53996022 100644 --- a/PostHog/PostHogVersion.swift +++ b/PostHog/PostHogVersion.swift @@ -9,7 +9,7 @@ import Foundation // if you change this, make sure to also change it in the podspec and check if the script scripts/bump-version.sh still works // This property is internal only -public var postHogVersion = "3.3.0-alpha.1" +public nonisolated(unsafe) var postHogVersion = "3.3.0-alpha.1" // This property is internal only -public var postHogSdkName = "posthog-ios" +public nonisolated(unsafe) var postHogSdkName = "posthog-ios" diff --git a/PostHog/Replay/PostHogReplayIntegration.swift b/PostHog/Replay/PostHogReplayIntegration.swift index f4793b57b..87e5de467 100644 --- a/PostHog/Replay/PostHogReplayIntegration.swift +++ b/PostHog/Replay/PostHogReplayIntegration.swift @@ -63,7 +63,7 @@ timer = nil } - private func generateSnapshot(_ view: UIView, _ screenName: String? = nil) { + @MainActor private func generateSnapshot(_ view: UIView, _ screenName: String? = nil) { var hasChanges = false let timestamp = Date().toMillis() @@ -127,7 +127,7 @@ style.paddingLeft = Int(insets.left) } - private func toWireframe(_ view: UIView, parentId: Int? = nil) -> RRWireframe? { + @MainActor private func toWireframe(_ view: UIView, parentId: Int? = nil) -> RRWireframe? { if !view.isVisible() { return nil } @@ -262,7 +262,7 @@ return wireframe } - static func getCurrentWindow() -> UIWindow? { + @MainActor static func getCurrentWindow() -> UIWindow? { guard let activeScene = UIApplication.shared.connectedScenes.first(where: { $0.activationState == .foregroundActive }) else { return nil } @@ -273,7 +273,7 @@ return window } - @objc private func snapshot() { + @MainActor @objc private func snapshot() { if !PostHogSDK.shared.isSessionReplayActive() { return } diff --git a/PostHog/Replay/UIApplicationTracker.swift b/PostHog/Replay/UIApplicationTracker.swift index 181a6b514..ca15e8a98 100644 --- a/PostHog/Replay/UIApplicationTracker.swift +++ b/PostHog/Replay/UIApplicationTracker.swift @@ -10,7 +10,7 @@ import UIKit enum UIApplicationTracker { - private static var hasSwizzled = false + private nonisolated(unsafe) static var hasSwizzled = false static func swizzleSendEvent() { if hasSwizzled { @@ -78,6 +78,7 @@ let data: [String: Any] = ["type": 3, "data": touchData, "timestamp": timestamp] snapshotsData.append(data) } + if !snapshotsData.isEmpty { DispatchQueue.global().async { PostHogSDK.shared.capture("$snapshot", properties: ["$snapshot_source": "mobile", "$snapshot_data": snapshotsData]) diff --git a/PostHog/Replay/URLSessionInterceptor.swift b/PostHog/Replay/URLSessionInterceptor.swift index 6108f29d2..88aeea8d1 100644 --- a/PostHog/Replay/URLSessionInterceptor.swift +++ b/PostHog/Replay/URLSessionInterceptor.swift @@ -8,7 +8,7 @@ import Foundation - class URLSessionInterceptor { + class URLSessionInterceptor: @unchecked Sendable { private let config: PostHogConfig init(_ config: PostHogConfig) { @@ -17,7 +17,7 @@ /// An internal queue for synchronising the access to `samplesByTask`. private let queue = DispatchQueue(label: "com.posthog.URLSessionInterceptor", target: .global(qos: .utility)) - private var samplesByTask: [URLSessionTask: NetworkSample] = [:] + private nonisolated(unsafe) var samplesByTask: [URLSessionTask: NetworkSample] = [:] // MARK: - Interception Flow @@ -37,7 +37,8 @@ } let date = Date() - queue.async { + queue.async { [weak self] in + guard let self = self else { return } let sample = NetworkSample(timeOrigin: date, url: url.absoluteString) self.samplesByTask[task] = sample } @@ -101,7 +102,7 @@ config.sessionReplayConfig.captureNetworkTelemetry && PostHogSDK.shared.isSessionReplayActive() } - private func finish(task: URLSessionTask, sample: NetworkSample) { + @MainActor private func finish(task: URLSessionTask, sample: NetworkSample) { if !isCaptureNetworkEnabled() { return } diff --git a/PostHog/Replay/URLSessionSwizzler.swift b/PostHog/Replay/URLSessionSwizzler.swift index a635af84f..6237e3705 100644 --- a/PostHog/Replay/URLSessionSwizzler.swift +++ b/PostHog/Replay/URLSessionSwizzler.swift @@ -56,129 +56,131 @@ // MARK: - Swizzlings - typealias CompletionHandler = (Data?, URLResponse?, Error?) -> Void + typealias CompletionHandler = @Sendable (Data?, URLResponse?, Error?) -> Void /// Swizzles the `URLSession.dataTask(with:completionHandler:)` for `URLRequest`. class DataTaskWithURLRequestAndCompletion: MethodSwizzler< @convention(c) (URLSession, Selector, URLRequest, CompletionHandler?) -> URLSessionDataTask, @convention(block) (URLSession, URLRequest, CompletionHandler?) -> URLSessionDataTask - > { - private static let selector = #selector( - URLSession.dataTask(with:completionHandler:) as (URLSession) -> (URLRequest, @escaping CompletionHandler) -> URLSessionDataTask + >, @unchecked Sendable { + private nonisolated(unsafe) static let selector = #selector( + URLSession.dataTask(with:completionHandler:) as (URLSession) -> (URLRequest, @escaping CompletionHandler) -> URLSessionDataTask + ) + + private let method: FoundMethod + private let interceptor: URLSessionInterceptor + + static func build(interceptor: URLSessionInterceptor) throws -> DataTaskWithURLRequestAndCompletion { + try DataTaskWithURLRequestAndCompletion( + selector: selector, + klass: URLSession.self, + interceptor: interceptor ) + } - private let method: FoundMethod - private let interceptor: URLSessionInterceptor - - static func build(interceptor: URLSessionInterceptor) throws -> DataTaskWithURLRequestAndCompletion { - try DataTaskWithURLRequestAndCompletion( - selector: selector, - klass: URLSession.self, - interceptor: interceptor - ) - } - - private init(selector: Selector, klass: AnyClass, interceptor: URLSessionInterceptor) throws { - method = try Self.findMethod(with: selector, in: klass) - self.interceptor = interceptor - super.init() - } + private init(selector: Selector, klass: AnyClass, interceptor: URLSessionInterceptor) throws { + method = try Self.findMethod(with: selector, in: klass) + self.interceptor = interceptor + super.init() + } - func swizzle() { - typealias Signature = @convention(block) (URLSession, URLRequest, CompletionHandler?) -> URLSessionDataTask - swizzle(method) { previousImplementation -> Signature in { session, urlRequest, completionHandler -> URLSessionDataTask in - let task: URLSessionDataTask - if completionHandler != nil { - var taskReference: URLSessionDataTask? - let newCompletionHandler: CompletionHandler = { data, response, error in - if let task = taskReference { // sanity check, should always succeed - if let data = data { - self.interceptor.taskReceivedData(task: task, data: data) - } - self.interceptor.taskCompleted(task: task, error: error) + func swizzle() { + typealias Signature = @convention(block) (URLSession, URLRequest, CompletionHandler?) -> URLSessionDataTask + swizzle(method) { previousImplementation -> Signature in { session, urlRequest, completionHandler -> URLSessionDataTask in + let task: URLSessionDataTask + if completionHandler != nil { + var taskReference: URLSessionDataTask? + let newCompletionHandler: CompletionHandler = { [weak self, taskReference] data, response, error in + guard let self = self else { return } + if let task = taskReference { // sanity check, should always succeed + if let data = data { + self.interceptor.taskReceivedData(task: task, data: data) } - completionHandler?(data, response, error) + self.interceptor.taskCompleted(task: task, error: error) } - - task = previousImplementation(session, Self.selector, urlRequest, newCompletionHandler) - taskReference = task - } else { - // The `completionHandler` can be `nil` in two cases: - // - on iOS 11 or 12, where `dataTask(with:)` (for `URL` and `URLRequest`) calls - // the `dataTask(with:completionHandler:)` (for `URLRequest`) internally by nullifying the completion block. - // - when `[session dataTaskWithURL:completionHandler:]` is called in Objective-C with explicitly passing - // `nil` as the `completionHandler` (it produces a warning, but compiles). - task = previousImplementation(session, Self.selector, urlRequest, completionHandler) + completionHandler?(data, response, error) } - self.interceptor.taskCreated(task: task, session: session) - return task - } + + task = previousImplementation(session, Self.selector, urlRequest, newCompletionHandler) + taskReference = task + } else { + // The `completionHandler` can be `nil` in two cases: + // - on iOS 11 or 12, where `dataTask(with:)` (for `URL` and `URLRequest`) calls + // the `dataTask(with:completionHandler:)` (for `URLRequest`) internally by nullifying the completion block. + // - when `[session dataTaskWithURL:completionHandler:]` is called in Objective-C with explicitly passing + // `nil` as the `completionHandler` (it produces a warning, but compiles). + task = previousImplementation(session, Self.selector, urlRequest, completionHandler) } + self.interceptor.taskCreated(task: task, session: session) + return task + } } } + } /// Swizzles the `URLSession.dataTask(with:completionHandler:)` for `URL`. class DataTaskWithURLAndCompletion: MethodSwizzler< @convention(c) (URLSession, Selector, URL, CompletionHandler?) -> URLSessionDataTask, @convention(block) (URLSession, URL, CompletionHandler?) -> URLSessionDataTask - > { - private static let selector = #selector( - URLSession.dataTask(with:completionHandler:) as (URLSession) -> (URL, @escaping CompletionHandler) -> URLSessionDataTask + >, @unchecked Sendable { + private nonisolated(unsafe) static let selector = #selector( + URLSession.dataTask(with:completionHandler:) as (URLSession) -> (URL, @escaping CompletionHandler) -> URLSessionDataTask + ) + + private let method: FoundMethod + private let interceptor: URLSessionInterceptor + + static func build(interceptor: URLSessionInterceptor) throws -> DataTaskWithURLAndCompletion { + try DataTaskWithURLAndCompletion( + selector: selector, + klass: URLSession.self, + interceptor: interceptor ) + } - private let method: FoundMethod - private let interceptor: URLSessionInterceptor - - static func build(interceptor: URLSessionInterceptor) throws -> DataTaskWithURLAndCompletion { - try DataTaskWithURLAndCompletion( - selector: selector, - klass: URLSession.self, - interceptor: interceptor - ) - } - - private init(selector: Selector, klass: AnyClass, interceptor: URLSessionInterceptor) throws { - method = try Self.findMethod(with: selector, in: klass) - self.interceptor = interceptor - super.init() - } + private init(selector: Selector, klass: AnyClass, interceptor: URLSessionInterceptor) throws { + method = try Self.findMethod(with: selector, in: klass) + self.interceptor = interceptor + super.init() + } - func swizzle() { - typealias Signature = @convention(block) (URLSession, URL, CompletionHandler?) -> URLSessionDataTask - swizzle(method) { previousImplementation -> Signature in { session, url, completionHandler -> URLSessionDataTask in - let task: URLSessionDataTask - if completionHandler != nil { - var taskReference: URLSessionDataTask? - let newCompletionHandler: CompletionHandler = { data, response, error in - if let task = taskReference { // sanity check, should always succeed - if let data = data { - self.interceptor.taskReceivedData(task: task, data: data) - } - self.interceptor.taskCompleted(task: task, error: error) + func swizzle() { + typealias Signature = @convention(block) (URLSession, URL, CompletionHandler?) -> URLSessionDataTask + swizzle(method) { previousImplementation -> Signature in { session, url, completionHandler -> URLSessionDataTask in + let task: URLSessionDataTask + if completionHandler != nil { + var taskReference: URLSessionDataTask? + let newCompletionHandler: CompletionHandler = { [weak self, taskReference] data, response, error in + guard let self = self else { return } + if let task = taskReference { // sanity check, should always succeed + if let data = data { + self.interceptor.taskReceivedData(task: task, data: data) } - completionHandler?(data, response, error) + self.interceptor.taskCompleted(task: task, error: error) } - task = previousImplementation(session, Self.selector, url, newCompletionHandler) - taskReference = task - } else { - // The `completionHandler` can be `nil` in one case: - // - when `[session dataTaskWithURL:completionHandler:]` is called in Objective-C with explicitly passing - // `nil` as the `completionHandler` (it produces a warning, but compiles). - task = previousImplementation(session, Self.selector, url, completionHandler) + completionHandler?(data, response, error) } - self.interceptor.taskCreated(task: task, session: session) - return task - } + task = previousImplementation(session, Self.selector, url, newCompletionHandler) + taskReference = task + } else { + // The `completionHandler` can be `nil` in one case: + // - when `[session dataTaskWithURL:completionHandler:]` is called in Objective-C with explicitly passing + // `nil` as the `completionHandler` (it produces a warning, but compiles). + task = previousImplementation(session, Self.selector, url, completionHandler) } + self.interceptor.taskCreated(task: task, session: session) + return task + } } } + } /// Swizzles the `URLSession.dataTask(with:)` for `URLRequest`. class DataTaskWithURLRequest: MethodSwizzler< @convention(c) (URLSession, Selector, URLRequest) -> URLSessionDataTask, @convention(block) (URLSession, URLRequest) -> URLSessionDataTask > { - private static let selector = #selector( + private nonisolated(unsafe) static let selector = #selector( URLSession.dataTask(with:) as (URLSession) -> (URLRequest) -> URLSessionDataTask ) @@ -215,7 +217,7 @@ @convention(c) (URLSession, Selector, URL) -> URLSessionDataTask, @convention(block) (URLSession, URL) -> URLSessionDataTask > { - private static let selector = #selector( + private nonisolated(unsafe) static let selector = #selector( URLSession.dataTask(with:) as (URLSession) -> (URL) -> URLSessionDataTask ) diff --git a/PostHog/Replay/ViewLayoutTracker.swift b/PostHog/Replay/ViewLayoutTracker.swift index d48029c09..a24d63053 100644 --- a/PostHog/Replay/ViewLayoutTracker.swift +++ b/PostHog/Replay/ViewLayoutTracker.swift @@ -3,8 +3,9 @@ import UIKit enum ViewLayoutTracker { - static var hasChanges = false - private static var hasSwizzled = false + // nonisolated because its manually locked with setupLock + nonisolated(unsafe) static var hasChanges = false + private nonisolated(unsafe) static var hasSwizzled = false static func viewDidLayout(view _: UIView) { hasChanges = true diff --git a/PostHog/Utils/Hedgelog.swift b/PostHog/Utils/Hedgelog.swift index f53ad8694..ddcf21ee7 100644 --- a/PostHog/Utils/Hedgelog.swift +++ b/PostHog/Utils/Hedgelog.swift @@ -7,7 +7,7 @@ import Foundation -var hedgeLogEnabled = false +nonisolated(unsafe) var hedgeLogEnabled = false func toggleHedgeLog(_ enabled: Bool) { hedgeLogEnabled = enabled diff --git a/PostHog/Utils/Reachability.swift b/PostHog/Utils/Reachability.swift index ea6cae487..cb32f91f1 100644 --- a/PostHog/Utils/Reachability.swift +++ b/PostHog/Utils/Reachability.swift @@ -45,7 +45,7 @@ import Foundation static let reachabilityChanged = Notification.Name("reachabilityChanged") } - public class Reachability { + public class Reachability: @unchecked Sendable { public typealias NetworkReachable = (Reachability) -> Void public typealias NetworkUnreachable = (Reachability) -> Void @@ -72,7 +72,7 @@ import Foundation } @available(*, deprecated, renamed: "unavailable") - public static let none: Connection = .unavailable + public nonisolated(unsafe) static let none: Connection = .unavailable } public var whenReachable: NetworkReachable? @@ -271,7 +271,7 @@ import Foundation } func notifyReachabilityChanged() { - let notify = { [weak self] in + let notify: @Sendable () -> Void = { [weak self] in guard let self = self else { return } self.connection != .unavailable ? self.whenReachable?(self) : self.whenUnreachable?(self) self.notificationCenter.post(name: .reachabilityChanged, object: self) diff --git a/PostHogExample/ContentView.swift b/PostHogExample/ContentView.swift index d2e258465..f752ac8a2 100644 --- a/PostHogExample/ContentView.swift +++ b/PostHogExample/ContentView.swift @@ -46,7 +46,7 @@ class FeatureFlagsModel: ObservableObject { NotificationCenter.default.addObserver(self, selector: #selector(reloaded), name: PostHogSDK.didReceiveFeatureFlags, object: nil) } - @objc func reloaded() { + @MainActor @objc func reloaded() { boolValue = PostHogSDK.shared.isFeatureEnabled("4535-funnel-bar-viz") stringValue = PostHogSDK.shared.getFeatureFlag("multivariant") as? String payloadValue = PostHogSDK.shared.getFeatureFlagPayload("multivariant") as? [String: String]