From c34f633f1506150bfc87acd121747fba0724df70 Mon Sep 17 00:00:00 2001 From: xiaoweii Date: Thu, 18 Jul 2024 23:12:36 +0800 Subject: [PATCH] fix: attributes thread safety issue --- .../Analytics/AnalyticsClient.swift | 21 +++++++++++++++++++ .../Clickstream/Event/ClickstreamEvent.swift | 9 +++++++- .../Clickstream/EventRecorderTest.swift | 14 +++++++++++++ 3 files changed, 43 insertions(+), 1 deletion(-) diff --git a/Sources/Clickstream/Dependency/Clickstream/Analytics/AnalyticsClient.swift b/Sources/Clickstream/Dependency/Clickstream/Analytics/AnalyticsClient.swift index e20d932..72df60c 100644 --- a/Sources/Clickstream/Dependency/Clickstream/Analytics/AnalyticsClient.swift +++ b/Sources/Clickstream/Dependency/Clickstream/Analytics/AnalyticsClient.swift @@ -29,6 +29,7 @@ class AnalyticsClient: AnalyticsClientBehaviour { private(set) var simpleUserAttributes: [String: Any] = [:] private let clickstream: ClickstreamContext private(set) var userId: String? + let attributeLock = NSLock() var autoRecordClient: AutoRecordEventClient init(clickstream: ClickstreamContext, @@ -45,6 +46,7 @@ class AnalyticsClient: AnalyticsClientBehaviour { } func addGlobalAttribute(_ attribute: AttributeValue, forKey key: String) { + attributeLock.lock() let eventError = EventChecker.checkAttribute( currentNumber: globalAttributes.count, key: key, @@ -54,9 +56,11 @@ class AnalyticsClient: AnalyticsClientBehaviour { } else { globalAttributes[key] = attribute } + attributeLock.unlock() } func addUserAttribute(_ attribute: AttributeValue, forKey key: String) { + attributeLock.lock() let eventError = EventChecker.checkUserAttribute(currentNumber: allUserAttributes.count, key: key, value: attribute) @@ -72,14 +76,19 @@ class AnalyticsClient: AnalyticsClientBehaviour { userAttribute["set_timestamp"] = Date().millisecondsSince1970 allUserAttributes[key] = userAttribute } + attributeLock.unlock() } func removeGlobalAttribute(forKey key: String) { + attributeLock.lock() globalAttributes[key] = nil + attributeLock.unlock() } func removeUserAttribute(forKey key: String) { + attributeLock.lock() allUserAttributes[key] = nil + attributeLock.unlock() } func updateUserId(_ id: String?) { @@ -87,7 +96,9 @@ class AnalyticsClient: AnalyticsClientBehaviour { userId = id UserDefaultsUtil.saveCurrentUserId(storage: clickstream.storage, userId: userId) if let newUserId = id, !newUserId.isEmpty { + attributeLock.lock() allUserAttributes = JsonObject() + attributeLock.unlock() let userInfo = UserDefaultsUtil.getNewUserInfo(storage: clickstream.storage, userId: newUserId) // swiftlint:disable force_cast clickstream.userUniqueId = userInfo["user_unique_id"] as! String @@ -105,7 +116,9 @@ class AnalyticsClient: AnalyticsClientBehaviour { } func updateUserAttributes() { + attributeLock.lock() UserDefaultsUtil.updateUserAttributes(storage: clickstream.storage, userAttributes: allUserAttributes) + attributeLock.unlock() } // MARK: - Event recording @@ -130,6 +143,9 @@ class AnalyticsClient: AnalyticsClientBehaviour { } func record(_ event: ClickstreamEvent) throws { + if event.eventType != Event.PresetEvent.CLICKSTREAM_ERROR{ + attributeLock.lock() + } for (key, attribute) in globalAttributes { event.addGlobalAttribute(attribute, forKey: key) } @@ -147,6 +163,9 @@ class AnalyticsClient: AnalyticsClientBehaviour { event.setUserAttribute(simpleUserAttributes) } try eventRecorder.save(event) + if event.eventType != Event.PresetEvent.CLICKSTREAM_ERROR{ + attributeLock.unlock() + } } func recordEventError(_ eventError: EventChecker.EventError) { @@ -167,6 +186,7 @@ class AnalyticsClient: AnalyticsClientBehaviour { } func getSimpleUserAttributes() -> [String: Any] { + attributeLock.lock() simpleUserAttributes = [:] simpleUserAttributes[Event.ReservedAttribute.USER_FIRST_TOUCH_TIMESTAMP] = allUserAttributes[Event.ReservedAttribute.USER_FIRST_TOUCH_TIMESTAMP] @@ -174,6 +194,7 @@ class AnalyticsClient: AnalyticsClientBehaviour { simpleUserAttributes[Event.ReservedAttribute.USER_ID] = allUserAttributes[Event.ReservedAttribute.USER_ID] } + attributeLock.unlock() return simpleUserAttributes } } diff --git a/Sources/Clickstream/Dependency/Clickstream/Event/ClickstreamEvent.swift b/Sources/Clickstream/Dependency/Clickstream/Event/ClickstreamEvent.swift index 301d9c7..27569d2 100644 --- a/Sources/Clickstream/Dependency/Clickstream/Event/ClickstreamEvent.swift +++ b/Sources/Clickstream/Dependency/Clickstream/Event/ClickstreamEvent.swift @@ -21,6 +21,7 @@ class ClickstreamEvent: AnalyticsPropertiesModel { private(set) lazy var attributes: [String: AttributeValue] = [:] private(set) lazy var items: [ClickstreamAttribute] = [] private(set) lazy var userAttributes: [String: Any] = [:] + let attributeLock = NSLock() let systemInfo: SystemInfo let netWorkType: String @@ -72,7 +73,11 @@ class ClickstreamEvent: AnalyticsPropertiesModel { } func setUserAttribute(_ attributes: [String: Any]) { - userAttributes = attributes + attributeLock.lock() + for attr in attributes { + userAttributes[attr.key] = attr.value + } + attributeLock.unlock() } func attribute(forKey key: String) -> AttributeValue? { @@ -109,9 +114,11 @@ class ClickstreamEvent: AnalyticsPropertiesModel { if !items.isEmpty { event["items"] = items } + attributeLock.lock() if !userAttributes.isEmpty { event["user"] = userAttributes } + attributeLock.unlock() event["attributes"] = getAttributeObject(from: attributes) return event } diff --git a/Tests/ClickstreamTests/Clickstream/EventRecorderTest.swift b/Tests/ClickstreamTests/Clickstream/EventRecorderTest.swift index 4b74edf..d4fe57c 100644 --- a/Tests/ClickstreamTests/Clickstream/EventRecorderTest.swift +++ b/Tests/ClickstreamTests/Clickstream/EventRecorderTest.swift @@ -182,6 +182,20 @@ class EventRecorderTest: XCTestCase { XCTAssertEqual(85.5, (userAttributes["score"] as! JsonObject)["value"] as! Double) XCTAssertEqual("carl", (userAttributes["_user_name"] as! JsonObject)["value"] as! String) } + + func testModifyGlobalUserAttributesWillNotAffectTheEventUserAttributes() throws { + var userInfo = JsonObject() + let currentTimeStamp = Date().millisecondsSince1970 + var userNameInfo = JsonObject() + userNameInfo["value"] = "carl" + userNameInfo["set_timestamp"] = currentTimeStamp + userInfo["_user_name"] = userNameInfo + clickstreamEvent.setUserAttribute(userInfo) + userInfo = JsonObject() + userNameInfo["value"] = "mike" + userInfo["_user_name"] = userNameInfo + XCTAssertEqual((clickstreamEvent.userAttributes["_user_name"] as! JsonObject)["value"] as! String, "carl") + } func testRecordEventWithInvalidAttribute() throws { clickstreamEvent.addGlobalAttribute(Decimal.nan, forKey: "invalidDecimal")