From 89014de7d1765d445be33a7cf0f5a24f7263bedd Mon Sep 17 00:00:00 2001 From: TizianoCoroneo Date: Tue, 23 Apr 2024 17:52:18 +0200 Subject: [PATCH] fix: Atomic increment is now performed within one enqueued operation (#330) * Version 1.5.10 * fix: Atomic increment is now performed within one enqueued operation --------- Co-authored-by: Brandon Sneed --- Examples/other_plugins/IDFACollection.swift | 13 +++++++---- .../Platforms/Vendors/AppleUtils.swift | 6 +++++ .../Segment/Plugins/SegmentDestination.swift | 4 +++- Sources/Segment/Utilities/Atomic.swift | 8 +++++++ .../Policies/CountBasedFlushPolicy.swift | 4 +++- Tests/Segment-Tests/Atomic_Tests.swift | 23 +++++++++++++++++++ 6 files changed, 52 insertions(+), 6 deletions(-) create mode 100644 Tests/Segment-Tests/Atomic_Tests.swift diff --git a/Examples/other_plugins/IDFACollection.swift b/Examples/other_plugins/IDFACollection.swift index 4ee1550c..ef733238 100644 --- a/Examples/other_plugins/IDFACollection.swift +++ b/Examples/other_plugins/IDFACollection.swift @@ -77,10 +77,15 @@ class IDFACollection: Plugin { extension IDFACollection: iOSLifecycle { func applicationDidBecomeActive(application: UIApplication?) { let status = ATTrackingManager.trackingAuthorizationStatus - if status == .notDetermined && !alreadyAsked { - // we don't know, so should ask the user. - alreadyAsked = true - askForPermission() + + _alreadyAsked.withValue { alreadyAsked in + if status == .notDetermined && !alreadyAsked { + // we don't know, so should ask the user. + alreadyAsked = true + DispatchQueue.main.async { + askForPermission() + } + } } } } diff --git a/Sources/Segment/Plugins/Platforms/Vendors/AppleUtils.swift b/Sources/Segment/Plugins/Platforms/Vendors/AppleUtils.swift index 60245d73..2f7021bf 100644 --- a/Sources/Segment/Plugins/Platforms/Vendors/AppleUtils.swift +++ b/Sources/Segment/Plugins/Platforms/Vendors/AppleUtils.swift @@ -76,6 +76,9 @@ internal class iOSVendorSystem: VendorSystem { // BKS: It was discovered that on some platforms there can be a delay in retrieval. // It has to be fetched on the main thread, so we've spun it off // async and cache it when it comes back. + // Note that due to how the `@Atomic` wrapper works, this boolean check may pass twice or more + // times before the value is updated, fetching the user agent multiple times as the result. + // This is not a big deal as the `userAgent` value is not expected to change often. if Self.asyncUserAgent == nil { DispatchQueue.main.async { Self.asyncUserAgent = WKWebView().value(forKey: "userAgent") as? String @@ -248,6 +251,9 @@ internal class MacOSVendorSystem: VendorSystem { // BKS: It was discovered that on some platforms there can be a delay in retrieval. // It has to be fetched on the main thread, so we've spun it off // async and cache it when it comes back. + // Note that due to how the `@Atomic` wrapper works, this boolean check may pass twice or more + // times before the value is updated, fetching the user agent multiple times as the result. + // This is not a big deal as the `userAgent` value is not expected to change often. if Self.asyncUserAgent == nil { DispatchQueue.main.async { Self.asyncUserAgent = WKWebView().value(forKey: "userAgent") as? String diff --git a/Sources/Segment/Plugins/SegmentDestination.swift b/Sources/Segment/Plugins/SegmentDestination.swift index 41fdd42c..1ab2260f 100644 --- a/Sources/Segment/Plugins/SegmentDestination.swift +++ b/Sources/Segment/Plugins/SegmentDestination.swift @@ -116,7 +116,9 @@ public class SegmentDestination: DestinationPlugin, Subscriber, FlushCompletion guard let storage = self.storage else { return } // Send Event to File System storage.write(.events, value: event) - eventCount += 1 + self._eventCount.withValue { count in + count += 1 + } } public func flush() { diff --git a/Sources/Segment/Utilities/Atomic.swift b/Sources/Segment/Utilities/Atomic.swift index 50984aba..1dc6c07e 100644 --- a/Sources/Segment/Utilities/Atomic.swift +++ b/Sources/Segment/Utilities/Atomic.swift @@ -29,4 +29,12 @@ public class Atomic { get { return queue.sync { return value } } set { queue.sync { value = newValue } } } + + @discardableResult + public func withValue(_ operation: (inout T) -> Void) -> T { + queue.sync { + operation(&self.value) + return self.value + } + } } diff --git a/Sources/Segment/Utilities/Policies/CountBasedFlushPolicy.swift b/Sources/Segment/Utilities/Policies/CountBasedFlushPolicy.swift index 0756067f..0a07edf1 100644 --- a/Sources/Segment/Utilities/Policies/CountBasedFlushPolicy.swift +++ b/Sources/Segment/Utilities/Policies/CountBasedFlushPolicy.swift @@ -37,7 +37,9 @@ public class CountBasedFlushPolicy: FlushPolicy { } public func updateState(event: RawEvent) { - count += 1 + _count.withValue { value in + value += 1 + } } public func reset() { diff --git a/Tests/Segment-Tests/Atomic_Tests.swift b/Tests/Segment-Tests/Atomic_Tests.swift new file mode 100644 index 00000000..6e1d1216 --- /dev/null +++ b/Tests/Segment-Tests/Atomic_Tests.swift @@ -0,0 +1,23 @@ +import XCTest +@testable import Segment + +final class Atomic_Tests: XCTestCase { + + func testAtomicIncrement() { + + @Atomic var counter = 0 + + DispatchQueue.concurrentPerform(iterations: 1000) { _ in + // counter += 1 would fail, because it is expanded to: + // `let oldValue = queue.sync { counter }` + // `queue.sync { counter = oldValue + 1 }` + // And the threads are free to suspend in between the two calls to `queue.sync`. + + _counter.withValue { value in + value += 1 + } + } + + XCTAssertEqual(counter, 1000) + } +}