diff --git a/.github/workflows/build.yml b/.github/workflows/build.yml index 38d8a174a..50bf1d651 100644 --- a/.github/workflows/build.yml +++ b/.github/workflows/build.yml @@ -87,7 +87,7 @@ jobs: -sdk "${{ matrix.sdk }}" \ -destination "${{ matrix.destination }}" \ -only-testing ${{ matrix.target }} \ - clean test | xcpretty + clean test build_objc_demo_app: name: "ObjC demo (iOS ${{ matrix.version.ios }})" diff --git a/Sources/Core/Emitter/Emitter.swift b/Sources/Core/Emitter/Emitter.swift index e9f3efbf6..ce3c9f5d4 100644 --- a/Sources/Core/Emitter/Emitter.swift +++ b/Sources/Core/Emitter/Emitter.swift @@ -18,9 +18,9 @@ let POST_WRAPPER_BYTES = 88 class Emitter: EmitterEventProcessing { - private var timer: Timer? + private var timer: InternalQueueTimer? - private var _pausedEmit = false + private var pausedEmit = false /// Custom NetworkConnection istance to handle connection outside the emitter. private let networkConnection: NetworkConnection @@ -30,9 +30,8 @@ class Emitter: EmitterEventProcessing { let eventStore: EventStore - private var _sending = false /// Whether the emitter is currently sending. - var isSending: Bool { return sync { _sending } } + var isSending: Bool = false /// Collector endpoint. var urlEndpoint: String? { @@ -80,29 +79,19 @@ class Emitter: EmitterEventProcessing { } } - private var _bufferOption: BufferOption = EmitterDefaults.bufferOption /// Buffer option - var bufferOption: BufferOption { - get { return sync { _bufferOption } } - set(bufferOption) { sync { _bufferOption = bufferOption } } - } + var bufferOption: BufferOption = EmitterDefaults.bufferOption - private weak var _callback: RequestCallback? /// Callbacks supplied with number of failures and successes of sent events. - var callback: RequestCallback? { - get { return sync { _callback } } - set(callback) { sync { _callback = callback } } - } + weak var callback: RequestCallback? private var _emitRange = EmitterDefaults.emitRange /// Number of events retrieved from the database when needed. var emitRange: Int { - get { return sync { _emitRange } } + get { return _emitRange } set(emitRange) { - sync { - if emitRange > 0 { - _emitRange = emitRange - } + if emitRange > 0 { + _emitRange = emitRange } } } @@ -127,13 +116,11 @@ class Emitter: EmitterEventProcessing { /// Byte limit for GET requests. private var _byteLimitGet = EmitterDefaults.byteLimitGet var byteLimitGet: Int { - get { return sync { _byteLimitGet } } + get { return _byteLimitGet } set(byteLimitGet) { - sync { - _byteLimitGet = byteLimitGet - if let networkConnection = networkConnection as? DefaultNetworkConnection { - networkConnection.byteLimitGet = byteLimitGet - } + _byteLimitGet = byteLimitGet + if let networkConnection = networkConnection as? DefaultNetworkConnection { + networkConnection.byteLimitGet = byteLimitGet } } } @@ -141,13 +128,11 @@ class Emitter: EmitterEventProcessing { private var _byteLimitPost = EmitterDefaults.byteLimitPost /// Byte limit for POST requests. var byteLimitPost: Int { - get { return sync { _byteLimitPost } } + get { return _byteLimitPost } set(byteLimitPost) { - sync { - _byteLimitPost = byteLimitPost - if let networkConnection = networkConnection as? DefaultNetworkConnection { - networkConnection.byteLimitPost = byteLimitPost - } + _byteLimitPost = byteLimitPost + if let networkConnection = networkConnection as? DefaultNetworkConnection { + networkConnection.byteLimitPost = byteLimitPost } } } @@ -200,16 +185,12 @@ class Emitter: EmitterEventProcessing { /// Custom retry rules for HTTP status codes. private var _customRetryForStatusCodes: [Int : Bool] = [:] var customRetryForStatusCodes: [Int : Bool]? { - get { return sync { return _customRetryForStatusCodes } } - set { sync { _customRetryForStatusCodes = newValue ?? [:] } } + get { return _customRetryForStatusCodes } + set { _customRetryForStatusCodes = newValue ?? [:] } } /// Whether retrying failed requests is allowed - private var _retryFailedRequests: Bool = EmitterDefaults.retryFailedRequests - var retryFailedRequests: Bool { - get { return sync { _retryFailedRequests } } - set { sync { _retryFailedRequests = newValue } } - } + var retryFailedRequests: Bool = EmitterDefaults.retryFailedRequests /// Returns the number of events in the DB. var dbCount: Int { @@ -270,117 +251,114 @@ class Emitter: EmitterEventProcessing { // MARK: - Pause/Resume methods func resumeTimer() { - weak var weakSelf = self - pauseTimer() - // NOTE: consider whether it is really necessary to use the main queue or we can dispatch on a background queue - DispatchQueue.main.async { - let timer = Timer.scheduledTimer(withTimeInterval: TimeInterval(kSPDefaultBufferTimeout), repeats: true) { [weak self] timer in - self?.flush() - } - weakSelf?.sync { - weakSelf?.timer = timer - } + self.timer = InternalQueue.startTimer(TimeInterval(kSPDefaultBufferTimeout)) { [weak self] in + self?.flush() } } /// Suspends timer for periodically sending events to collector. func pauseTimer() { - sync { - timer?.invalidate() - timer = nil - } + timer?.invalidate() + timer = nil } /// Allows sending events to collector. func resumeEmit() { - sync { _pausedEmit = false } + pausedEmit = false flush() } /// Suspends sending events to collector. func pauseEmit() { - sync { _pausedEmit = true } + pausedEmit = true } /// Insert a Payload object into the buffer to be sent to collector. /// This method will add the payload to the database and flush (send all events). /// - Parameter eventPayload: A Payload containing a completed event to be added into the buffer. func addPayload(toBuffer eventPayload: Payload) { - DispatchQueue.global(qos: .default).async { [weak self] in - self?.eventStore.addEvent(eventPayload) - self?.flush() - } + self.eventStore.addEvent(eventPayload) + self.flush() } /// Empties the buffer of events using the respective HTTP request method. func flush() { if requestToStartSending() { - emitAsync { - self.attemptEmit() - self.sync { self._sending = false } - } + self.attemptEmit() } } // MARK: - Control methods private func attemptEmit() { - if eventStore.count() == 0 { + InternalQueue.onQueuePrecondition() + + let events = eventStore.emittableEvents(withQueryLimit: UInt(emitRange)) + if events.isEmpty { logDebug(message: "Database empty. Returning.") + stopSending() return } - - let events = eventStore.emittableEvents(withQueryLimit: UInt(emitRange)) + let requests = buildRequests(fromEvents: events) - let sendResults = networkConnection.sendRequests(requests) - - logVerbose(message: "Processing emitter results.") - - var successCount = 0 - var failedWillRetryCount = 0 - var failedWontRetryCount = 0 - var removableEvents: [Int64] = [] - - for result in sendResults { - let resultIndexArray = result.storeIds - if result.isSuccessful { - successCount += resultIndexArray?.count ?? 0 - if let array = resultIndexArray { - removableEvents.append(contentsOf: array) + + let processResults: ([RequestResult]) -> Void = { sendResults in + logVerbose(message: "Processing emitter results.") + + var successCount = 0 + var failedWillRetryCount = 0 + var failedWontRetryCount = 0 + var removableEvents: [Int64] = [] + + for result in sendResults { + let resultIndexArray = result.storeIds + if result.isSuccessful { + successCount += resultIndexArray?.count ?? 0 + if let array = resultIndexArray { + removableEvents.append(contentsOf: array) + } + } else if result.shouldRetry(self.customRetryForStatusCodes, retryAllowed: self.retryFailedRequests) { + failedWillRetryCount += resultIndexArray?.count ?? 0 + } else { + failedWontRetryCount += resultIndexArray?.count ?? 0 + if let array = resultIndexArray { + removableEvents.append(contentsOf: array) + } + logError(message: String(format: "Sending events to Collector failed with status %ld. Events will be dropped.", result.statusCode ?? -1)) } - } else if result.shouldRetry(customRetryForStatusCodes, retryAllowed: retryFailedRequests) { - failedWillRetryCount += resultIndexArray?.count ?? 0 - } else { - failedWontRetryCount += resultIndexArray?.count ?? 0 - if let array = resultIndexArray { - removableEvents.append(contentsOf: array) + } + let allFailureCount = failedWillRetryCount + failedWontRetryCount + + _ = self.eventStore.removeEvents(withIds: removableEvents) + + logDebug(message: String(format: "Success Count: %d", successCount)) + logDebug(message: String(format: "Failure Count: %d", allFailureCount)) + + if let callback = self.callback { + if allFailureCount == 0 { + callback.onSuccess(withCount: successCount) + } else { + callback.onFailure(withCount: allFailureCount, successCount: successCount) } - logError(message: String(format: "Sending events to Collector failed with status %ld. Events will be dropped.", result.statusCode ?? -1)) } - } - let allFailureCount = failedWillRetryCount + failedWontRetryCount - - let _ = eventStore.removeEvents(withIds: removableEvents) - - logDebug(message: String(format: "Success Count: %d", successCount)) - logDebug(message: String(format: "Failure Count: %d", allFailureCount)) - - if let callback = callback { - if allFailureCount == 0 { - callback.onSuccess(withCount: successCount) + + if failedWillRetryCount > 0 && successCount == 0 { + logDebug(message: "Ending emitter run as all requests failed.") + + self.scheduleStopSending() } else { - callback.onFailure(withCount: allFailureCount, successCount: successCount) + self.attemptEmit() } } - - if failedWillRetryCount > 0 && successCount == 0 { - logDebug(message: "Ending emitter run as all requests failed.") - Thread.sleep(forTimeInterval: 5) - return - } else { - self.attemptEmit() + + emitAsync { + let sendResults = self.networkConnection.sendRequests(requests) + + InternalQueue.async { + processResults(sendResults) + } } } @@ -388,15 +366,9 @@ class Emitter: EmitterEventProcessing { var requests: [Request] = [] let sendingTime = Utilities.getTimestamp() - let httpMethod = method - let (bufferOptionValue, byteLimit) = sync { - return ( - _bufferOption.rawValue, - httpMethod == .get ? _byteLimitGet : _byteLimitPost - ) - } + let byteLimit = method == .get ? byteLimitGet : byteLimitPost - if httpMethod == .get { + if method == .get { for event in events { let payload = event.payload addSendingTime(to: payload, timestamp: sendingTime) @@ -410,7 +382,7 @@ class Emitter: EmitterEventProcessing { var eventArray: [Payload] = [] var indexArray: [Int64] = [] - let iUntil = min(i + bufferOptionValue, events.count) + let iUntil = min(i + bufferOption.rawValue, events.count) for j in i.. Bool { - return sync { - if !_sending && !_pausedEmit { - _sending = true - return true - } else { - return false - } + if !isSending && !pausedEmit { + isSending = true + return true + } else { + return false } } - // MARK: - dispatch queues - - private let dispatchQueue = DispatchQueue(label: "snowplow.tracker.emitter") + private func scheduleStopSending() { + InternalQueue.asyncAfter(TimeInterval(5)) { [weak self] in + self?.stopSending() + } + } - private func sync(_ callback: () -> T) -> T { - dispatchPrecondition(condition: .notOnQueue(dispatchQueue)) - - return dispatchQueue.sync(execute: callback) + private func stopSending() { + isSending = false } - private let emitQueue = DispatchQueue(label: "snowplow.tracker.emitter.emit") + // MARK: - dispatch queues + + private let emitQueue = DispatchQueue(label: "snowplow.emitter") private func emitAsync(_ callback: @escaping () -> Void) { emitQueue.async(execute: callback) diff --git a/Sources/Core/InternalQueue/EcommerceControllerIQWrapper.swift b/Sources/Core/InternalQueue/EcommerceControllerIQWrapper.swift new file mode 100644 index 000000000..da2477cba --- /dev/null +++ b/Sources/Core/InternalQueue/EcommerceControllerIQWrapper.swift @@ -0,0 +1,39 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class EcommerceControllerIQWrapper: EcommerceController { + + private let controller: EcommerceController + + init(controller: EcommerceController) { + self.controller = controller + } + + func setEcommerceScreen(_ screen: EcommerceScreenEntity) { + InternalQueue.sync { controller.setEcommerceScreen(screen) } + } + + func setEcommerceUser(_ user: EcommerceUserEntity) { + InternalQueue.sync { controller.setEcommerceUser(user) } + } + + func removeEcommerceScreen() { + InternalQueue.sync { controller.removeEcommerceScreen() } + } + + func removeEcommerceUser() { + InternalQueue.sync { controller.removeEcommerceUser() } + } +} diff --git a/Sources/Core/InternalQueue/EmitterControllerIQWrapper.swift b/Sources/Core/InternalQueue/EmitterControllerIQWrapper.swift new file mode 100644 index 000000000..ce60d20af --- /dev/null +++ b/Sources/Core/InternalQueue/EmitterControllerIQWrapper.swift @@ -0,0 +1,93 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class EmitterControllerIQWrapper: EmitterController { + + private let controller: EmitterController + + init(controller: EmitterController) { + self.controller = controller + } + + // MARK: - Properties + + var bufferOption: BufferOption { + get { return InternalQueue.sync { controller.bufferOption } } + set { InternalQueue.sync { controller.bufferOption = newValue } } + } + + var byteLimitGet: Int { + get { return InternalQueue.sync { controller.byteLimitGet } } + set { InternalQueue.sync { controller.byteLimitGet = newValue } } + } + + var byteLimitPost: Int { + get { return InternalQueue.sync { controller.byteLimitPost } } + set { InternalQueue.sync { controller.byteLimitPost = newValue } } + } + + var serverAnonymisation: Bool { + get { return InternalQueue.sync { controller.serverAnonymisation } } + set { InternalQueue.sync { controller.serverAnonymisation = newValue } } + } + + var emitRange: Int { + get { return InternalQueue.sync { controller.emitRange } } + set { InternalQueue.sync { controller.emitRange = newValue } } + } + + var threadPoolSize: Int { + get { return InternalQueue.sync { controller.threadPoolSize } } + set { InternalQueue.sync { controller.threadPoolSize = newValue } } + } + + var requestCallback: RequestCallback? { + get { return InternalQueue.sync { controller.requestCallback } } + set { InternalQueue.sync { controller.requestCallback = newValue } } + } + + var dbCount: Int { + return InternalQueue.sync { controller.dbCount } + } + + var isSending: Bool { + return InternalQueue.sync { controller.isSending } + } + + var customRetryForStatusCodes: [Int : Bool]? { + get { return InternalQueue.sync { controller.customRetryForStatusCodes } } + set { InternalQueue.sync { controller.customRetryForStatusCodes = newValue } } + } + + var retryFailedRequests: Bool { + get { return InternalQueue.sync { controller.retryFailedRequests } } + set { InternalQueue.sync { controller.retryFailedRequests = newValue } } + } + + // MARK: - Methods + + func flush() { + InternalQueue.sync { controller.flush() } + } + + func pause() { + InternalQueue.sync { controller.pause() } + } + + func resume() { + InternalQueue.sync { controller.resume() } + } + +} diff --git a/Sources/Core/InternalQueue/GDPRControllerIQWrapper.swift b/Sources/Core/InternalQueue/GDPRControllerIQWrapper.swift new file mode 100644 index 000000000..a03cecc8b --- /dev/null +++ b/Sources/Core/InternalQueue/GDPRControllerIQWrapper.swift @@ -0,0 +1,60 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class GDPRControllerIQWrapper: GDPRController { + + private let controller: GDPRController + + init(controller: GDPRController) { + self.controller = controller + } + + // MARK: - Methods + + func reset(basis: GDPRProcessingBasis, documentId: String?, documentVersion: String?, documentDescription: String?) { + InternalQueue.sync { + controller.reset(basis: basis, documentId: documentId, documentVersion: documentVersion, documentDescription: documentDescription) + } + } + + func disable() { + InternalQueue.sync { controller.disable() } + } + + var isEnabled: Bool { + return InternalQueue.sync { controller.isEnabled } + } + + func enable() -> Bool { + InternalQueue.sync { controller.enable() } + } + + var basisForProcessing: GDPRProcessingBasis { + InternalQueue.sync { controller.basisForProcessing } + } + + var documentId: String? { + InternalQueue.sync { controller.documentId } + } + + var documentVersion: String? { + InternalQueue.sync { controller.documentVersion } + } + + var documentDescription: String? { + InternalQueue.sync { controller.documentDescription } + } + +} diff --git a/Sources/Core/InternalQueue/GlobalContextsControllerIQWrapper.swift b/Sources/Core/InternalQueue/GlobalContextsControllerIQWrapper.swift new file mode 100644 index 000000000..86c47375d --- /dev/null +++ b/Sources/Core/InternalQueue/GlobalContextsControllerIQWrapper.swift @@ -0,0 +1,41 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class GlobalContextsControllerIQWrapper: GlobalContextsController { + + private let controller: GlobalContextsController + + init(controller: GlobalContextsController) { + self.controller = controller + } + + var contextGenerators: [String : GlobalContext] { + get { InternalQueue.sync { controller.contextGenerators } } + set { InternalQueue.sync { controller.contextGenerators = newValue } } + } + + func add(tag: String, contextGenerator generator: GlobalContext) -> Bool { + return InternalQueue.sync { controller.add(tag: tag, contextGenerator: generator) } + } + + func remove(tag: String) -> GlobalContext? { + return InternalQueue.sync { controller.remove(tag: tag) } + } + + var tags: [String] { + return InternalQueue.sync { controller.tags } + } + +} diff --git a/Sources/Core/InternalQueue/InternalQueue.swift b/Sources/Core/InternalQueue/InternalQueue.swift new file mode 100644 index 000000000..93f739597 --- /dev/null +++ b/Sources/Core/InternalQueue/InternalQueue.swift @@ -0,0 +1,56 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class InternalQueue { + static func sync(_ callback: () -> T) -> T { + dispatchPrecondition(condition: .notOnQueue(serialQueue)) + + return serialQueue.sync(execute: callback) + } + + static func async(_ callback: @escaping () -> Void) { + serialQueue.async(execute: callback) + } + + static func asyncAfter(_ interval: TimeInterval, _ callback: @escaping () -> Void) { + serialQueue.asyncAfter(deadline: .now() + interval, execute: callback) + } + + static func startTimer(_ interval: TimeInterval, _ callback: @escaping () -> Void) -> InternalQueueTimer { + let timer = InternalQueueTimer() + + asyncAfter(interval) { + timerFired(timer: timer, interval: interval, callback: callback) + } + + return timer + } + + static private func timerFired(timer: InternalQueueTimer, interval: TimeInterval, callback: @escaping () -> Void) { + if timer.active { + asyncAfter(interval) { + timerFired(timer: timer, interval: interval, callback: callback) + } + + callback() + } + } + + static func onQueuePrecondition() { + dispatchPrecondition(condition: .onQueue(serialQueue)) + } + + private static let serialQueue = DispatchQueue(label: "snowplow") +} diff --git a/Sources/Core/InternalQueue/InternalQueueTimer.swift b/Sources/Core/InternalQueue/InternalQueueTimer.swift new file mode 100644 index 000000000..f11103018 --- /dev/null +++ b/Sources/Core/InternalQueue/InternalQueueTimer.swift @@ -0,0 +1,22 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class InternalQueueTimer { + var active = true + + func invalidate() { + active = false + } +} diff --git a/Sources/Core/InternalQueue/MediaControllerIQWrapper.swift b/Sources/Core/InternalQueue/MediaControllerIQWrapper.swift new file mode 100644 index 000000000..7c766721a --- /dev/null +++ b/Sources/Core/InternalQueue/MediaControllerIQWrapper.swift @@ -0,0 +1,59 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation +#if !os(watchOS) +import AVKit +#endif + +class MediaControllerIQWrapper: MediaController { + + private let controller: MediaController + + init(controller: MediaController) { + self.controller = controller + } + + func startMediaTracking(id: String) -> MediaTracking { + return InternalQueue.sync { + MediaTrackingIQWrapper(tracking: controller.startMediaTracking(id: id)) + } + } + + func startMediaTracking(id: String, player: MediaPlayerEntity? = nil) -> MediaTracking { + return InternalQueue.sync { + MediaTrackingIQWrapper(tracking: controller.startMediaTracking(id: id, player: player)) + } + } + + func startMediaTracking(configuration: MediaTrackingConfiguration) -> MediaTracking { + return InternalQueue.sync { + MediaTrackingIQWrapper(tracking: controller.startMediaTracking(configuration: configuration)) + } + } + +#if !os(watchOS) + func startMediaTracking(player: AVPlayer, + configuration: MediaTrackingConfiguration) -> MediaTracking { + return InternalQueue.sync { controller.startMediaTracking(player: player, configuration: configuration) } + } +#endif + + func mediaTracking(id: String) -> MediaTracking? { + return InternalQueue.sync { controller.mediaTracking(id: id) } + } + + func endMediaTracking(id: String) { + InternalQueue.sync { controller.endMediaTracking(id: id) } + } +} diff --git a/Sources/Core/InternalQueue/MediaTrackingIQWrapper.swift b/Sources/Core/InternalQueue/MediaTrackingIQWrapper.swift new file mode 100644 index 000000000..7d27f86ce --- /dev/null +++ b/Sources/Core/InternalQueue/MediaTrackingIQWrapper.swift @@ -0,0 +1,68 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class MediaTrackingIQWrapper: MediaTracking { + + private let tracking: MediaTracking + + init(tracking: MediaTracking) { + self.tracking = tracking + } + + var id: String { + return InternalQueue.sync { tracking.id } + } + + // MARK: Update methods overloads + + func update(player: MediaPlayerEntity?) { + return InternalQueue.sync { tracking.update(player: player) } + } + + func update(player: MediaPlayerEntity?, ad: MediaAdEntity?, adBreak: MediaAdBreakEntity?) { + return InternalQueue.sync { tracking.update(player: player, ad: ad, adBreak: adBreak) } + } + + // MARK: Track methods overloads + + func track(_ event: Event) { + InternalQueue.sync { tracking.track(event) } + } + + func track(_ event: Event, player: MediaPlayerEntity?) { + InternalQueue.sync { tracking.track(event, player: player) } + } + + func track(_ event: Event, ad: MediaAdEntity?) { + InternalQueue.sync { tracking.track(event, ad: ad) } + } + + func track(_ event: Event, player: MediaPlayerEntity?, ad: MediaAdEntity?) { + InternalQueue.sync { tracking.track(event, player: player, ad: ad) } + } + + func track(_ event: Event, adBreak: MediaAdBreakEntity?) { + InternalQueue.sync { tracking.track(event, adBreak: adBreak) } + } + + func track(_ event: Event, player: MediaPlayerEntity?, adBreak: MediaAdBreakEntity?) { + InternalQueue.sync { tracking.track(event, player: player, adBreak: adBreak) } + } + + func track(_ event: Event, player: MediaPlayerEntity?, ad: MediaAdEntity?, adBreak: MediaAdBreakEntity?) { + InternalQueue.sync { tracking.track(event, player: player, ad: ad, adBreak: adBreak) } + } + +} diff --git a/Sources/Core/InternalQueue/NetworkControllerIQWrapper.swift b/Sources/Core/InternalQueue/NetworkControllerIQWrapper.swift new file mode 100644 index 000000000..634f3b2f6 --- /dev/null +++ b/Sources/Core/InternalQueue/NetworkControllerIQWrapper.swift @@ -0,0 +1,46 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class NetworkControllerIQWrapper: NetworkController { + + private let controller: NetworkController + + init(controller: NetworkController) { + self.controller = controller + } + + // MARK: - Properties + + var endpoint: String? { + get { return InternalQueue.sync { controller.endpoint } } + set { InternalQueue.sync { controller.endpoint = newValue } } + } + + var method: HttpMethodOptions { + get { return InternalQueue.sync { controller.method } } + set { InternalQueue.sync { controller.method = newValue } } + } + + var customPostPath: String? { + get { return InternalQueue.sync { controller.customPostPath } } + set { InternalQueue.sync { controller.customPostPath = newValue } } + } + + var requestHeaders: [String : String]? { + get { return InternalQueue.sync { controller.requestHeaders } } + set { InternalQueue.sync { controller.requestHeaders = newValue } } + } + +} diff --git a/Sources/Core/InternalQueue/PluginsControllerIQWrapper.swift b/Sources/Core/InternalQueue/PluginsControllerIQWrapper.swift new file mode 100644 index 000000000..6611af1c9 --- /dev/null +++ b/Sources/Core/InternalQueue/PluginsControllerIQWrapper.swift @@ -0,0 +1,35 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class PluginsControllerIQWrapper: PluginsController { + + private let controller: PluginsController + + init(controller: PluginsController) { + self.controller = controller + } + + var identifiers: [String] { + return InternalQueue.sync { controller.identifiers } + } + + func add(plugin: PluginIdentifiable) { + InternalQueue.sync { controller.add(plugin: plugin) } + } + + func remove(identifier: String) { + InternalQueue.sync { controller.remove(identifier: identifier) } + } +} diff --git a/Sources/Core/InternalQueue/SessionControllerIQWrapper.swift b/Sources/Core/InternalQueue/SessionControllerIQWrapper.swift new file mode 100644 index 000000000..8529c9322 --- /dev/null +++ b/Sources/Core/InternalQueue/SessionControllerIQWrapper.swift @@ -0,0 +1,87 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class SessionControllerIQWrapper: SessionController { + + private let controller: SessionController + + init(controller: SessionController) { + self.controller = controller + } + + func pause() { + InternalQueue.sync { controller.pause() } + } + + func resume() { + InternalQueue.sync { controller.resume() } + } + + func startNewSession() { + InternalQueue.sync { controller.startNewSession() } + } + + // MARK: - Properties + + var foregroundTimeout: Measurement { + get { InternalQueue.sync { controller.foregroundTimeout } } + set { InternalQueue.sync { controller.foregroundTimeout = newValue } } + } + + var foregroundTimeoutInSeconds: Int { + get { InternalQueue.sync { controller.foregroundTimeoutInSeconds } } + set { InternalQueue.sync { controller.foregroundTimeoutInSeconds = newValue } } + } + + var backgroundTimeout: Measurement { + get { InternalQueue.sync { controller.backgroundTimeout } } + set { InternalQueue.sync { controller.backgroundTimeout = newValue } } + } + + var backgroundTimeoutInSeconds: Int { + get { InternalQueue.sync { controller.backgroundTimeoutInSeconds } } + set { InternalQueue.sync { controller.backgroundTimeoutInSeconds = newValue } } + } + + var onSessionStateUpdate: ((_ sessionState: SessionState) -> Void)? { + get { InternalQueue.sync { controller.onSessionStateUpdate } } + set { InternalQueue.sync { controller.onSessionStateUpdate = newValue } } + } + + var sessionIndex: Int { + InternalQueue.sync { controller.sessionIndex } + } + + var sessionId: String? { + InternalQueue.sync { controller.sessionId } + } + + var userId: String? { + InternalQueue.sync { controller.userId } + } + + var isInBackground: Bool { + InternalQueue.sync { controller.isInBackground } + } + + var backgroundIndex: Int { + InternalQueue.sync { controller.backgroundIndex } + } + + var foregroundIndex: Int { + InternalQueue.sync { controller.foregroundIndex } + } + +} diff --git a/Sources/Core/InternalQueue/SubjectControllerIQWrapper.swift b/Sources/Core/InternalQueue/SubjectControllerIQWrapper.swift new file mode 100644 index 000000000..b2c2a6a1d --- /dev/null +++ b/Sources/Core/InternalQueue/SubjectControllerIQWrapper.swift @@ -0,0 +1,118 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class SubjectControllerIQWrapper: SubjectController { + + private let controller: SubjectController + + init(controller: SubjectController) { + self.controller = controller + } + + // MARK: - Properties + + var userId: String? { + get { return InternalQueue.sync { controller.userId } } + set { InternalQueue.sync { controller.userId = newValue } } + } + + var networkUserId: String? { + get { return InternalQueue.sync { controller.networkUserId } } + set { InternalQueue.sync { controller.networkUserId = newValue } } + } + + var domainUserId: String? { + get { return InternalQueue.sync { controller.domainUserId } } + set { InternalQueue.sync { controller.domainUserId = newValue } } + } + + var useragent: String? { + get { return InternalQueue.sync { controller.useragent } } + set { InternalQueue.sync { controller.useragent = newValue } } + } + + var ipAddress: String? { + get { return InternalQueue.sync { controller.ipAddress } } + set { InternalQueue.sync { controller.ipAddress = newValue } } + } + + var timezone: String? { + get { return InternalQueue.sync { controller.timezone } } + set { InternalQueue.sync { controller.timezone = newValue } } + } + + var language: String? { + get { return InternalQueue.sync { controller.language } } + set { InternalQueue.sync { controller.language = newValue } } + } + + var screenResolution: SPSize? { + get { return InternalQueue.sync { controller.screenResolution } } + set { InternalQueue.sync { controller.screenResolution = newValue } } + } + + var screenViewPort: SPSize? { + get { return InternalQueue.sync { controller.screenViewPort } } + set { InternalQueue.sync { controller.screenViewPort = newValue } } + } + + var colorDepth: NSNumber? { + get { return InternalQueue.sync { controller.colorDepth } } + set { InternalQueue.sync { controller.colorDepth = newValue } } + } + + // MARK: - GeoLocalization + + var geoLatitude: NSNumber? { + get { return InternalQueue.sync { controller.geoLatitude } } + set { InternalQueue.sync { controller.geoLatitude = newValue } } + } + + var geoLongitude: NSNumber? { + get { return InternalQueue.sync { controller.geoLongitude } } + set { InternalQueue.sync { controller.geoLongitude = newValue } } + } + + var geoLatitudeLongitudeAccuracy: NSNumber? { + get { return InternalQueue.sync { controller.geoLatitudeLongitudeAccuracy } } + set { InternalQueue.sync { controller.geoLatitudeLongitudeAccuracy = newValue } } + } + + var geoAltitude: NSNumber? { + get { return InternalQueue.sync { controller.geoAltitude } } + set { InternalQueue.sync { controller.geoAltitude = newValue } } + } + + var geoAltitudeAccuracy: NSNumber? { + get { return InternalQueue.sync { controller.geoAltitudeAccuracy } } + set { InternalQueue.sync { controller.geoAltitudeAccuracy = newValue } } + } + + var geoSpeed: NSNumber? { + get { return InternalQueue.sync { controller.geoSpeed } } + set { InternalQueue.sync { controller.geoSpeed = newValue } } + } + + var geoBearing: NSNumber? { + get { return InternalQueue.sync { controller.geoBearing } } + set { InternalQueue.sync { controller.geoBearing = newValue } } + } + + var geoTimestamp: NSNumber? { + get { return InternalQueue.sync { controller.geoTimestamp } } + set { InternalQueue.sync { controller.geoTimestamp = newValue } } + } + +} diff --git a/Sources/Core/InternalQueue/TrackerControllerIQWrapper.swift b/Sources/Core/InternalQueue/TrackerControllerIQWrapper.swift new file mode 100644 index 000000000..7e3def3bd --- /dev/null +++ b/Sources/Core/InternalQueue/TrackerControllerIQWrapper.swift @@ -0,0 +1,228 @@ +// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. +// +// This program is licensed to you under the Apache License Version 2.0, +// and you may not use this file except in compliance with the Apache License +// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at +// http://www.apache.org/licenses/LICENSE-2.0. +// +// Unless required by applicable law or agreed to in writing, +// software distributed under the Apache License Version 2.0 is distributed on +// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either +// express or implied. See the Apache License Version 2.0 for the specific +// language governing permissions and limitations there under. + +import Foundation + +class TrackerControllerIQWrapper: TrackerController { + + private let controller: TrackerControllerImpl + + init(controller: TrackerControllerImpl) { + self.controller = controller + } + + // MARK: - Controllers + + var network: NetworkController? { + return InternalQueue.sync { + if let network = controller.network { + return NetworkControllerIQWrapper(controller: network) + } else { + return nil + } + } + } + + var emitter: EmitterController? { + return InternalQueue.sync { + if let emitter = controller.emitter { + return EmitterControllerIQWrapper(controller: emitter) + } else { + return nil + } + } + } + + var gdpr: GDPRController? { + return InternalQueue.sync { + if let gdpr = controller.gdpr { + return GDPRControllerIQWrapper(controller: gdpr) + } else { + return nil + } + } + } + + var globalContexts: GlobalContextsController? { + return InternalQueue.sync { + if let globalContexts = controller.globalContexts { + return GlobalContextsControllerIQWrapper(controller: globalContexts) + } else { + return nil + } + } + } + + var subject: SubjectController? { + return InternalQueue.sync { + if let subject = controller.subject { + return SubjectControllerIQWrapper(controller: subject) + } else { + return nil + } + } + } + + var session: SessionController? { + return InternalQueue.sync { + if let session = controller.session { + return SessionControllerIQWrapper(controller: session) + } else { + return nil + } + } + } + + var plugins: PluginsController { + return InternalQueue.sync { PluginsControllerIQWrapper(controller: controller.plugins) } + } + + var media: MediaController { + return InternalQueue.sync { MediaControllerIQWrapper(controller: controller.media) } + } + + var ecommerce: EcommerceController { + return InternalQueue.sync { EcommerceControllerIQWrapper(controller: controller.ecommerce) } + } + + // MARK: - Control methods + + func pause() { + InternalQueue.sync { controller.pause() } + } + + func resume() { + InternalQueue.sync { controller.resume() } + } + + func track(_ event: Event) -> UUID { + let eventId = UUID() + InternalQueue.async { self.controller.track(event, eventId: eventId) } + return eventId + } + + // MARK: - Properties' setters and getters + + var appId: String { + get { return InternalQueue.sync { controller.appId } } + set { InternalQueue.sync { controller.appId = newValue } } + } + + var namespace: String { + return InternalQueue.sync { controller.namespace } + } + + var devicePlatform: DevicePlatform { + get { return InternalQueue.sync { controller.devicePlatform } } + set { InternalQueue.sync { controller.devicePlatform = newValue } } + } + + var base64Encoding: Bool { + get { return InternalQueue.sync { controller.base64Encoding } } + set { InternalQueue.sync { controller.base64Encoding = newValue } } + } + + var logLevel: LogLevel { + get { return InternalQueue.sync { controller.logLevel } } + set { InternalQueue.sync { controller.logLevel = newValue } } + } + + var loggerDelegate: LoggerDelegate? { + get { return InternalQueue.sync { controller.loggerDelegate } } + set { InternalQueue.sync { controller.loggerDelegate = newValue } } + } + + var applicationContext: Bool { + get { return InternalQueue.sync { controller.applicationContext } } + set { InternalQueue.sync { controller.applicationContext = newValue } } + } + + var platformContext: Bool { + get { return InternalQueue.sync { controller.platformContext } } + set { InternalQueue.sync { controller.platformContext = newValue } } + } + + var platformContextProperties: [PlatformContextProperty]? { + get { return InternalQueue.sync { controller.platformContextProperties } } + set { InternalQueue.sync { controller.platformContextProperties = newValue } } + } + + var geoLocationContext: Bool { + get { return InternalQueue.sync { controller.geoLocationContext } } + set { InternalQueue.sync { controller.geoLocationContext = newValue } } + } + + var diagnosticAutotracking: Bool { + get { return InternalQueue.sync { controller.diagnosticAutotracking } } + set { InternalQueue.sync { controller.diagnosticAutotracking = newValue } } + } + + var exceptionAutotracking: Bool { + get { return InternalQueue.sync { controller.exceptionAutotracking } } + set { InternalQueue.sync { controller.exceptionAutotracking = newValue } } + } + + var installAutotracking: Bool { + get { return InternalQueue.sync { controller.installAutotracking } } + set { InternalQueue.sync { controller.installAutotracking = newValue } } + } + + var lifecycleAutotracking: Bool { + get { return InternalQueue.sync { controller.lifecycleAutotracking } } + set { InternalQueue.sync { controller.lifecycleAutotracking = newValue } } + } + + var deepLinkContext: Bool { + get { return InternalQueue.sync { controller.deepLinkContext } } + set { InternalQueue.sync { controller.deepLinkContext = newValue } } + } + + var screenContext: Bool { + get { return InternalQueue.sync { controller.screenContext } } + set { InternalQueue.sync { controller.screenContext = newValue } } + } + + var screenViewAutotracking: Bool { + get { return InternalQueue.sync { controller.screenViewAutotracking } } + set { InternalQueue.sync { controller.screenViewAutotracking = newValue } } + } + + var trackerVersionSuffix: String? { + get { return InternalQueue.sync { controller.trackerVersionSuffix } } + set { InternalQueue.sync { controller.trackerVersionSuffix = newValue } } + } + + var sessionContext: Bool { + get { return InternalQueue.sync { controller.sessionContext } } + set { InternalQueue.sync { controller.sessionContext = newValue } } + } + + var userAnonymisation: Bool { + get { return InternalQueue.sync { controller.userAnonymisation } } + set { InternalQueue.sync { controller.userAnonymisation = newValue } } + } + + var advertisingIdentifierRetriever: (() -> UUID?)? { + get { return InternalQueue.sync { controller.advertisingIdentifierRetriever } } + set { InternalQueue.sync { controller.advertisingIdentifierRetriever = newValue } } + } + + var isTracking: Bool { + return InternalQueue.sync { controller.isTracking } + } + + var version: String { + return InternalQueue.sync { controller.version } + } + +} diff --git a/Sources/Core/Media/Controllers/AVPlayerSubscription.swift b/Sources/Core/Media/Controllers/AVPlayerSubscription.swift index a5c319570..0591ab111 100644 --- a/Sources/Core/Media/Controllers/AVPlayerSubscription.swift +++ b/Sources/Core/Media/Controllers/AVPlayerSubscription.swift @@ -47,21 +47,23 @@ class AVPlayerSubscription { // add a playback rate observer to find out when the user plays or pauses the videos rateObserver = player.observe(\.rate, options: [.old, .new]) { [weak self] player, change in - guard let oldRate = change.oldValue else { return } - guard let newRate = change.newValue else { return } - - if oldRate != 0 && newRate == 0 { // paused - self?.lastPauseTime = player.currentTime() - self?.track(MediaPauseEvent()) - } else if oldRate == 0 && newRate != 0 { // started playing - // when the current time diverges significantly, i.e. more than 1 second, from what it was when last paused, track a seek event - if let lastPauseTime = self?.lastPauseTime { - if abs(player.currentTime().seconds - lastPauseTime.seconds) > 1 { - self?.track(MediaSeekEndEvent()) + InternalQueue.async { + guard let oldRate = change.oldValue else { return } + guard let newRate = change.newValue else { return } + + if oldRate != 0 && newRate == 0 { // paused + self?.lastPauseTime = player.currentTime() + self?.track(MediaPauseEvent()) + } else if oldRate == 0 && newRate != 0 { // started playing + // when the current time diverges significantly, i.e. more than 1 second, from what it was when last paused, track a seek event + if let lastPauseTime = self?.lastPauseTime { + if abs(player.currentTime().seconds - lastPauseTime.seconds) > 1 { + self?.track(MediaSeekEndEvent()) + } } + self?.lastPauseTime = nil + self?.track(MediaPlayEvent()) } - self?.lastPauseTime = nil - self?.track(MediaPlayEvent()) } } @@ -92,15 +94,17 @@ class AVPlayerSubscription { /// Handles notifications from the notification center subscriptions @objc private func handleNotification(_ notification: Notification) { - switch notification.name { - case .AVPlayerItemPlaybackStalled: - track(MediaBufferStartEvent()) - case .AVPlayerItemDidPlayToEndTime: - track(MediaEndEvent()) - case .AVPlayerItemFailedToPlayToEndTime: - track(MediaErrorEvent(errorDescription: player.error?.localizedDescription)) - default: - return + InternalQueue.async { + switch notification.name { + case .AVPlayerItemPlaybackStalled: + self.track(MediaBufferStartEvent()) + case .AVPlayerItemDidPlayToEndTime: + self.track(MediaEndEvent()) + case .AVPlayerItemFailedToPlayToEndTime: + self.track(MediaErrorEvent(errorDescription: self.player.error?.localizedDescription)) + default: + return + } } } @@ -135,7 +139,9 @@ class AVPlayerSubscription { positionObserverToken = player.addPeriodicTimeObserver(forInterval: interval, queue: .main) { [weak self] _ in - self?.update() + InternalQueue.async { + self?.update() + } } } diff --git a/Sources/Core/Media/Controllers/MediaPingInterval.swift b/Sources/Core/Media/Controllers/MediaPingInterval.swift index 03e8369b7..b57a7b7c0 100644 --- a/Sources/Core/Media/Controllers/MediaPingInterval.swift +++ b/Sources/Core/Media/Controllers/MediaPingInterval.swift @@ -16,8 +16,8 @@ import Foundation class MediaPingInterval { var pingInterval: Int - private var timer: Timer? - private var timerProvider: Timer.Type + private var timer: InternalQueueTimer? + private var startTimer: (TimeInterval, @escaping () -> Void) -> InternalQueueTimer private var paused: Bool? private var numPausedPings: Int = 0 private var maxPausedPings: Int = 1 @@ -25,12 +25,12 @@ class MediaPingInterval { init(pingInterval: Int? = nil, maxPausedPings: Int? = nil, - timerProvider: Timer.Type = Timer.self) { + startTimer: @escaping (TimeInterval, @escaping () -> Void) -> InternalQueueTimer = InternalQueue.startTimer) { if let maxPausedPings = maxPausedPings { self.maxPausedPings = maxPausedPings } self.pingInterval = pingInterval ?? 30 - self.timerProvider = timerProvider + self.startTimer = startTimer } func update(player: MediaPlayerEntity) { @@ -41,8 +41,8 @@ class MediaPingInterval { func subscribe(callback: @escaping () -> ()) { end() - timer = timerProvider.scheduledTimer(withTimeInterval: TimeInterval(pingInterval), - repeats: true) { _ in + timer = startTimer(TimeInterval(pingInterval)) { [weak self] in + guard let self = self else { return } if !self.isPaused || self.numPausedPings < self.maxPausedPings { if self.isPaused { self.numPausedPings += 1 diff --git a/Sources/Core/Media/Controllers/MediaTrackingImpl.swift b/Sources/Core/Media/Controllers/MediaTrackingImpl.swift index c71f06429..ce143f6dc 100644 --- a/Sources/Core/Media/Controllers/MediaTrackingImpl.swift +++ b/Sources/Core/Media/Controllers/MediaTrackingImpl.swift @@ -115,8 +115,6 @@ class MediaTrackingImpl: MediaTracking { player: MediaPlayerEntity? = nil, ad: MediaAdEntity? = nil, adBreak: MediaAdBreakEntity? = nil) { - objc_sync_enter(self) - // update state if let player = player { self.player.update(from: player) @@ -143,8 +141,6 @@ class MediaTrackingImpl: MediaTracking { if let event = event { adTracking.updateForNextEvent(event: event) } - - objc_sync_exit(self) } private func addEntitiesAndTrack(event: Event) { diff --git a/Sources/Core/Session/Session.swift b/Sources/Core/Session/Session.swift index 6c4197695..1f97d07b7 100644 --- a/Sources/Core/Session/Session.swift +++ b/Sources/Core/Session/Session.swift @@ -20,18 +20,12 @@ class Session { // MARK: - Private properties - private var _backgroundIndex = 0 - private var _backgroundTimeout = TrackerDefaults.backgroundTimeout private var dataPersistence: DataPersistence? /// The event index private var eventIndex = 0 - private var _foregroundIndex = 0 - private var _foregroundTimeout = TrackerDefaults.foregroundTimeout private var isNewSession = true private var isSessionCheckerEnabled = false - private var _inBackground = false private var lastSessionCheck: NSNumber = Utilities.getTimestamp() - private var _onSessionStateUpdate: ((_ sessionState: SessionState) -> Void)? /// Returns the current session state private var state: SessionState? /// The current tracker associated with the session @@ -42,30 +36,21 @@ class Session { /// The session's userId let userId: String /// Whether the application is in the background or foreground - var inBackground: Bool { return sync { self._inBackground } } + var inBackground: Bool = false /// The foreground index count - var foregroundIndex: Int { return sync { self._foregroundIndex } } + var foregroundIndex = 0 /// The background index count - var backgroundIndex: Int { return sync { self._backgroundIndex } } + var backgroundIndex = 0 /// Callback to be called when the session is updated - var onSessionStateUpdate: ((_ sessionState: SessionState) -> Void)? { - get { return sync { self._onSessionStateUpdate } } - set { sync { self._onSessionStateUpdate = newValue } } - } + var onSessionStateUpdate: ((_ sessionState: SessionState) -> Void)? /// The currently set Foreground Timeout in milliseconds - var foregroundTimeout: Int { - get { return sync { self._foregroundTimeout } } - set { sync { self._foregroundTimeout = newValue } } - } + var foregroundTimeout = TrackerDefaults.foregroundTimeout /// The currently set Background Timeout in milliseconds - var backgroundTimeout: Int { - get { return sync { self._backgroundTimeout } } - set { sync { self._backgroundTimeout = newValue } } - } - var sessionIndex: Int? { return sync { state?.sessionIndex } } - var sessionId: String? { return sync { state?.sessionId } } - var previousSessionId: String? { return sync { state?.previousSessionId } } - var firstEventId: String? { return sync { state?.firstEventId } } + var backgroundTimeout = TrackerDefaults.backgroundTimeout + var sessionIndex: Int? { return state?.sessionIndex } + var sessionId: String? { return state?.sessionId } + var previousSessionId: String? { return state?.previousSessionId } + var firstEventId: String? { return state?.firstEventId } // MARK: - Constructor and destructor @@ -77,8 +62,8 @@ class Session { /// - Returns: a SnowplowSession init(foregroundTimeout: Int, backgroundTimeout: Int, trackerNamespace: String? = nil, tracker: Tracker? = nil) { - self._foregroundTimeout = foregroundTimeout * 1000 - self._backgroundTimeout = backgroundTimeout * 1000 + self.foregroundTimeout = foregroundTimeout * 1000 + self.backgroundTimeout = backgroundTimeout * 1000 self.tracker = tracker if let namespace = trackerNamespace { dataPersistence = DataPersistence.getFor(namespace: namespace) @@ -122,18 +107,18 @@ class Session { /// Starts the recurring timer check for sessions func startChecker() { - sync { isSessionCheckerEnabled = true } + isSessionCheckerEnabled = true } /// Stops the recurring timer check for sessions func stopChecker() { - sync { isSessionCheckerEnabled = false } + isSessionCheckerEnabled = false } /// Expires the current session and starts a new one func startNewSession() { // TODO: when the sesssion has been renewed programmatically, it has to be reported in the session context to the collector. - sync { isNewSession = true } + isNewSession = true } /// Returns the session dictionary @@ -145,24 +130,22 @@ class Session { func getDictWithEventId(_ eventId: String?, eventTimestamp: Int64, userAnonymisation: Bool) -> [String : Any]? { var context: [String : Any]? = nil - sync { - if isSessionCheckerEnabled { - if shouldUpdate() { - update(eventId: eventId, eventTimestamp: eventTimestamp) - if let onSessionStateUpdate = _onSessionStateUpdate, let state = state { - DispatchQueue.global(qos: .default).async { - onSessionStateUpdate(state) - } + if isSessionCheckerEnabled { + if shouldUpdate() { + update(eventId: eventId, eventTimestamp: eventTimestamp) + if let onSessionStateUpdate = onSessionStateUpdate, let state = state { + DispatchQueue.global(qos: .default).async { + onSessionStateUpdate(state) } } - lastSessionCheck = Utilities.getTimestamp() } - - eventIndex += 1 - - context = state?.sessionContext - context?[kSPSessionEventIndex] = NSNumber(value: eventIndex) + lastSessionCheck = Utilities.getTimestamp() } + + eventIndex += 1 + + context = state?.sessionContext + context?[kSPSessionEventIndex] = NSNumber(value: eventIndex) if userAnonymisation { // mask the user identifier @@ -203,7 +186,7 @@ class Session { } let lastAccess = lastSessionCheck.int64Value let now = Utilities.getTimestamp().int64Value - let timeout = _inBackground ? _backgroundTimeout : _foregroundTimeout + let timeout = inBackground ? backgroundTimeout : foregroundTimeout return now < lastAccess || Int(now - lastAccess) > timeout } @@ -234,56 +217,34 @@ class Session { // MARK: - background and foreground notifications @objc func updateInBackground() { - backgroundUpdateSync { - if tracker?.lifecycleEvents ?? false { + InternalQueue.async { + if self.tracker?.lifecycleEvents ?? false { guard let backgroundIndex = self.incrementBackgroundIndexIfNotInBackground() else { return } - _ = self.tracker?.track(Background(index: backgroundIndex), synchronous: true) - sync { self._inBackground = true } + _ = self.tracker?.track(Background(index: backgroundIndex)) + self.inBackground = true } } } @objc func updateInForeground() { - backgroundUpdateSync { - if tracker?.lifecycleEvents ?? false { + InternalQueue.async { + if self.tracker?.lifecycleEvents ?? false { guard let foregroundIndex = self.incrementForegroundIndexIfInBackground() else { return } - _ = self.tracker?.track(Foreground(index: foregroundIndex), synchronous: true) - sync { self._inBackground = false } + _ = self.tracker?.track(Foreground(index: foregroundIndex)) + self.inBackground = false } } } private func incrementBackgroundIndexIfNotInBackground() -> Int? { - return sync { - if self._inBackground { return nil } - self._backgroundIndex += 1 - return self._backgroundIndex - } + if self.inBackground { return nil } + self.backgroundIndex += 1 + return self.backgroundIndex } private func incrementForegroundIndexIfInBackground() -> Int? { - return sync { - if !self._inBackground { return nil } - self._foregroundIndex += 1 - return self._foregroundIndex - } - } - - // MARK: - Dispatch queues - - private let dispatchQueue = DispatchQueue(label: "snowplow.session") - - private func sync(_ callback: () -> T) -> T { - dispatchPrecondition(condition: .notOnQueue(dispatchQueue)) - - return dispatchQueue.sync(execute: callback) - } - - private let backgroundUpdateQueue = DispatchQueue(label: "snowplow.session.background") - - private func backgroundUpdateSync(_ callback: () -> T) -> T { - dispatchPrecondition(condition: .notOnQueue(backgroundUpdateQueue)) - - return backgroundUpdateQueue.sync(execute: callback) + if !self.inBackground { return nil } + self.foregroundIndex += 1 + return self.foregroundIndex } } diff --git a/Sources/Core/StateMachine/StateFuture.swift b/Sources/Core/StateMachine/StateFuture.swift index c6d9f73ff..81a109b20 100644 --- a/Sources/Core/StateMachine/StateFuture.swift +++ b/Sources/Core/StateMachine/StateFuture.swift @@ -31,8 +31,6 @@ class StateFuture { } func computeState() -> State? { - objc_sync_enter(self) - defer { objc_sync_exit(self) } if computedState == nil { if let stateMachine = stateMachine, let event = event { computedState = stateMachine.transition(from: event, state: previousState?.computeState()) diff --git a/Sources/Core/StateMachine/StateManager.swift b/Sources/Core/StateMachine/StateManager.swift index 4e5e910b0..b4b911d5d 100644 --- a/Sources/Core/StateMachine/StateManager.swift +++ b/Sources/Core/StateMachine/StateManager.swift @@ -23,9 +23,6 @@ class StateManager { private var trackerState = TrackerState() func addOrReplaceStateMachine(_ stateMachine: StateMachineProtocol) { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - if let previousStateMachine = identifierToStateMachine[stateMachine.identifier] { if type(of: stateMachine) == type(of: previousStateMachine) { return @@ -56,9 +53,6 @@ class StateManager { } func removeStateMachine(_ stateMachineIdentifier: String) -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - guard let stateMachine = identifierToStateMachine[stateMachineIdentifier] else { return false } @@ -88,9 +82,6 @@ class StateManager { } func trackerState(forProcessedEvent event: Event) -> TrackerStateSnapshot? { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - if let sdEvent = event as? SelfDescribingAbstract { var stateMachines = Array(eventSchemaToStateMachine[sdEvent.schema] ?? []) stateMachines.append(contentsOf: eventSchemaToStateMachine["*"] ?? []) @@ -121,9 +112,6 @@ class StateManager { } func filter(event: InspectableEvent & StateMachineEvent) -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - guard let schema = event.schema ?? event.eventName else { return true } var stateMachines = eventSchemaToFilter[schema] ?? [] stateMachines.append(contentsOf: eventSchemaToFilter["*"] ?? []) @@ -138,9 +126,6 @@ class StateManager { } func entities(forProcessedEvent event: InspectableEvent & StateMachineEvent) -> [SelfDescribingJson] { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - guard let schema = event.schema ?? event.eventName else { return [] } var result: [SelfDescribingJson] = [] var stateMachines = eventSchemaToEntitiesGenerator[schema] ?? [] @@ -156,9 +141,6 @@ class StateManager { } func addPayloadValues(to event: InspectableEvent & StateMachineEvent) -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - guard let schema = event.schema else { return true } var failures = 0 var stateMachines = eventSchemaToPayloadUpdater[schema] ?? [] @@ -177,10 +159,8 @@ class StateManager { func afterTrack(event: InspectableEvent & StateMachineEvent) { guard let schema = event.schema ?? event.eventName else { return } - objc_sync_enter(self) var stateMachines = eventSchemaToAfterTrackCallback[schema] ?? [] stateMachines.append(contentsOf: eventSchemaToAfterTrackCallback["*"] ?? []) - objc_sync_exit(self) if !stateMachines.isEmpty { DispatchQueue.global(qos: .default).async { diff --git a/Sources/Core/StateMachine/TrackerState.swift b/Sources/Core/StateMachine/TrackerState.swift index 3157260d8..611e04d4b 100644 --- a/Sources/Core/StateMachine/TrackerState.swift +++ b/Sources/Core/StateMachine/TrackerState.swift @@ -19,29 +19,20 @@ class TrackerState: TrackerStateSnapshot { /// Set a future computable state with a specific state identifier func setStateFuture(_ state: StateFuture, identifier stateIdentifier: String) { - objc_sync_enter(self) trackerState[stateIdentifier] = state - objc_sync_exit(self) } /// Get a future computable state associated with a state identifier func stateFuture(withIdentifier stateIdentifier: String) -> StateFuture? { - objc_sync_enter(self) - defer { objc_sync_exit(self) } return trackerState[stateIdentifier] } func remove(withIdentifier stateIdentifer: String) { - objc_sync_enter(self) trackerState.removeValue(forKey: stateIdentifer) - objc_sync_exit(self) } /// Get an immutable copy of the whole tracker state func snapshot() -> TrackerStateSnapshot? { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - let newTrackerState = TrackerState() newTrackerState.trackerState = trackerState return newTrackerState diff --git a/Sources/Core/Storage/MemoryEventStore.swift b/Sources/Core/Storage/MemoryEventStore.swift index 1e0f569b7..46a26c763 100644 --- a/Sources/Core/Storage/MemoryEventStore.swift +++ b/Sources/Core/Storage/MemoryEventStore.swift @@ -19,7 +19,6 @@ class MemoryEventStore: NSObject, EventStore { var index: Int64 var orderedSet: NSMutableOrderedSet - convenience override init() { self.init(limit: 250) } @@ -33,22 +32,16 @@ class MemoryEventStore: NSObject, EventStore { // Interface methods func addEvent(_ payload: Payload) { - objc_sync_enter(self) let item = EmitterEvent(payload: payload, storeId: index) orderedSet.add(item) - objc_sync_exit(self) index += 1 } func count() -> UInt { - objc_sync_enter(self) - defer { objc_sync_exit(self) } return UInt(orderedSet.count) } func emittableEvents(withQueryLimit queryLimit: UInt) -> [EmitterEvent] { - objc_sync_enter(self) - defer { objc_sync_exit(self) } let setCount = (orderedSet).count if setCount <= 0 { return [] @@ -71,8 +64,6 @@ class MemoryEventStore: NSObject, EventStore { } func removeAllEvents() -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } orderedSet.removeAllObjects() return true } @@ -82,8 +73,6 @@ class MemoryEventStore: NSObject, EventStore { } func removeEvents(withIds storeIds: [Int64]) -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } var itemsToRemove: [EmitterEvent] = [] for item in orderedSet { guard let item = item as? EmitterEvent else { diff --git a/Sources/Core/Subject/PlatformContext.swift b/Sources/Core/Subject/PlatformContext.swift index db59ecd39..2b8b0bbee 100644 --- a/Sources/Core/Subject/PlatformContext.swift +++ b/Sources/Core/Subject/PlatformContext.swift @@ -54,7 +54,6 @@ class PlatformContext { /// - Parameter userAnonymisation: Whether to anonymise user identifiers (IDFA values) func fetchPlatformDict(userAnonymisation: Bool, advertisingIdentifierRetriever: (() -> UUID?)?) -> Payload { #if os(iOS) - objc_sync_enter(self) let now = Date().timeIntervalSince1970 if now - lastUpdatedEphemeralMobileDict >= mobileDictUpdateFrequency { setEphemeralMobileDict() @@ -62,7 +61,6 @@ class PlatformContext { if now - lastUpdatedEphemeralNetworkDict >= networkDictUpdateFrequency { setEphemeralNetworkDict() } - objc_sync_exit(self) #endif if userAnonymisation { // mask user identifiers diff --git a/Sources/Core/Subject/Subject.swift b/Sources/Core/Subject/Subject.swift index 8af407a2d..610e4ee10 100644 --- a/Sources/Core/Subject/Subject.swift +++ b/Sources/Core/Subject/Subject.swift @@ -20,22 +20,14 @@ class Subject : NSObject { private var platformContextManager: PlatformContext private var geoDict: [String : NSObject] = [:] - private var _platformContext = false - var platformContext: Bool { - get { return sync { _platformContext } } - set { sync { _platformContext = newValue } } - } + var platformContext = false var platformContextProperties: [PlatformContextProperty]? { - get { return sync { platformContextManager.platformContextProperties } } - set { sync { platformContextManager.platformContextProperties = newValue } } + get { return platformContextManager.platformContextProperties } + set { platformContextManager.platformContextProperties = newValue } } - var _geoLocationContext = false - var geoLocationContext: Bool { - get { return sync { _geoLocationContext } } - set { sync { _geoLocationContext = newValue } } - } + var geoLocationContext = false // MARK: - Standard Dictionary @@ -44,127 +36,107 @@ class Subject : NSObject { private var _userId: String? /// The user's ID. var userId: String? { - get { return sync { _userId } } + get { return _userId } set(uid) { - sync { - _userId = uid - standardDict[kSPUid] = uid - } + _userId = uid + standardDict[kSPUid] = uid } } private var _networkUserId: String? var networkUserId: String? { - get { return sync { _networkUserId } } + get { return _networkUserId } set(nuid) { - sync { - _networkUserId = nuid - standardDict[kSPNetworkUid] = nuid - } + _networkUserId = nuid + standardDict[kSPNetworkUid] = nuid } } private var _domainUserId: String? /// The domain UID. var domainUserId: String? { - get { return sync { _domainUserId } } + get { return _domainUserId } set(duid) { - sync { - _domainUserId = duid - standardDict[kSPDomainUid] = duid - } + _domainUserId = duid + standardDict[kSPDomainUid] = duid } } private var _useragent: String? /// The user agent (also known as browser string). var useragent: String? { - get { return sync { _useragent } } + get { return _useragent } set(useragent) { - sync { - _useragent = useragent - standardDict[kSPUseragent] = useragent - } + _useragent = useragent + standardDict[kSPUseragent] = useragent } } private var _ipAddress: String? /// The user's IP address. var ipAddress: String? { - get { return sync { _ipAddress } } + get { return _ipAddress } set(ip) { - sync { - _ipAddress = ip - standardDict[kSPIpAddress] = ip - } + _ipAddress = ip + standardDict[kSPIpAddress] = ip } } private var _timezone: String? /// The user's timezone. var timezone: String? { - get { return sync { _timezone } } + get { return _timezone } set(timezone) { - sync { - _timezone = timezone - standardDict[kSPTimezone] = timezone - } + _timezone = timezone + standardDict[kSPTimezone] = timezone } } private var _language: String? /// The user's language. var language: String? { - get { return sync { _language } } + get { return _language } set(lang) { - sync { - _language = lang - standardDict[kSPLanguage] = lang - } + _language = lang + standardDict[kSPLanguage] = lang } } private var _colorDepth: NSNumber? /// The user's color depth. var colorDepth: NSNumber? { - get { return sync { _colorDepth } } + get { return _colorDepth } set(depth) { - sync { - _colorDepth = depth - let res = "\(depth?.stringValue ?? "")" - standardDict[kSPColorDepth] = res - } + _colorDepth = depth + let res = "\(depth?.stringValue ?? "")" + standardDict[kSPColorDepth] = res } } var _screenResolution: SPSize? var screenResolution: SPSize? { - get { return sync { _screenResolution } } + get { return _screenResolution } set { - sync { - _screenResolution = newValue - if let size = newValue { - let res = "\((NSNumber(value: size.width)).stringValue)x\((NSNumber(value: size.height)).stringValue)" - standardDict[kSPResolution] = res - } else { - standardDict.removeValue(forKey: kSPResolution) - } + _screenResolution = newValue + if let size = newValue { + let res = "\((NSNumber(value: size.width)).stringValue)x\((NSNumber(value: size.height)).stringValue)" + standardDict[kSPResolution] = res + } else { + standardDict.removeValue(forKey: kSPResolution) } } } var _screenViewPort: SPSize? var screenViewPort: SPSize? { - get { return sync { _screenViewPort } } + get { return _screenViewPort } set { - sync { - _screenViewPort = newValue - if let size = newValue { - let res = "\((NSNumber(value: size.width)).stringValue)x\((NSNumber(value: size.height)).stringValue)" - standardDict[kSPViewPort] = res - } else { - standardDict.removeValue(forKey: kSPViewPort) - } + _screenViewPort = newValue + if let size = newValue { + let res = "\((NSNumber(value: size.width)).stringValue)x\((NSNumber(value: size.height)).stringValue)" + standardDict[kSPViewPort] = res + } else { + standardDict.removeValue(forKey: kSPViewPort) } } } @@ -175,49 +147,49 @@ class Subject : NSObject { /// Latitude value for the geolocation context. var geoLatitude: NSNumber? { - get { return sync { geoDict[kSPGeoLatitude] as? NSNumber } } - set(latitude) { sync { geoDict[kSPGeoLatitude] = latitude } } + get { return geoDict[kSPGeoLatitude] as? NSNumber } + set(latitude) { geoDict[kSPGeoLatitude] = latitude } } /// Longitude value for the geo context. var geoLongitude: NSNumber? { - get { return sync { geoDict[kSPGeoLongitude] as? NSNumber } } - set(longitude) { sync { geoDict[kSPGeoLongitude] = longitude } } + get { return geoDict[kSPGeoLongitude] as? NSNumber } + set(longitude) { geoDict[kSPGeoLongitude] = longitude } } /// LatitudeLongitudeAccuracy value for the geolocation context. var geoLatitudeLongitudeAccuracy: NSNumber? { - get { return sync { geoDict[kSPGeoLatLongAccuracy] as? NSNumber } } - set { sync { geoDict[kSPGeoLatLongAccuracy] = newValue } } + get { return geoDict[kSPGeoLatLongAccuracy] as? NSNumber } + set { geoDict[kSPGeoLatLongAccuracy] = newValue } } /// Altitude value for the geolocation context. var geoAltitude: NSNumber? { - get { return sync { geoDict[kSPGeoAltitude] as? NSNumber } } - set(altitude) { sync { geoDict[kSPGeoAltitude] = altitude } } + get { return geoDict[kSPGeoAltitude] as? NSNumber } + set(altitude) { geoDict[kSPGeoAltitude] = altitude } } /// AltitudeAccuracy value for the geolocation context. var geoAltitudeAccuracy: NSNumber? { - get { return sync { geoDict[kSPGeoAltitudeAccuracy] as? NSNumber } } - set(altitudeAccuracy) { sync { geoDict[kSPGeoAltitudeAccuracy] = altitudeAccuracy } } + get { return geoDict[kSPGeoAltitudeAccuracy] as? NSNumber } + set(altitudeAccuracy) { geoDict[kSPGeoAltitudeAccuracy] = altitudeAccuracy } } var geoBearing: NSNumber? { - get { return sync { geoDict[kSPGeoBearing] as? NSNumber } } - set(bearing) { sync { geoDict[kSPGeoBearing] = bearing } } + get { return geoDict[kSPGeoBearing] as? NSNumber } + set(bearing) { geoDict[kSPGeoBearing] = bearing } } /// Speed value for the geolocation context. var geoSpeed: NSNumber? { - get { return sync { geoDict[kSPGeoSpeed] as? NSNumber } } - set(speed) { sync { geoDict[kSPGeoSpeed] = speed } } + get { return geoDict[kSPGeoSpeed] as? NSNumber } + set(speed) { geoDict[kSPGeoSpeed] = speed } } /// Timestamp value for the geolocation context. var geoTimestamp: NSNumber? { - get { return sync { geoDict[kSPGeoTimestamp] as? NSNumber } } - set(timestamp) { sync { geoDict[kSPGeoTimestamp] = timestamp } } + get { return geoDict[kSPGeoTimestamp] as? NSNumber } + set(timestamp) { geoDict[kSPGeoTimestamp] = timestamp } } init(platformContext: Bool = false, @@ -227,8 +199,8 @@ class Subject : NSObject { self.platformContextManager = PlatformContext(platformContextProperties: platformContextProperties) super.init() platformContextManager.platformContextProperties = platformContextProperties - _platformContext = platformContext - _geoLocationContext = geoContext + self.platformContext = platformContext + self.geoLocationContext = geoContext screenResolution = Utilities.resolution screenViewPort = Utilities.viewPort @@ -266,56 +238,43 @@ class Subject : NSObject { //#pragma clang diagnostic pop func standardDict(userAnonymisation: Bool) -> [String : String] { - var copy = sync { self.standardDict } if userAnonymisation { + var copy = self.standardDict copy.removeValue(forKey: kSPUid) copy.removeValue(forKey: kSPDomainUid) copy.removeValue(forKey: kSPNetworkUid) copy.removeValue(forKey: kSPIpAddress) + return copy } - return copy + return self.standardDict } /// Gets all platform dictionary pairs to decorate event with. Returns nil if not enabled. /// - Parameter userAnonymisation: Whether to anonymise user identifiers /// - Returns: A SPPayload with all platform specific pairs. func platformDict(userAnonymisation: Bool, advertisingIdentifierRetriever: (() -> UUID?)?) -> Payload? { - return sync { - if _platformContext { - return platformContextManager.fetchPlatformDict( - userAnonymisation: userAnonymisation, - advertisingIdentifierRetriever: advertisingIdentifierRetriever) - } else { - return nil - } + if platformContext { + return platformContextManager.fetchPlatformDict( + userAnonymisation: userAnonymisation, + advertisingIdentifierRetriever: advertisingIdentifierRetriever) + } else { + return nil } } /// Gets the geolocation dictionary if the required keys are available. Returns nil if not enabled. /// - Returns: A dictionary with key-value pairs of the geolocation context. public var geoLocationDict: [String : NSObject]? { - return sync { - if _geoLocationContext { - if geoDict[kSPGeoLatitude] != nil && geoDict[kSPGeoLongitude] != nil { - return geoDict - } else { - logDebug(message: "GeoLocation missing required fields; cannot get.") - return nil - } + if geoLocationContext { + if geoDict[kSPGeoLatitude] != nil && geoDict[kSPGeoLongitude] != nil { + return geoDict } else { + logDebug(message: "GeoLocation missing required fields; cannot get.") return nil } + } else { + return nil } } - // MARK: - Dispatch queue - - private let dispatchQueue = DispatchQueue(label: "snowplow.subject") - - private func sync(_ callback: () -> T) -> T { - dispatchPrecondition(condition: .notOnQueue(dispatchQueue)) - - return dispatchQueue.sync(execute: callback) - } - } diff --git a/Sources/Core/Tracker/Tracker.swift b/Sources/Core/Tracker/Tracker.swift index e6b6facd4..ffe2e2c28 100644 --- a/Sources/Core/Tracker/Tracker.swift +++ b/Sources/Core/Tracker/Tracker.swift @@ -37,283 +37,261 @@ func uncaughtExceptionHandler(_ exception: NSException) { } /// This class is used for tracking events, and delegates them to other classes responsible for sending, storage, etc. -class Tracker { - - // MARK: - Private properties - +class Tracker: NSObject { + private var platformContextSchema: String = "" private var dataCollection = true private var builderFinished = false - private let trackerData: TrackerData - + /// The object used for sessionization, i.e. it characterizes user activity. - private(set) var session: Session? { - get { return sync { self.trackerData.session } } - set { sync { self.trackerData.session = newValue } } - } + private(set) var session: Session? /// Previous screen view state. - private(set) var previousScreenState: ScreenState? { - get { return sync { self.trackerData.previousScreenState } } - set { sync { self.trackerData.previousScreenState = newValue } } - } + private(set) var previousScreenState: ScreenState? /// Current screen view state. - private(set) var currentScreenState: ScreenState? { - get { return sync { self.trackerData.currentScreenState } } - set { sync { self.trackerData.currentScreenState = newValue } } + private(set) var currentScreenState: ScreenState? + + private func trackerPayloadData() -> [String : String] { + var trackerVersion = kSPVersion + if trackerVersionSuffix.count != 0 { + var allowedCharSet = CharacterSet.alphanumerics + allowedCharSet.formUnion(CharacterSet(charactersIn: ".-")) + let suffix = trackerVersionSuffix.components(separatedBy: allowedCharSet.inverted).joined(separator: "") + if suffix.count != 0 { + trackerVersion = "\(trackerVersion) \(suffix)" + } + } + return [ + kSPTrackerVersion: trackerVersion, + kSPNamespace: trackerNamespace, + kSPAppId: appId + ] } - - private let stateManager = StateManager() - private let _emitter: Emitter - - // MARK: - Properties - + + // MARK: - Setter + /// The emitter used to send events. - var emitter: Emitter { - return sync { self._emitter } - } - + let emitter: Emitter + /// The subject used to represent the current user and persist user information. - var subject: Subject? { - get { return sync { self.trackerData.subject } } - set { sync { self.trackerData.subject = newValue } } - } + var subject: Subject? /// Whether to use Base64 encoding for events. - var base64Encoded: Bool { - get { return sync { self.trackerData.base64Encoded } } - set { sync { self.trackerData.base64Encoded = newValue } } - } + var base64Encoded = TrackerDefaults.base64Encoded /// A unique identifier for an application. - var appId: String { - get { return sync { self.trackerData.appId } } - set { sync { self.trackerData.appId = newValue } } - } + var appId: String /// The identifier for the current tracker. - var trackerNamespace: String { - get { return sync { self.trackerData.trackerNamespace } } - } + let trackerNamespace: String /// Version suffix for tracker wrappers. - var trackerVersionSuffix: String { - get { return sync { self.trackerData.trackerVersionSuffix } } - set { sync { self.trackerData.trackerVersionSuffix = newValue } } - } - - var devicePlatform: DevicePlatform { - get { return sync { self.trackerData.devicePlatform } } - set { sync { self.trackerData.devicePlatform = newValue } } - } + var trackerVersionSuffix: String = TrackerDefaults.trackerVersionSuffix + var devicePlatform: DevicePlatform = TrackerDefaults.devicePlatform + var logLevel: LogLevel { - get { return sync { self.trackerData.logLevel } } - set { sync { self.trackerData.logLevel = newValue } } + get { + return Logger.logLevel + } + set { + Logger.logLevel = newValue + } } - + var loggerDelegate: LoggerDelegate? { - get { return sync { self.trackerData.loggerDelegate } } - set { sync { self.trackerData.loggerDelegate = newValue } } + get { + return Logger.delegate + } + set(delegate) { + Logger.delegate = delegate + } } + private var _sessionContext = false var sessionContext: Bool { - get { return sync { self.trackerData.sessionContext } } + get { + return _sessionContext + } set(sessionContext) { - sync { - self.trackerData.sessionContext = sessionContext - if self.trackerData.session != nil && !self.trackerData.sessionContext { - self.trackerData.session?.stopChecker() - self.trackerData.session = nil - } else if self.builderFinished && self.trackerData.session == nil && sessionContext { - self.trackerData.session = Session( - foregroundTimeout: self.trackerData.foregroundTimeout, - backgroundTimeout: self.trackerData.backgroundTimeout, - trackerNamespace: self.trackerData.trackerNamespace, - tracker: self - ) - } + _sessionContext = sessionContext + if session != nil && !sessionContext { + session?.stopChecker() + session = nil + } else if builderFinished && session == nil && sessionContext { + session = Session( + foregroundTimeout: foregroundTimeout, + backgroundTimeout: backgroundTimeout, + trackerNamespace: trackerNamespace, + tracker: self) } } } + private var _deepLinkContext = false var deepLinkContext: Bool { - get { sync { return self.trackerData.deepLinkContext } } + get { + return _deepLinkContext + } set(deepLinkContext) { - sync { - self.trackerData.deepLinkContext = deepLinkContext - if deepLinkContext { - self.stateManager.addOrReplaceStateMachine(DeepLinkStateMachine()) - } else { - _ = self.stateManager.removeStateMachine(DeepLinkStateMachine.identifier) - } + self._deepLinkContext = deepLinkContext + if deepLinkContext { + self.addOrReplace(stateMachine: DeepLinkStateMachine()) + } else { + _ = self.stateManager.removeStateMachine(DeepLinkStateMachine.identifier) } } } + private var _screenContext = false var screenContext: Bool { - get { return sync { self.trackerData.screenContext } } + get { + return _screenContext + } set(screenContext) { - sync { - self.trackerData.screenContext = screenContext - if screenContext { - self.stateManager.addOrReplaceStateMachine(ScreenStateMachine()) - } else { - _ = self.stateManager.removeStateMachine(ScreenStateMachine.identifier) - } + self._screenContext = screenContext + if screenContext { + self.addOrReplace(stateMachine: ScreenStateMachine()) + } else { + _ = self.stateManager.removeStateMachine(ScreenStateMachine.identifier) } } } - var applicationContext: Bool { - get { return sync { self.trackerData.applicationContext } } - set { sync { self.trackerData.applicationContext = newValue } } - } + var applicationContext = TrackerDefaults.applicationContext - var autotrackScreenViews: Bool { - get { return sync { self.trackerData.autotrackScreenViews } } - set { sync { self.trackerData.autotrackScreenViews = newValue } } - } + var autotrackScreenViews = TrackerDefaults.autotrackScreenViews + private var _foregroundTimeout = TrackerDefaults.foregroundTimeout var foregroundTimeout: Int { - get { return sync { self.trackerData.foregroundTimeout } } + get { + return _foregroundTimeout + } set(foregroundTimeout) { - sync { - self.trackerData.foregroundTimeout = foregroundTimeout - if self.builderFinished { - self.trackerData.session?.foregroundTimeout = foregroundTimeout - } + _foregroundTimeout = foregroundTimeout + if builderFinished && session != nil { + session?.foregroundTimeout = foregroundTimeout } } } + private var _backgroundTimeout = TrackerDefaults.backgroundTimeout var backgroundTimeout: Int { - get { return sync { self.trackerData.backgroundTimeout } } + get { + return _backgroundTimeout + } set(backgroundTimeout) { - sync { - self.trackerData.backgroundTimeout = backgroundTimeout - if self.builderFinished { - self.trackerData.session?.backgroundTimeout = backgroundTimeout - } + _backgroundTimeout = backgroundTimeout + if builderFinished && session != nil { + session?.backgroundTimeout = backgroundTimeout } } } + private var _lifecycleEvents = false /// Returns whether lifecyle events is enabled. /// - Returns: Whether background and foreground events are sent. var lifecycleEvents: Bool { - get { return sync { self.trackerData.lifecycleEvents } } + get { + return _lifecycleEvents + } set(lifecycleEvents) { - sync { - self.trackerData.lifecycleEvents = lifecycleEvents - if lifecycleEvents { - self.stateManager.addOrReplaceStateMachine(LifecycleStateMachine()) - } else { - _ = self.stateManager.removeStateMachine(LifecycleStateMachine.identifier) - } + self._lifecycleEvents = lifecycleEvents + if lifecycleEvents { + self.addOrReplace(stateMachine: LifecycleStateMachine()) + } else { + _ = self.stateManager.removeStateMachine(LifecycleStateMachine.identifier) } } } - var exceptionEvents: Bool { - get { return sync { self.trackerData.exceptionEvents } } - set { sync { self.trackerData.exceptionEvents = newValue } } - } - var installEvent: Bool { - get { return sync { self.trackerData.installEvent } } - set { sync { self.trackerData.installEvent = newValue } } - } - var trackerDiagnostic: Bool { - get { return sync { self.trackerData.trackerDiagnostic } } - set { sync { self.trackerData.trackerDiagnostic = newValue } } - } + var exceptionEvents = TrackerDefaults.exceptionEvents + var installEvent = TrackerDefaults.installEvent + var trackerDiagnostic = TrackerDefaults.trackerDiagnostic + private var _userAnonymisation = TrackerDefaults.userAnonymisation var userAnonymisation: Bool { - get { return sync { self.trackerData.userAnonymisation } } + get { + return _userAnonymisation + } set(userAnonymisation) { - sync { - if self.trackerData.userAnonymisation != userAnonymisation { - self.trackerData.userAnonymisation = userAnonymisation - if self.builderFinished { - self.trackerData.session?.startNewSession() - } - } + if _userAnonymisation != userAnonymisation { + _userAnonymisation = userAnonymisation + if let session = session { session.startNewSession() } } } } - + /// GDPR context /// You can enable or disable the context by setting this property - var gdprContext: GDPRContext? { - get { return sync { self.trackerData.gdprContext } } - set { sync { self.trackerData.gdprContext = newValue } } - } + var gdprContext: GDPRContext? + private var stateManager = StateManager() + var inBackground: Bool { - return sync { self.trackerData.session?.inBackground ?? false } + return session?.inBackground ?? false } - + var isTracking: Bool { - return sync { self.dataCollection } + return dataCollection } - var advertisingIdentifierRetriever: (() -> UUID?)? { - get { return sync { self.trackerData.advertisingIdentifierRetriever } } - set { sync { self.trackerData.advertisingIdentifierRetriever = newValue } } - } - - // MARK: - Constructor and destructor and related functions - + var advertisingIdentifierRetriever: (() -> UUID?)? + init(trackerNamespace: String, appId: String?, emitter: Emitter, builder: ((Tracker) -> (Void))) { - self.trackerData = TrackerData(appId: appId ?? "", trackerNamespace: trackerNamespace) - self._emitter = emitter + self.emitter = emitter + self.appId = appId ?? "" + self.trackerNamespace = trackerNamespace + super.init() builder(self) + #if os(iOS) + platformContextSchema = kSPMobileContextSchema + #else + platformContextSchema = kSPDesktopContextSchema + #endif + setup() checkInstall() } - - deinit { - NotificationCenter.default.removeObserver(self) - } - + private func setup() { if sessionContext { - self.trackerData.session = Session( - foregroundTimeout: self.trackerData.foregroundTimeout, - backgroundTimeout: self.trackerData.backgroundTimeout, - trackerNamespace: self.trackerData.trackerNamespace, + session = Session( + foregroundTimeout: foregroundTimeout, + backgroundTimeout: backgroundTimeout, + trackerNamespace: trackerNamespace, tracker: self) } - + UIKitScreenViewTracking.setup() NotificationCenter.default.addObserver( self, selector: #selector(receiveScreenViewNotification(_:)), name: NSNotification.Name("SPScreenViewDidAppear"), object: nil) - + NotificationCenter.default.addObserver( self, selector: #selector(receiveDiagnosticNotification(_:)), name: NSNotification.Name("SPTrackerDiagnostic"), object: nil) - + NotificationCenter.default.addObserver( self, selector: #selector(receiveCrashReporting(_:)), name: NSNotification.Name("SPCrashReporting"), object: nil) - + if exceptionEvents { NSSetUncaughtExceptionHandler(uncaughtExceptionHandler) } - + builderFinished = true } - + private func checkInstall() { if installEvent { DispatchQueue.global(qos: .default).async { [weak self] in @@ -332,54 +310,46 @@ class Tracker { } } - // MARK: - Functions - /// Add or replace state machine in the state manager func addOrReplace(stateMachine: StateMachineProtocol) { - sync { - self.stateManager.addOrReplaceStateMachine(stateMachine) - } + stateManager.addOrReplaceStateMachine(stateMachine) } /// Remove stata machine from the state manager func remove(stateMachineIdentifier: String) { - sync { - _ = self.stateManager.removeStateMachine(stateMachineIdentifier) - } + _ = stateManager.removeStateMachine(stateMachineIdentifier) } - + + // MARK: - Extra Functions + /// Pauses all event tracking, storage and session checking. func pauseEventTracking() { - sync { - self.dataCollection = false - self._emitter.pauseTimer() - self.trackerData.session?.stopChecker() - } + dataCollection = false + emitter.pauseTimer() + session?.stopChecker() } - + func resumeEventTracking() { - sync { - self.dataCollection = true - self._emitter.resumeTimer() - self.trackerData.session?.startChecker() - } + dataCollection = true + emitter.resumeTimer() + session?.startChecker() } - + // MARK: - Notifications management @objc func receiveScreenViewNotification(_ notification: Notification) { - asyncNotification { - guard let name = notification.userInfo?["name"] as? String else { return } - - var type: String? - if let typeId = (notification.userInfo?["type"] as? NSNumber)?.intValue, - let screenType = ScreenType(rawValue: typeId) { - type = ScreenView.stringWithScreenType(screenType) - } - - let topViewControllerClassName = notification.userInfo?["topViewControllerClassName"] as? String - let viewControllerClassName = notification.userInfo?["viewControllerClassName"] as? String + guard let name = notification.userInfo?["name"] as? String else { return } + + var type: String? + if let typeId = (notification.userInfo?["type"] as? NSNumber)?.intValue, + let screenType = ScreenType(rawValue: typeId) { + type = ScreenView.stringWithScreenType(screenType) + } + + let topViewControllerClassName = notification.userInfo?["topViewControllerClassName"] as? String + let viewControllerClassName = notification.userInfo?["viewControllerClassName"] as? String + InternalQueue.async { if self.autotrackScreenViews { let event = ScreenView(name: name, screenId: nil) event.type = type @@ -389,28 +359,28 @@ class Tracker { } } } - + @objc func receiveDiagnosticNotification(_ notification: Notification) { - asyncNotification { - let userInfo = notification.userInfo - guard let tag = userInfo?["tag"] as? String, - let message = userInfo?["message"] as? String else { return } - let error = userInfo?["error"] as? Error - let exception = userInfo?["exception"] as? NSException - + let userInfo = notification.userInfo + guard let tag = userInfo?["tag"] as? String, + let message = userInfo?["message"] as? String else { return } + let error = userInfo?["error"] as? Error + let exception = userInfo?["exception"] as? NSException + + InternalQueue.async { if self.trackerDiagnostic { let event = TrackerError(source: tag, message: message, error: error, exception: exception) let _ = self.track(event) } } } - + @objc func receiveCrashReporting(_ notification: Notification) { - asyncNotification { - let userInfo = notification.userInfo - guard let message = userInfo?["message"] as? String else { return } - let stacktrace = userInfo?["stacktrace"] as? String - + let userInfo = notification.userInfo + guard let message = userInfo?["message"] as? String else { return } + let stacktrace = userInfo?["stacktrace"] as? String + + InternalQueue.async { if self.exceptionEvents { let event = SNOWError(message: message) event.stackTrace = stacktrace @@ -418,66 +388,186 @@ class Tracker { } } } - + // MARK: - Event Tracking Functions - + /// Tracks an event despite its specific type. /// - Parameter event: The event to track - /// - Parameter synchronous: Whether to track the event synchronously or asynchronously - /// - Returns: The event ID - func track(_ event: Event, synchronous: Bool = false) -> UUID { - let eventId = UUID() + /// - Returns: The event ID or nil in case tracking is paused + func track(_ event: Event, eventId: UUID = UUID()) -> UUID { + InternalQueue.onQueuePrecondition() - let track = { - if !self.dataCollection { return } - + if dataCollection { event.beginProcessing(withTracker: self) self.processEvent(event, eventId) event.endProcessing(withTracker: self) } - - if synchronous { - self.sync(track) - } else { - self.async(track) - } - return eventId } - + // MARK: - Event Decoration - private func processEvent(_ event: Event, _ eventId: UUID) { + func processEvent(_ event: Event, _ eventId: UUID) { let stateSnapshot = stateManager.trackerState(forProcessedEvent: event) let trackerEvent = TrackerEvent(event: event, eventId: eventId, state: stateSnapshot) - let payloadBuilder = TrackerPayloadBuilder() - if let payload = payloadBuilder.payload(event: trackerEvent, tracker: self.trackerData, stateManager: self.stateManager) { - _emitter.addPayload(toBuffer: payload) + if let payload = self.payload(with: trackerEvent) { + emitter.addPayload(toBuffer: payload) stateManager.afterTrack(event: trackerEvent) } else { logDebug(message: "Event not tracked due to filtering") } } - - // MARK: - Serial dispatch queue - - private let serialQueue = DispatchQueue(label: "snowplow.tracker") - - private func sync(_ callback: () -> T) -> T { - dispatchPrecondition(condition: .notOnQueue(serialQueue)) - return serialQueue.sync(execute: callback) + func payload(with event: TrackerEvent) -> Payload? { + let payload = Payload() + payload.allowDiagnostic = !event.isService + + // Payload properties + setApplicationInstallEventTimestamp(event) + addBasicProperties(to: payload, event: event) + addStateMachinePayloadValues(event: event) + + // Context entities + addBasicContexts(event: event) + addStateMachineEntities(event: event) + + event.wrapProperties(to: payload, base64Encoded: base64Encoded) + event.wrapContexts(to: payload, base64Encoded: base64Encoded) + + // Decide whether to track the event or not + if !stateManager.filter(event: event) { + return nil + } + + // Workaround for campaign attribution + if !event.isPrimitive { + // TODO: To remove when Atomic table refactoring is finished + workaround(forCampaignAttributionEnrichment: payload, event: event) + } + return payload } - private func async(_ callback: @escaping () -> Void) { - serialQueue.async(execute: callback) + private func setApplicationInstallEventTimestamp(_ event: TrackerEvent) { + // Application_install event needs the timestamp to the real installation event. + if (event.schema == kSPApplicationInstallSchema) { + if let trueTimestamp = event.trueTimestamp { + event.timestamp = Int64(trueTimestamp.timeIntervalSince1970 * 1000) + event.trueTimestamp = nil + } + } } - - // MARK: - Notification dispatch queue - - private let notificationQueue = DispatchQueue(label: "snowplow.tracker.notifications", attributes: .concurrent) - - private func asyncNotification(_ callback: @escaping () -> Void) { - notificationQueue.async(execute: callback) + + func addBasicProperties(to payload: Payload, event: TrackerEvent) { + // Event ID + payload.addValueToPayload(event.eventId.uuidString, forKey: kSPEid) + // Timestamps + payload.addValueToPayload(String(format: "%lld", event.timestamp), forKey: kSPTimestamp) + if let trueTimestamp = event.trueTimestamp { + let ttInMilliSeconds = Int64(trueTimestamp.timeIntervalSince1970 * 1000) + payload.addValueToPayload(String(format: "%lld", ttInMilliSeconds), forKey: kSPTrueTimestamp) + } + // Tracker info (version, namespace, app ID) + payload.addDictionaryToPayload(trackerPayloadData()) + // Subject + if let subjectDict = subject?.standardDict(userAnonymisation: userAnonymisation) { + payload.addDictionaryToPayload(subjectDict) + } + // Platform + payload.addValueToPayload(devicePlatformToString(devicePlatform), forKey: kSPPlatform) + // Event name + if event.isPrimitive { + payload.addValueToPayload(event.eventName, forKey: kSPEvent) + } else { + payload.addValueToPayload(kSPEventUnstructured, forKey: kSPEvent) + } + } + + /* + This is needed because the campaign-attribution-enrichment (in the pipeline) is able to parse + the `url` and `referrer` only if they are part of a PageView event. + The PageView event is an atomic event but the DeepLinkReceived and ScreenView are SelfDescribing events. + For this reason we copy these two fields in the atomic fields in order to let the enrichment + to process correctly the fields even if the event is not a PageView and it's a SelfDescribing event. + This is a hack that should be removed once the atomic event table is dismissed and all the events + will be SelfDescribing. + */ + func workaround(forCampaignAttributionEnrichment payload: Payload, event: TrackerEvent) { + var url: String? + var referrer: String? + + if event.schema == DeepLinkReceived.schema { + url = event.payload[DeepLinkReceived.paramUrl] as? String + referrer = event.payload[DeepLinkReceived.paramReferrer] as? String + } else if event.schema == kSPScreenViewSchema { + for entity in event.entities { + if entity.schema == DeepLinkEntity.schema { + let data = entity.data + url = data[DeepLinkEntity.paramUrl] as? String + referrer = data[DeepLinkEntity.paramReferrer] as? String + break + } + } + } + + if let url = url { + payload.addValueToPayload(Utilities.truncateUrlScheme(url), forKey: kSPPageUrl) + } + if let referrer = referrer { + payload.addValueToPayload(Utilities.truncateUrlScheme(referrer), forKey: kSPPageRefr) + } + } + + func addBasicContexts(event: TrackerEvent) { + if subject != nil { + if let platformDict = subject?.platformDict( + userAnonymisation: userAnonymisation, + advertisingIdentifierRetriever: advertisingIdentifierRetriever)?.dictionary { + event.addContextEntity(SelfDescribingJson(schema: platformContextSchema, andDictionary: platformDict)) + } + if let geoLocationDict = subject?.geoLocationDict { + event.addContextEntity(SelfDescribingJson(schema: kSPGeoContextSchema, andDictionary: geoLocationDict)) + } + } + + if applicationContext { + if let contextJson = Utilities.applicationContext { + event.addContextEntity(contextJson) + } + } + + if event.isService { + return + } + + // Add session + if let session = session { + if let sessionDict = session.getDictWithEventId(event.eventId.uuidString, + eventTimestamp: event.timestamp, + userAnonymisation: userAnonymisation) { + event.addContextEntity(SelfDescribingJson(schema: kSPSessionContextSchema, andDictionary: sessionDict)) + } else { + logDiagnostic(message: String(format: "Unable to get session context for eventId: %@", event.eventId.uuidString)) + } + } + + // Add GDPR context + if let gdprContext = gdprContext?.context { + event.addContextEntity(gdprContext) + } + } + + private func addStateMachinePayloadValues(event: TrackerEvent) { + _ = stateManager.addPayloadValues(to: event) + } + + func addStateMachineEntities(event: TrackerEvent) { + let stateManagerEntities = stateManager.entities(forProcessedEvent: event) + for entity in stateManagerEntities { + event.addContextEntity(entity) + } + } + + deinit { + NotificationCenter.default.removeObserver(self) } } diff --git a/Sources/Core/Tracker/TrackerControllerImpl.swift b/Sources/Core/Tracker/TrackerControllerImpl.swift index bab1a7bee..62b56618f 100644 --- a/Sources/Core/Tracker/TrackerControllerImpl.swift +++ b/Sources/Core/Tracker/TrackerControllerImpl.swift @@ -69,6 +69,10 @@ class TrackerControllerImpl: Controller, TrackerController { dirtyConfig.isPaused = false tracker.resumeEventTracking() } + + func track(_ event: Event, eventId: UUID) { + _ = tracker.track(event, eventId: eventId) + } func track(_ event: Event) -> UUID { return tracker.track(event) diff --git a/Sources/Core/Tracker/TrackerData.swift b/Sources/Core/Tracker/TrackerData.swift deleted file mode 100644 index 875fea8a4..000000000 --- a/Sources/Core/Tracker/TrackerData.swift +++ /dev/null @@ -1,61 +0,0 @@ -// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. -// -// This program is licensed to you under the Apache License Version 2.0, -// and you may not use this file except in compliance with the Apache License -// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at -// http://www.apache.org/licenses/LICENSE-2.0. -// -// Unless required by applicable law or agreed to in writing, -// software distributed under the Apache License Version 2.0 is distributed on -// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either -// express or implied. See the Apache License Version 2.0 for the specific -// language governing permissions and limitations there under. - -import Foundation - -class TrackerData { - var advertisingIdentifierRetriever: (() -> UUID?)? - var applicationContext: Bool = TrackerDefaults.applicationContext - var appId: String - var autotrackScreenViews: Bool = TrackerDefaults.autotrackScreenViews - var base64Encoded: Bool = TrackerDefaults.base64Encoded - var backgroundTimeout: Int = TrackerDefaults.backgroundTimeout - /// Current screen view state. - var currentScreenState: ScreenState? - var devicePlatform: DevicePlatform = TrackerDefaults.devicePlatform - var deepLinkContext: Bool = false - var exceptionEvents = TrackerDefaults.exceptionEvents - var foregroundTimeout: Int = TrackerDefaults.foregroundTimeout - var gdprContext: GDPRContext? - var inBackground: Bool { - return session?.inBackground ?? false - } - var installEvent = TrackerDefaults.installEvent - var lifecycleEvents: Bool = false - var logLevel: LogLevel { - get { return Logger.logLevel } - set { Logger.logLevel = newValue } - } - var loggerDelegate: LoggerDelegate? { - get { return Logger.delegate } - set { Logger.delegate = newValue } - } - /// Previous screen view state. - var previousScreenState: ScreenState? - /// The object used for sessionization, i.e. it characterizes user activity. - var session: Session? - var sessionContext: Bool = false - var screenContext: Bool = false - var stateManager = StateManager() - var subject: Subject? - var trackerDiagnostic = TrackerDefaults.trackerDiagnostic - var trackerNamespace: String - var trackerVersionSuffix: String = TrackerDefaults.trackerVersionSuffix - var userAnonymisation: Bool = TrackerDefaults.userAnonymisation - - init(appId: String, trackerNamespace: String) { - self.appId = appId - self.trackerNamespace = trackerNamespace - } - -} diff --git a/Sources/Core/Tracker/TrackerPayloadBuilder.swift b/Sources/Core/Tracker/TrackerPayloadBuilder.swift deleted file mode 100644 index 3fd4d59c2..000000000 --- a/Sources/Core/Tracker/TrackerPayloadBuilder.swift +++ /dev/null @@ -1,191 +0,0 @@ -// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. -// -// This program is licensed to you under the Apache License Version 2.0, -// and you may not use this file except in compliance with the Apache License -// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at -// http://www.apache.org/licenses/LICENSE-2.0. -// -// Unless required by applicable law or agreed to in writing, -// software distributed under the Apache License Version 2.0 is distributed on -// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either -// express or implied. See the Apache License Version 2.0 for the specific -// language governing permissions and limitations there under. - -import Foundation - -class TrackerPayloadBuilder { - - func payload(event: TrackerEvent, tracker: TrackerData, stateManager: StateManager) -> Payload? { - let payload = Payload() - payload.allowDiagnostic = !event.isService - - // Payload properties - setApplicationInstallEventTimestamp(event) - addBasicProperties(to: payload, event: event, tracker: tracker) - addStateMachinePayloadValues(event: event, stateManager: stateManager) - - // Context entities - addBasicContexts(event: event, tracker: tracker) - addStateMachineEntities(event: event, stateManager: stateManager) - - event.wrapProperties(to: payload, base64Encoded: tracker.base64Encoded) - event.wrapContexts(to: payload, base64Encoded: tracker.base64Encoded) - - // Decide whether to track the event or not - if !stateManager.filter(event: event) { - return nil - } - - // Workaround for campaign attribution - if !event.isPrimitive { - // TODO: To remove when Atomic table refactoring is finished - workaround(forCampaignAttributionEnrichment: payload, event: event) - } - return payload - } - - private func trackerPayloadData(tracker: TrackerData) -> [String : String] { - var trackerVersion = kSPVersion - if tracker.trackerVersionSuffix.count != 0 { - var allowedCharSet = CharacterSet.alphanumerics - allowedCharSet.formUnion(CharacterSet(charactersIn: ".-")) - let suffix = tracker.trackerVersionSuffix.components(separatedBy: allowedCharSet.inverted).joined(separator: "") - if suffix.count != 0 { - trackerVersion = "\(trackerVersion) \(suffix)" - } - } - return [ - kSPTrackerVersion: trackerVersion, - kSPNamespace: tracker.trackerNamespace, - kSPAppId: tracker.appId - ] - } - - private func addBasicProperties(to payload: Payload, event: TrackerEvent, tracker: TrackerData) { - // Event ID - payload.addValueToPayload(event.eventId.uuidString, forKey: kSPEid) - // Timestamps - payload.addValueToPayload(String(format: "%lld", event.timestamp), forKey: kSPTimestamp) - if let trueTimestamp = event.trueTimestamp { - let ttInMilliSeconds = Int64(trueTimestamp.timeIntervalSince1970 * 1000) - payload.addValueToPayload(String(format: "%lld", ttInMilliSeconds), forKey: kSPTrueTimestamp) - } - // Tracker info (version, namespace, app ID) - payload.addDictionaryToPayload(trackerPayloadData(tracker: tracker)) - // Subject - if let subject = tracker.subject { - let subjectDict = subject.standardDict(userAnonymisation: tracker.userAnonymisation) - payload.addDictionaryToPayload(subjectDict) - } - // Platform - payload.addValueToPayload(devicePlatformToString(tracker.devicePlatform), forKey: kSPPlatform) - // Event name - if event.isPrimitive { - payload.addValueToPayload(event.eventName, forKey: kSPEvent) - } else { - payload.addValueToPayload(kSPEventUnstructured, forKey: kSPEvent) - } - } - - private func setApplicationInstallEventTimestamp(_ event: TrackerEvent) { - // Application_install event needs the timestamp to the real installation event. - if (event.schema == kSPApplicationInstallSchema) { - if let trueTimestamp = event.trueTimestamp { - event.timestamp = Int64(trueTimestamp.timeIntervalSince1970 * 1000) - event.trueTimestamp = nil - } - } - } - - /* - This is needed because the campaign-attribution-enrichment (in the pipeline) is able to parse - the `url` and `referrer` only if they are part of a PageView event. - The PageView event is an atomic event but the DeepLinkReceived and ScreenView are SelfDescribing events. - For this reason we copy these two fields in the atomic fields in order to let the enrichment - to process correctly the fields even if the event is not a PageView and it's a SelfDescribing event. - This is a hack that should be removed once the atomic event table is dismissed and all the events - will be SelfDescribing. - */ - private func workaround(forCampaignAttributionEnrichment payload: Payload, event: TrackerEvent) { - var url: String? - var referrer: String? - - if event.schema == DeepLinkReceived.schema { - url = event.payload[DeepLinkReceived.paramUrl] as? String - referrer = event.payload[DeepLinkReceived.paramReferrer] as? String - } else if event.schema == kSPScreenViewSchema { - for entity in event.entities { - if entity.schema == DeepLinkEntity.schema { - let data = entity.data - url = data[DeepLinkEntity.paramUrl] as? String - referrer = data[DeepLinkEntity.paramReferrer] as? String - break - } - } - } - - if let url = url { - payload.addValueToPayload(Utilities.truncateUrlScheme(url), forKey: kSPPageUrl) - } - if let referrer = referrer { - payload.addValueToPayload(Utilities.truncateUrlScheme(referrer), forKey: kSPPageRefr) - } - } - - private func addBasicContexts(event: TrackerEvent, tracker: TrackerData) { -#if os(iOS) - let platformContextSchema = kSPMobileContextSchema -#else - let platformContextSchema = kSPDesktopContextSchema -#endif - - if let subject = tracker.subject { - if let platformDict = subject.platformDict( - userAnonymisation: tracker.userAnonymisation, - advertisingIdentifierRetriever: tracker.advertisingIdentifierRetriever)?.dictionary { - event.addContextEntity(SelfDescribingJson(schema: platformContextSchema, andDictionary: platformDict)) - } - if let geoLocationDict = subject.geoLocationDict { - event.addContextEntity(SelfDescribingJson(schema: kSPGeoContextSchema, andDictionary: geoLocationDict)) - } - } - - if tracker.applicationContext { - if let contextJson = Utilities.applicationContext { - event.addContextEntity(contextJson) - } - } - - if event.isService { - return - } - - // Add session - if let session = tracker.session { - if let sessionDict = session.getDictWithEventId(event.eventId.uuidString, - eventTimestamp: event.timestamp, - userAnonymisation: tracker.userAnonymisation) { - event.addContextEntity(SelfDescribingJson(schema: kSPSessionContextSchema, andDictionary: sessionDict)) - } else { - logDiagnostic(message: String(format: "Unable to get session context for eventId: %@", event.eventId.uuidString)) - } - } - - // Add GDPR context - if let gdprContext = tracker.gdprContext?.context { - event.addContextEntity(gdprContext) - } - } - - private func addStateMachinePayloadValues(event: TrackerEvent, stateManager: StateManager) { - _ = stateManager.addPayloadValues(to: event) - } - - private func addStateMachineEntities(event: TrackerEvent, stateManager: StateManager) { - let stateManagerEntities = stateManager.entities(forProcessedEvent: event) - for entity in stateManagerEntities { - event.addContextEntity(entity) - } - } - -} diff --git a/Sources/Core/Utils/DataPersistence.swift b/Sources/Core/Utils/DataPersistence.swift index 9d2d3b887..c59851944 100644 --- a/Sources/Core/Utils/DataPersistence.swift +++ b/Sources/Core/Utils/DataPersistence.swift @@ -24,8 +24,6 @@ var sessionKey = "session" class DataPersistence { var data: [String : [String : Any]] { get { - objc_sync_enter(self) - defer { objc_sync_exit(self) } if !isStoredOnFile { return ((UserDefaults.standard.dictionary(forKey: userDefaultsKey) ?? [:]) as? [String : [String : Any]]) ?? [:] } @@ -51,13 +49,11 @@ class DataPersistence { return result ?? [:] } set(data) { - objc_sync_enter(self) if let fileUrl = fileUrl { let _ = storeDictionary(data, fileURL: fileUrl) } else { UserDefaults.standard.set(data, forKey: userDefaultsKey) } - objc_sync_exit(self) } } @@ -66,11 +62,9 @@ class DataPersistence { return (data)[sessionKey] } set(session) { - objc_sync_enter(self) var data = self.data data[sessionKey] = session self.data = data - objc_sync_exit(self) } } @@ -100,8 +94,6 @@ class DataPersistence { if escapedNamespace.count <= 0 { return nil } - objc_sync_enter(DataPersistence.self) - defer { objc_sync_exit(DataPersistence.self) } if let instances = instances { if let instance = instances[escapedNamespace] { @@ -118,9 +110,7 @@ class DataPersistence { class func remove(withNamespace namespace: String) -> Bool { if let instance = DataPersistence.getFor(namespace: namespace) { - objc_sync_enter(DataPersistence.self) instances?.removeValue(forKey: instance.escapedNamespace) - objc_sync_exit(DataPersistence.self) let _ = instance.removeAll() } return true @@ -139,9 +129,6 @@ class DataPersistence { func removeAll() -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - UserDefaults.standard.removeObject(forKey: userDefaultsKey) if let fileUrl = fileUrl { diff --git a/Sources/Snowplow/Network/DefaultNetworkConnection.swift b/Sources/Snowplow/Network/DefaultNetworkConnection.swift index 1c5c6535b..143339764 100644 --- a/Sources/Snowplow/Network/DefaultNetworkConnection.swift +++ b/Sources/Snowplow/Network/DefaultNetworkConnection.swift @@ -19,83 +19,64 @@ public class DefaultNetworkConnection: NSObject, NetworkConnection { // The protocol for connection to the collector @objc public var `protocol`: ProtocolOptions { - get { return sync { _protocol } } - set { sync { _protocol = newValue; setup() } } + get { return _protocol } + set { _protocol = newValue; setup() } } private var _urlString: String /// The collector endpoint. @objc public var urlString: String { - get { return sync { urlEndpoint?.absoluteString ?? _urlString } } - set { sync { _urlString = newValue; setup() } } + get { return urlEndpoint?.absoluteString ?? _urlString } + set { _urlString = newValue; setup() } } private var _urlEndpoint: URL? - public var urlEndpoint: URL? { sync { return _urlEndpoint } } + public var urlEndpoint: URL? { return _urlEndpoint } private var _httpMethod: HttpMethodOptions = .post /// HTTP method, should be .get or .post. @objc public var httpMethod: HttpMethodOptions { - get { return sync { _httpMethod } } - set(method) { sync { _httpMethod = method; setup() } } + get { return _httpMethod } + set(method) { _httpMethod = method; setup() } } private var _emitThreadPoolSize = 15 /// The number of threads used by the emitter. @objc public var emitThreadPoolSize: Int { - get { sync { return _emitThreadPoolSize } } + get { return _emitThreadPoolSize } set(emitThreadPoolSize) { - sync { - self._emitThreadPoolSize = emitThreadPoolSize - if dataOperationQueue.maxConcurrentOperationCount != emitThreadPoolSize { - dataOperationQueue.maxConcurrentOperationCount = emitThreadPoolSize - } + self._emitThreadPoolSize = emitThreadPoolSize + if dataOperationQueue.maxConcurrentOperationCount != emitThreadPoolSize { + dataOperationQueue.maxConcurrentOperationCount = emitThreadPoolSize } } } - private var _byteLimitGet: Int = 40000 /// Maximum event size for a GET request. @objc - public var byteLimitGet: Int { - get { return sync { _byteLimitGet } } - set { sync { _byteLimitGet = newValue } } - } + public var byteLimitGet: Int = 40000 - private var _byteLimitPost = 40000 /// Maximum event size for a POST request. @objc - public var byteLimitPost: Int { - get { return sync { _byteLimitPost } } - set { sync { _byteLimitPost = newValue } } - } + public var byteLimitPost = 40000 private var _customPostPath: String? /// A custom path that is used on the endpoint to send requests. - @objc - public var customPostPath: String? { - get { return sync { _customPostPath } } - set { sync { _customPostPath = newValue; setup() } } + @objc public var customPostPath: String? { + get { return _customPostPath } + set { _customPostPath = newValue; setup() } } - private var _requestHeaders: [String : String]? /// Custom headers (key, value) for http requests. @objc - public var requestHeaders: [String : String]? { - get { return sync { _requestHeaders } } - set { sync { _requestHeaders = newValue } } - } - /// Whether to anonymise server-side user identifiers including the `network_userid` and `user_ipaddress` + public var requestHeaders: [String : String]? - private var _serverAnonymisation = false + /// Whether to anonymise server-side user identifiers including the `network_userid` and `user_ipaddress` @objc - public var serverAnonymisation: Bool { - get { return sync { _serverAnonymisation } } - set { sync { _serverAnonymisation = newValue } } - } + public var serverAnonymisation = false private var dataOperationQueue = OperationQueue() @objc @@ -239,13 +220,4 @@ public class DefaultNetworkConnection: NSObject, NetworkConnection { }) } - // MARK: - dispatch queues - - private let dispatchQueue = DispatchQueue(label: "snowplow.tracker.network_connection") - - private func sync(_ callback: () -> T) -> T { - dispatchPrecondition(condition: .notOnQueue(dispatchQueue)) - - return dispatchQueue.sync(execute: callback) - } } diff --git a/Sources/Snowplow/Payload/Payload.swift b/Sources/Snowplow/Payload/Payload.swift index f6aa86e49..8485f206d 100644 --- a/Sources/Snowplow/Payload/Payload.swift +++ b/Sources/Snowplow/Payload/Payload.swift @@ -22,8 +22,6 @@ public class Payload: NSObject { /// Returns the payload of that particular SPPayload object. /// - Returns: NSDictionary of data in the object. public var dictionary: [String : Any] { - objc_sync_enter(self) - defer { objc_sync_exit(self) } return payload } @@ -31,8 +29,6 @@ public class Payload: NSObject { /// - Returns: A long representing the byte size of the payload. @objc public var byteSize: Int { - objc_sync_enter(self) - defer { objc_sync_exit(self) } if let data = try? JSONSerialization.data(withJSONObject: payload) { return data.count } @@ -66,7 +62,6 @@ public class Payload: NSObject { /// - key: A key of type NSString @objc public func addValueToPayload(_ value: Any?, forKey key: String) { - objc_sync_enter(self) if value == nil { if payload[key] != nil { payload.removeValue(forKey: key) @@ -74,7 +69,6 @@ public class Payload: NSObject { } else { payload[key] = value } - objc_sync_exit(self) } /// Adds a dictionary of attributes to be appended into the SPPayload instance. It does NOT overwrite the existing data in the object. diff --git a/Sources/Snowplow/Snowplow.swift b/Sources/Snowplow/Snowplow.swift index f8fec0898..3a3ecaaf9 100644 --- a/Sources/Snowplow/Snowplow.swift +++ b/Sources/Snowplow/Snowplow.swift @@ -199,13 +199,15 @@ public class Snowplow: NSObject { /// - Returns: The tracker instance created. @objc public class func createTracker(namespace: String, network networkConfiguration: NetworkConfiguration, configurations: [ConfigurationProtocol] = []) -> TrackerController? { - if let serviceProvider = serviceProviderInstances[namespace] { - serviceProvider.reset(configurations: configurations + [networkConfiguration]) - return serviceProvider.trackerController - } else { - let serviceProvider = ServiceProvider(namespace: namespace, network: networkConfiguration, configurations: configurations) - let _ = registerInstance(serviceProvider) - return serviceProvider.trackerController + InternalQueue.sync { + if let serviceProvider = serviceProviderInstances[namespace] { + serviceProvider.reset(configurations: configurations + [networkConfiguration]) + return TrackerControllerIQWrapper(controller: serviceProvider.trackerController) + } else { + let serviceProvider = ServiceProvider(namespace: namespace, network: networkConfiguration, configurations: configurations) + let _ = registerInstance(serviceProvider) + return TrackerControllerIQWrapper(controller: serviceProvider.trackerController) + } } } @@ -248,7 +250,12 @@ public class Snowplow: NSObject { /// calling `setTrackerAsDefault(TrackerController)`. @objc public class func defaultTracker() -> TrackerController? { - return defaultServiceProvider?.trackerController + InternalQueue.sync { + if let controller = defaultServiceProvider?.trackerController { + return TrackerControllerIQWrapper(controller: controller) + } + return nil + } } /// Using the namespace identifier is possible to get the trackerController if already instanced. @@ -257,7 +264,12 @@ public class Snowplow: NSObject { /// - Returns: The tracker if it exist with that namespace. @objc public class func tracker(namespace: String) -> TrackerController? { - return serviceProviderInstances[namespace]?.trackerController + InternalQueue.sync { + if let controller = serviceProviderInstances[namespace]?.trackerController { + return TrackerControllerIQWrapper(controller: controller) + } + return nil + } } /// Set the passed tracker as default tracker if it's registered as an active tracker in the app. @@ -269,13 +281,14 @@ public class Snowplow: NSObject { /// - Returns: Whether the tracker passed is registered among the active trackers of the app. @objc public class func setAsDefault(tracker trackerController: TrackerController?) -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - - if let namespace = trackerController?.namespace, - let serviceProvider = serviceProviderInstances[namespace] { - defaultServiceProvider = serviceProvider - return true + if let namespace = trackerController?.namespace { + return InternalQueue.sync { + if let serviceProvider = serviceProviderInstances[namespace] { + defaultServiceProvider = serviceProvider + return true + } + return false + } } return false } @@ -291,20 +304,12 @@ public class Snowplow: NSObject { /// - Returns: Whether it has been able to remove it. @objc public class func remove(tracker trackerController: TrackerController?) -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } - if let namespace = trackerController?.namespace, - let serviceProvider = (serviceProviderInstances)[namespace] { - serviceProvider.shutdown() - serviceProviderInstances.removeValue(forKey: namespace) - if serviceProvider == defaultServiceProvider { - defaultServiceProvider = nil - } - return true + if let namespace = trackerController?.namespace { + return remove(namespace: namespace) } return false } - + /// Remove all the trackers. /// /// The removed tracker is always stopped. @@ -312,20 +317,22 @@ public class Snowplow: NSObject { /// See ``remove(tracker:)`` to remove a specific tracker. @objc public class func removeAllTrackers() { - objc_sync_enter(self) - defaultServiceProvider = nil - let serviceProviders = serviceProviderInstances.values - serviceProviderInstances.removeAll() - for sp in serviceProviders { - sp.shutdown() + InternalQueue.sync { + defaultServiceProvider = nil + let serviceProviders = serviceProviderInstances.values + serviceProviderInstances.removeAll() + for sp in serviceProviders { + sp.shutdown() + } } - objc_sync_exit(self) } /// - Returns: Set of namespace of the active trackers in the app. @objc class public var instancedTrackerNamespaces: [String] { - return Array(serviceProviderInstances.keys) + InternalQueue.sync { + return Array(serviceProviderInstances.keys) + } } #if os(iOS) || os(macOS) @@ -345,8 +352,6 @@ public class Snowplow: NSObject { // MARK: - Private methods private class func registerInstance(_ serviceProvider: ServiceProvider) -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } let namespace = serviceProvider.namespace let isOverriding = serviceProviderInstances[namespace] != nil serviceProviderInstances[namespace] = serviceProvider @@ -359,7 +364,6 @@ public class Snowplow: NSObject { private class func createTrackers(configurationBundles bundles: [ConfigurationBundle]) -> [String] { var namespaces: [String]? = [] for bundle in bundles { - objc_sync_enter(self) if let networkConfiguration = bundle.networkConfiguration { if let _ = createTracker( namespace: bundle.namespace, @@ -369,12 +373,24 @@ public class Snowplow: NSObject { } } else { // remove tracker if it exists - if let tracker = tracker(namespace: bundle.namespace) { - let _ = remove(tracker: tracker) - } + _ = remove(namespace: bundle.namespace) } - objc_sync_exit(self) } return namespaces ?? [] } + + private class func remove(namespace: String) -> Bool { + InternalQueue.sync { + if let serviceProvider = (serviceProviderInstances)[namespace] { + serviceProvider.shutdown() + serviceProviderInstances.removeValue(forKey: namespace) + if serviceProvider == defaultServiceProvider { + defaultServiceProvider = nil + } + return true + } + return false + } + } + } diff --git a/Tests/Configurations/TestMultipleInstances.swift b/Tests/Configurations/TestMultipleInstances.swift index 7cf3fa36b..dec5fc8e9 100644 --- a/Tests/Configurations/TestMultipleInstances.swift +++ b/Tests/Configurations/TestMultipleInstances.swift @@ -29,7 +29,7 @@ class TestMultipleInstances: XCTestCase { let t2 = Snowplow.createTracker(namespace: "t1", network: NetworkConfiguration(endpoint: "snowplowanalytics.fake2")) XCTAssertEqual(t2?.network?.endpoint, "https://snowplowanalytics.fake2/com.snowplowanalytics.snowplow/tp2") XCTAssertEqual(["t1"], Snowplow.instancedTrackerNamespaces) - XCTAssertTrue(t1 === t2) + XCTAssertTrue(t1?.network?.endpoint == t2?.network?.endpoint) } func testMultipleInstances() { diff --git a/Tests/Legacy Tests/LegacyTestEmitter.swift b/Tests/Legacy Tests/LegacyTestEmitter.swift index 1933cf3c1..c9ec1485a 100644 --- a/Tests/Legacy Tests/LegacyTestEmitter.swift +++ b/Tests/Legacy Tests/LegacyTestEmitter.swift @@ -115,7 +115,7 @@ class LegacyTestEmitter: XCTestCase { RunLoop.main.run(until: Date(timeIntervalSinceNow: 1)) emitter.resumeTimer() - emitter.flush() + flush(emitter) } // MARK: - Emitting tests @@ -135,7 +135,7 @@ class LegacyTestEmitter: XCTestCase { func testEmitSingleGetEventWithSuccess() { let networkConnection = MockNetworkConnection(requestOption: .get, statusCode: 200) let emitter = self.emitter(with: networkConnection, bufferOption: .single) - emitter.addPayload(toBuffer: generatePayloads(1).first!) + addPayload(generatePayloads(1).first!, emitter) for _ in 0..<10 { Thread.sleep(forTimeInterval: 1) @@ -146,13 +146,13 @@ class LegacyTestEmitter: XCTestCase { XCTAssertTrue(networkConnection.previousResults.first!.first!.isSuccessful) XCTAssertEqual(0, emitter.dbCount) - emitter.flush() + flush(emitter) } func testEmitSingleGetEventWithNoSuccess() { let networkConnection = MockNetworkConnection(requestOption: .get, statusCode: 500) let emitter = self.emitter(with: networkConnection, bufferOption: .single) - emitter.addPayload(toBuffer: generatePayloads(1).first!) + addPayload(generatePayloads(1).first!, emitter) for _ in 0..<10 { Thread.sleep(forTimeInterval: 1) @@ -163,7 +163,7 @@ class LegacyTestEmitter: XCTestCase { XCTAssertFalse(networkConnection.previousResults.first!.first!.isSuccessful) XCTAssertEqual(1, emitter.dbCount) - emitter.flush() + flush(emitter) } func testEmitTwoGetEventsWithSuccess() { @@ -171,7 +171,7 @@ class LegacyTestEmitter: XCTestCase { let emitter = self.emitter(with: networkConnection, bufferOption: .single) for payload in generatePayloads(2) { - emitter.addPayload(toBuffer: payload) + addPayload(payload, emitter) } for _ in 0..<10 { @@ -188,7 +188,7 @@ class LegacyTestEmitter: XCTestCase { } XCTAssertEqual(2, totEvents) - emitter.flush() + flush(emitter) } func testEmitTwoGetEventsWithNoSuccess() { @@ -196,7 +196,7 @@ class LegacyTestEmitter: XCTestCase { let emitter = self.emitter(with: networkConnection, bufferOption: .single) for payload in generatePayloads(2) { - emitter.addPayload(toBuffer: payload) + addPayload(payload, emitter) } for _ in 0..<10 { @@ -210,14 +210,14 @@ class LegacyTestEmitter: XCTestCase { } } - emitter.flush() + flush(emitter) } func testEmitSinglePostEventWithSuccess() { let networkConnection = MockNetworkConnection(requestOption: .post, statusCode: 200) let emitter = self.emitter(with: networkConnection, bufferOption: .single) - emitter.addPayload(toBuffer: generatePayloads(1).first!) + addPayload(generatePayloads(1).first!, emitter) for _ in 0..<10 { Thread.sleep(forTimeInterval: 1) @@ -228,16 +228,17 @@ class LegacyTestEmitter: XCTestCase { XCTAssertTrue(networkConnection.previousResults.first!.first!.isSuccessful) XCTAssertEqual(0, emitter.dbCount) - emitter.flush() + flush(emitter) } func testEmitEventsPostAsGroup() { + let payloads = generatePayloads(15) + let networkConnection = MockNetworkConnection(requestOption: .post, statusCode: 500) let emitter = self.emitter(with: networkConnection, bufferOption: .defaultGroup) - - let payloads = generatePayloads(15) + for i in 0..<14 { - emitter.addPayload(toBuffer: payloads[i]) + addPayload(payloads[i], emitter) } for _ in 0..<10 { @@ -247,7 +248,7 @@ class LegacyTestEmitter: XCTestCase { XCTAssertEqual(14, emitter.dbCount) networkConnection.statusCode = 200 let prevSendingCount = networkConnection.sendingCount - emitter.addPayload(toBuffer: payloads[14]) + addPayload(payloads[14], emitter) for _ in 0..<10 { Thread.sleep(forTimeInterval: 1) @@ -268,7 +269,7 @@ class LegacyTestEmitter: XCTestCase { XCTAssertEqual(15, totEvents) XCTAssertTrue(areGrouped) - emitter.flush() + flush(emitter) } func testEmitOversizeEventsPostAsGroup() { @@ -278,7 +279,7 @@ class LegacyTestEmitter: XCTestCase { let payloads = generatePayloads(15) for i in 0..<14 { - emitter.addPayload(toBuffer: payloads[i]) + addPayload(payloads[i], emitter) } for _ in 0..<10 { @@ -288,7 +289,7 @@ class LegacyTestEmitter: XCTestCase { XCTAssertEqual(0, emitter.dbCount) networkConnection.statusCode = 200 _ = networkConnection.sendingCount - emitter.addPayload(toBuffer: payloads[14]) + addPayload(payloads[14], emitter) for _ in 0..<10 { Thread.sleep(forTimeInterval: 1) @@ -296,14 +297,14 @@ class LegacyTestEmitter: XCTestCase { XCTAssertEqual(0, emitter.dbCount) - emitter.flush() + flush(emitter) } func testRemovesEventsFromQueueOnNoRetryStatus() { let networkConnection = MockNetworkConnection(requestOption: .get, statusCode: 403) let emitter = self.emitter(with: networkConnection, bufferOption: .single) - emitter.addPayload(toBuffer: generatePayloads(1).first!) + addPayload(generatePayloads(1).first!, emitter) Thread.sleep(forTimeInterval: 1) @@ -314,7 +315,7 @@ class LegacyTestEmitter: XCTestCase { } } - emitter.flush() + flush(emitter) } func testFollowCustomRetryRules() { @@ -326,7 +327,7 @@ class LegacyTestEmitter: XCTestCase { customRules[500] = false emitter.customRetryForStatusCodes = customRules - emitter.addPayload(toBuffer: generatePayloads(1).first!) + addPayload(generatePayloads(1).first!, emitter) Thread.sleep(forTimeInterval: 1) @@ -335,14 +336,14 @@ class LegacyTestEmitter: XCTestCase { networkConnection.statusCode = 403 - emitter.addPayload(toBuffer: generatePayloads(1).first!) + addPayload(generatePayloads(1).first!, emitter) Thread.sleep(forTimeInterval: 1) // event still in queue because retrying is enabled for 403 XCTAssertEqual(1, emitter.dbCount) - emitter.flush() + flush(emitter) } func testDoesNotRetryFailedRequestsIfDisabled() { @@ -350,7 +351,7 @@ class LegacyTestEmitter: XCTestCase { let emitter = self.emitter(with: networkConnection, bufferOption: .single) emitter.retryFailedRequests = false - emitter.addPayload(toBuffer: generatePayloads(1).first!) + addPayload(generatePayloads(1).first!, emitter) Thread.sleep(forTimeInterval: 1) @@ -359,14 +360,14 @@ class LegacyTestEmitter: XCTestCase { emitter.retryFailedRequests = true - emitter.addPayload(toBuffer: generatePayloads(1).first!) + addPayload(generatePayloads(1).first!, emitter) Thread.sleep(forTimeInterval: 1) // event still in queue because retrying is enabled XCTAssertEqual(1, emitter.dbCount) - emitter.flush() + flush(emitter) } // MARK: - Emitter builder @@ -392,5 +393,17 @@ class LegacyTestEmitter: XCTestCase { } return payloads } + + private func addPayload(_ eventPayload: Payload, _ emitter: Emitter) { + InternalQueue.sync { + emitter.addPayload(toBuffer: eventPayload) + } + } + + private func flush(_ emitter: Emitter) { + InternalQueue.sync { + emitter.flush() + } + } } //#pragma clang diagnostic pop diff --git a/Tests/Media/TestMediaController.swift b/Tests/Media/TestMediaController.swift index ab3d9f1e8..03ed1256c 100644 --- a/Tests/Media/TestMediaController.swift +++ b/Tests/Media/TestMediaController.swift @@ -337,7 +337,7 @@ class TestMediaController: XCTestCase { // MARK: Ping events func testStartsSendingPingEventsAfterSessionStarts() { - let pingInterval = MediaPingInterval(timerProvider: MockTimer.self) + let pingInterval = MediaPingInterval(startTimer: MockTimer.startTimer) _ = MediaTrackingImpl(id: "media1", tracker: tracker!, pingInterval: pingInterval) MockTimer.currentTimer.fire() @@ -349,7 +349,7 @@ class TestMediaController: XCTestCase { } func testShouldSendPingEventsRegardlessOfOtherEvents() { - let pingInterval = MediaPingInterval(timerProvider: MockTimer.self) + let pingInterval = MediaPingInterval(startTimer: MockTimer.startTimer) let media = MediaTrackingImpl(id: "media1", tracker: tracker!, pingInterval: pingInterval) media.track(MediaPlayEvent()) @@ -363,7 +363,7 @@ class TestMediaController: XCTestCase { } func testShouldStopSendingPingEventsWhenPaused() { - let pingInterval = MediaPingInterval(maxPausedPings: 2, timerProvider: MockTimer.self) + let pingInterval = MediaPingInterval(maxPausedPings: 2, startTimer: MockTimer.startTimer) let media = MediaTrackingImpl(id: "media1", tracker: tracker!, pingInterval: pingInterval) media.update(player: MediaPlayerEntity(paused: true)) @@ -377,7 +377,7 @@ class TestMediaController: XCTestCase { } func testShouldNotStopSendingPingEventsWhenPlaying() { - let pingInterval = MediaPingInterval(maxPausedPings: 2, timerProvider: MockTimer.self) + let pingInterval = MediaPingInterval(maxPausedPings: 2, startTimer: MockTimer.startTimer) let media = MediaTrackingImpl(id: "media1", tracker: tracker!, pingInterval: pingInterval) media.update(player: MediaPlayerEntity(paused: false)) diff --git a/Tests/TestLifecycleState.swift b/Tests/TestLifecycleState.swift index c4e76620b..2914ed94e 100644 --- a/Tests/TestLifecycleState.swift +++ b/Tests/TestLifecycleState.swift @@ -25,15 +25,18 @@ class TestLifecycleState: XCTestCase { func testLifecycleStateMachine() { let eventStore = MockEventStore() - let emitter = Emitter(namespace: "namespace", urlEndpoint: "http://snowplow-fake-url.com", eventStore: eventStore) + let emitter = Emitter( + networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), + namespace: "namespace", + eventStore: eventStore + ) let tracker = Tracker(trackerNamespace: "namespace", appId: nil, emitter: emitter) { tracker in tracker.base64Encoded = false tracker.lifecycleEvents = true } // Send events - _ = tracker.track(Timing(category: "category", variable: "variable", timing: 123)) - Thread.sleep(forTimeInterval: 1) + track(Timing(category: "category", variable: "variable", timing: 123), tracker) if eventStore.lastInsertedRow == -1 { XCTFail() } @@ -43,8 +46,7 @@ class TestLifecycleState: XCTestCase { XCTAssertNotNil(entities) XCTAssertTrue(entities!.contains("\"isVisible\":true")) - _ = tracker.track(Background(index: 1)) - Thread.sleep(forTimeInterval: 1) + track(Background(index: 1), tracker) if eventStore.lastInsertedRow == -1 { XCTFail() } @@ -54,8 +56,7 @@ class TestLifecycleState: XCTestCase { XCTAssertNotNil(entities) XCTAssertTrue(entities!.contains("\"isVisible\":false")) - _ = tracker.track(Timing(category: "category", variable: "variable", timing: 123)) - Thread.sleep(forTimeInterval: 1) + track(Timing(category: "category", variable: "variable", timing: 123), tracker) if eventStore.lastInsertedRow == -1 { XCTFail() } @@ -64,8 +65,7 @@ class TestLifecycleState: XCTestCase { entities = (payload?["co"]) as? String XCTAssertTrue(entities!.contains("\"isVisible\":false")) - _ = tracker.track(Foreground(index: 1)) - Thread.sleep(forTimeInterval: 1) + track(Foreground(index: 1), tracker) if eventStore.lastInsertedRow == -1 { XCTFail() } @@ -76,8 +76,7 @@ class TestLifecycleState: XCTestCase { XCTAssertTrue(entities!.contains("\"isVisible\":true")) let uuid = UUID() - _ = tracker.track(ScreenView(name: "screen1", screenId: uuid)) - Thread.sleep(forTimeInterval: 1) + track(ScreenView(name: "screen1", screenId: uuid), tracker) if eventStore.lastInsertedRow == -1 { XCTFail() } @@ -87,4 +86,10 @@ class TestLifecycleState: XCTestCase { XCTAssertNotNil(entities) XCTAssertTrue(entities!.contains("\"isVisible\":true")) } + + private func track(_ event: Event, _ tracker: Tracker) { + InternalQueue.sync { + _ = tracker.track(event) + } + } } diff --git a/Tests/TestRequest.swift b/Tests/TestRequest.swift index 0d3e9c225..330c6932a 100644 --- a/Tests/TestRequest.swift +++ b/Tests/TestRequest.swift @@ -124,7 +124,9 @@ class TestRequest: XCTestCase, RequestCallback { if emitter?.dbCount == 0 { break } - emitter?.flush() + InternalQueue.sync { + emitter?.flush() + } Thread.sleep(forTimeInterval: 5) } Thread.sleep(forTimeInterval: 3) @@ -153,7 +155,7 @@ class TestRequest: XCTestCase, RequestCallback { event.property = "DemoProperty" event.value = NSNumber(value: 5) event.entities = customContext() - _ = tracker_.track(event) + track(event, tracker_) return 1 } @@ -166,7 +168,7 @@ class TestRequest: XCTestCase, RequestCallback { andDictionary: data) let event = SelfDescribing(eventData: sdj) event.entities = customContext() - _ = tracker_.track(event) + track(event, tracker_) return 1 } @@ -175,14 +177,14 @@ class TestRequest: XCTestCase, RequestCallback { event.pageTitle = "DemoPageTitle" event.referrer = "DemoPageReferrer" event.entities = customContext() - _ = tracker_.track(event) + track(event, tracker_) return 1 } func trackScreenView(with tracker_: Tracker) -> Int { let event = ScreenView(name: "DemoScreenName", screenId: nil) event.entities = customContext() - _ = tracker_.track(event) + track(event, tracker_) return 1 } @@ -190,7 +192,7 @@ class TestRequest: XCTestCase, RequestCallback { let event = Timing(category: "DemoTimingCategory", variable: "DemoTimingVariable", timing: 5) event.label = "DemoTimingLabel" event.entities = customContext() - _ = tracker_.track(event) + track(event, tracker_) return 1 } @@ -212,7 +214,7 @@ class TestRequest: XCTestCase, RequestCallback { event.country = "USA" event.currency = "USD" event.entities = customContext() - _ = tracker_.track(event) + track(event, tracker_) return 2 } @@ -225,4 +227,10 @@ class TestRequest: XCTestCase, RequestCallback { andDictionary: data) return [context] } + + private func track(_ event: Event, _ tracker: Tracker) { + InternalQueue.sync { + _ = tracker.track(event) + } + } } diff --git a/Tests/TestScreenState.swift b/Tests/TestScreenState.swift index d8769eb53..d7218489b 100644 --- a/Tests/TestScreenState.swift +++ b/Tests/TestScreenState.swift @@ -68,7 +68,7 @@ class TestScreenState: XCTestCase { emitter.pauseEmit() // Send events - _ = tracker.track(Timing(category: "category", variable: "variable", timing: 123)) + track(Timing(category: "category", variable: "variable", timing: 123), tracker) Thread.sleep(forTimeInterval: 1) if eventStore.lastInsertedRow == -1 { XCTFail() @@ -79,7 +79,7 @@ class TestScreenState: XCTestCase { XCTAssertNil(entities) let uuid = UUID() - _ = tracker.track(ScreenView(name: "screen1", screenId: uuid)) + track(ScreenView(name: "screen1", screenId: uuid), tracker) Thread.sleep(forTimeInterval: 1) if eventStore.lastInsertedRow == -1 { XCTFail() @@ -90,7 +90,7 @@ class TestScreenState: XCTestCase { XCTAssertNotNil(entities) XCTAssertTrue(entities!.contains(uuid.uuidString)) - _ = tracker.track(Timing(category: "category", variable: "variable", timing: 123)) + track(Timing(category: "category", variable: "variable", timing: 123), tracker) Thread.sleep(forTimeInterval: 1) if eventStore.lastInsertedRow == -1 { XCTFail() @@ -102,7 +102,7 @@ class TestScreenState: XCTestCase { XCTAssertTrue(entities!.contains(uuid.uuidString)) let uuid2 = UUID() - _ = tracker.track(ScreenView(name: "screen2", screenId: uuid2)) + track(ScreenView(name: "screen2", screenId: uuid2), tracker) Thread.sleep(forTimeInterval: 1) if eventStore.lastInsertedRow == -1 { XCTFail() @@ -117,7 +117,7 @@ class TestScreenState: XCTestCase { XCTAssertTrue(eventPayload!.contains(uuid.uuidString)) XCTAssertTrue(eventPayload!.contains(uuid2.uuidString)) - _ = tracker.track(Timing(category: "category", variable: "variable", timing: 123)) + track(Timing(category: "category", variable: "variable", timing: 123), tracker) Thread.sleep(forTimeInterval: 1) if eventStore.lastInsertedRow == -1 { XCTFail() @@ -128,4 +128,10 @@ class TestScreenState: XCTestCase { XCTAssertNotNil(entities) XCTAssertTrue(entities!.contains(uuid2.uuidString)) } + + private func track(_ event: Event, _ tracker: Tracker) { + InternalQueue.sync { + _ = tracker.track(event) + } + } } diff --git a/Tests/TestServiceProvider.swift b/Tests/TestServiceProvider.swift index 26617a807..739425eda 100644 --- a/Tests/TestServiceProvider.swift +++ b/Tests/TestServiceProvider.swift @@ -48,13 +48,17 @@ class TestServiceProvider: XCTestCase { serviceProvider.reset(configurations: [EmitterConfiguration()]) // track event and check that emitter is paused - _ = serviceProvider.trackerController.track(Structured(category: "cat", action: "act")) + InternalQueue.sync { + _ = serviceProvider.trackerController.track(Structured(category: "cat", action: "act")) + } Thread.sleep(forTimeInterval: 3) XCTAssertEqual(1, serviceProvider.emitter.dbCount) XCTAssertEqual(0, networkConnection.sendingCount) // resume emitting - serviceProvider.emitterController.resume() + InternalQueue.sync { + serviceProvider.emitterController.resume() + } Thread.sleep(forTimeInterval: 3) XCTAssertEqual(1, networkConnection.sendingCount) XCTAssertEqual(0, serviceProvider.emitter.dbCount) diff --git a/Tests/TestSession.swift b/Tests/TestSession.swift index 79c530024..cce57c780 100644 --- a/Tests/TestSession.swift +++ b/Tests/TestSession.swift @@ -102,7 +102,7 @@ class TestSession: XCTestCase { func testBackgroundEventsOnWhenLifecycleEventsDisabled() { cleanFile(withNamespace: "tracker") - let emitter = Emitter(namespace: "tracker", urlEndpoint: "") + let emitter = Emitter(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), namespace: "tracker") let tracker = Tracker(trackerNamespace: "tracker", appId: nil, emitter: emitter) { tracker in tracker.lifecycleEvents = false tracker.sessionContext = true @@ -126,7 +126,7 @@ class TestSession: XCTestCase { func testBackgroundEventsOnSameSession() { cleanFile(withNamespace: "t1") - let emitter = Emitter(namespace: "t1", urlEndpoint: "") + let emitter = Emitter(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), namespace: "t1") let tracker = Tracker(trackerNamespace: "t1", appId: nil, emitter: emitter) { tracker in tracker.installEvent = false tracker.lifecycleEvents = true @@ -137,6 +137,8 @@ class TestSession: XCTestCase { let session = tracker.session session?.updateInBackground() // It sends a background event + + Thread.sleep(forTimeInterval: 1) let sessionId = session?.sessionId @@ -184,7 +186,7 @@ class TestSession: XCTestCase { func testMixedEventsOnManySessions() { cleanFile(withNamespace: "t2") - let emitter = Emitter(namespace: "t2", urlEndpoint: "") + let emitter = Emitter(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), namespace: "t2") let tracker = Tracker(trackerNamespace: "t2", appId: nil, emitter: emitter) { tracker in tracker.lifecycleEvents = true tracker.sessionContext = true @@ -267,7 +269,7 @@ class TestSession: XCTestCase { func testBackgroundTimeBiggerThanBackgroundTimeoutCausesNewSession() { cleanFile(withNamespace: "tracker") - let emitter = Emitter(namespace: "tracker", urlEndpoint: "") + let emitter = Emitter(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), namespace: "tracker") let tracker = Tracker(trackerNamespace: "tracker", appId: nil, emitter: emitter) { tracker in tracker.lifecycleEvents = true tracker.sessionContext = true @@ -287,6 +289,7 @@ class TestSession: XCTestCase { session?.updateInBackground() // Sends a background event Thread.sleep(forTimeInterval: 3) // Bigger than background timeout session?.updateInForeground() // Sends a foreground event + Thread.sleep(forTimeInterval: 1) XCTAssertEqual(oldSessionId, session?.previousSessionId) XCTAssertEqual(2, session?.sessionIndex) @@ -298,7 +301,7 @@ class TestSession: XCTestCase { func testBackgroundTimeSmallerThanBackgroundTimeoutDoesntCauseNewSession() { cleanFile(withNamespace: "tracker") - let emitter = Emitter(namespace: "tracker", urlEndpoint: "") + let emitter = Emitter(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), namespace: "tracker") let tracker = Tracker(trackerNamespace: "tracker", appId: nil, emitter: emitter) { tracker in tracker.lifecycleEvents = true tracker.sessionContext = true @@ -318,6 +321,7 @@ class TestSession: XCTestCase { session?.updateInBackground() // Sends a background event Thread.sleep(forTimeInterval: 1) // Smaller than background timeout session?.updateInForeground() // Sends a foreground event + Thread.sleep(forTimeInterval: 1) XCTAssertEqual(oldSessionId, session?.sessionId) XCTAssertEqual(1, session?.sessionIndex) @@ -343,21 +347,21 @@ class TestSession: XCTestCase { cleanFile(withNamespace: "tracker1") cleanFile(withNamespace: "tracker2") - let emitter1 = Emitter(namespace: "tracker1", urlEndpoint: "") + let emitter1 = Emitter(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), namespace: "tracker1") let tracker1 = Tracker(trackerNamespace: "tracker1", appId: nil, emitter: emitter1) { tracker in tracker.sessionContext = true tracker.foregroundTimeout = 10 tracker.backgroundTimeout = 10 } - let emitter2 = Emitter(namespace: "tracker2", urlEndpoint: "") + let emitter2 = Emitter(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), namespace: "tracker2") let tracker2 = Tracker(trackerNamespace: "tracker2", appId: nil, emitter: emitter2) { tracker in tracker.sessionContext = true tracker.foregroundTimeout = 10 tracker.backgroundTimeout = 10 } let event = Structured(category: "c", action: "a") - _ = tracker1.track(event) - _ = tracker2.track(event) + track(event, tracker1) + track(event, tracker2) guard let initialValue1 = tracker1.session?.sessionIndex else { return XCTFail() } guard let id1 = tracker1.session?.sessionId else { return XCTFail() } @@ -366,11 +370,11 @@ class TestSession: XCTestCase { // Retrigger session in tracker1 Thread.sleep(forTimeInterval: 7) - _ = tracker1.track(event) + track(event, tracker1) Thread.sleep(forTimeInterval: 5) // Send event to force update of session on tracker2 - _ = tracker2.track(event) + track(event, tracker2) id2 = tracker2.session!.sessionId! // Check sessions have the correct state @@ -383,7 +387,7 @@ class TestSession: XCTestCase { tracker.foregroundTimeout = 5 tracker.backgroundTimeout = 5 } - _ = tracker2b.track(event) + track(event, tracker2b) guard let initialValue2b = tracker2b.session?.sessionIndex else { return XCTFail() } guard let previousId2b = tracker2b.session?.previousSessionId else { return XCTFail() } @@ -397,12 +401,12 @@ class TestSession: XCTestCase { cleanFile(withNamespace: "tracker") storeAsV3_0(withNamespace: "tracker", eventId: "eventId", sessionId: "sessionId", sessionIndex: 123, userId: "userId") - let emitter = Emitter(namespace: "tracker", urlEndpoint: "") + let emitter = Emitter(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 500), namespace: "tracker") let tracker = Tracker(trackerNamespace: "tracker", appId: nil, emitter: emitter) { tracker in tracker.sessionContext = true } let event = Structured(category: "c", action: "a") - _ = tracker.track(event) + track(event, tracker) guard let session = tracker.session else { return XCTFail() } XCTAssertEqual("sessionId", session.previousSessionId!) @@ -467,4 +471,10 @@ class TestSession: XCTestCase { let userDefaults = UserDefaults.standard userDefaults.set(userId, forKey: kSPInstallationUserId) } + + private func track(_ event: Event, _ tracker: Tracker) { + InternalQueue.sync { + _ = tracker.track(event) + } + } } diff --git a/Tests/TestStateManager.swift b/Tests/TestStateManager.swift index 83d3690c2..10bad847e 100644 --- a/Tests/TestStateManager.swift +++ b/Tests/TestStateManager.swift @@ -244,22 +244,4 @@ class TestStateManager: XCTestCase { ) ) } - - @available(iOS 13, macOS 10.15, watchOS 6, tvOS 13, *) - func testConcurrentRemoveStateMachineWithAddOrReplaceStateMachine() async throws { - let stateManager = StateManager() - await withTaskGroup(of: Task.self) { group in - (1...100).forEach { element in - group.addTask { - Task.detached { - if Int(element).isMultiple(of: 2) { - _ = stateManager.removeStateMachine("MockStateMachine-»\(element-1)") - } else { - stateManager.addOrReplaceStateMachine(MockStateMachine("MockStateMachine-»\(element)")) - } - } - } - } - } - } } diff --git a/Tests/Tracker/TestTracker.swift b/Tests/Tracker/TestTracker.swift index d6eaa60f9..0cd71644c 100644 --- a/Tests/Tracker/TestTracker.swift +++ b/Tests/Tracker/TestTracker.swift @@ -28,6 +28,42 @@ class TestTracker: XCTestCase { tracker.sessionContext = true } } + + func testTrackerPayload() { + let subject = Subject(platformContext: true, geoLocationContext: true) + let emitter = Emitter(namespace: "aNamespace", urlEndpoint: "not-real.com") + + let tracker = Tracker(trackerNamespace: "aNamespace", appId: "anAppId", emitter: emitter) { tracker in + tracker.subject = subject + tracker.devicePlatform = .general + tracker.base64Encoded = false + tracker.sessionContext = true + tracker.foregroundTimeout = 300 + tracker.backgroundTimeout = 150 + } + + let event = Structured(category: "Category", action: "Action") + let trackerEvent = TrackerEvent(event: event, state: nil) + + var payload = tracker.payload(with: trackerEvent) + + var payloadDict = payload!.dictionary + + XCTAssertEqual(payloadDict[kSPPlatform] as? String, devicePlatformToString(.general)) + XCTAssertEqual(payloadDict[kSPAppId] as? String, "anAppId") + XCTAssertEqual(payloadDict[kSPNamespace] as? String, "aNamespace") + + // Test setting variables to new values + + tracker.devicePlatform = .desktop + tracker.appId = "newAppId" + + payload = tracker.payload(with: trackerEvent) + payloadDict = payload!.dictionary + + XCTAssertEqual(payloadDict[kSPPlatform] as? String, "pc") + XCTAssertEqual(payloadDict[kSPAppId] as? String, "newAppId") + } func testTrackerBuilderAndOptions() { let eventSink = EventSink() @@ -61,7 +97,7 @@ class TestTracker: XCTestCase { tracker.pauseEventTracking() XCTAssertEqual(tracker.isTracking, false) - _ = tracker.track(Structured(category: "c", action: "a")) + track(Structured(category: "c", action: "a"), tracker) tracker.resumeEventTracking() XCTAssertEqual(tracker.isTracking, true) @@ -70,7 +106,7 @@ class TestTracker: XCTestCase { XCTAssertEqual(eventSink.trackedEvents.count, 0) // tracks event after tracking resumed - _ = tracker.track(Structured(category: "c", action: "a")) + track(Structured(category: "c", action: "a"), tracker) Thread.sleep(forTimeInterval: 0.5) XCTAssertEqual(eventSink.trackedEvents.count, 1) @@ -98,5 +134,11 @@ class TestTracker: XCTestCase { XCTAssertNotNil(tracker.session) XCTAssertFalse(oldSessionManager === tracker.session) } + + private func track(_ event: Event, _ tracker: Tracker) { + InternalQueue.sync { + _ = tracker.track(event) + } + } } diff --git a/Tests/Tracker/TestTrackerPayloadBuilder.swift b/Tests/Tracker/TestTrackerPayloadBuilder.swift deleted file mode 100644 index 406ebcfb8..000000000 --- a/Tests/Tracker/TestTrackerPayloadBuilder.swift +++ /dev/null @@ -1,59 +0,0 @@ -// Copyright (c) 2013-2023 Snowplow Analytics Ltd. All rights reserved. -// -// This program is licensed to you under the Apache License Version 2.0, -// and you may not use this file except in compliance with the Apache License -// Version 2.0. You may obtain a copy of the Apache License Version 2.0 at -// http://www.apache.org/licenses/LICENSE-2.0. -// -// Unless required by applicable law or agreed to in writing, -// software distributed under the Apache License Version 2.0 is distributed on -// an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either -// express or implied. See the Apache License Version 2.0 for the specific -// language governing permissions and limitations there under. - -import Foundation - -import XCTest -@testable import SnowplowTracker - -class TestTrackerPayloadBuilder: XCTestCase { - - func testTrackerPayload() { - let subject = Subject(platformContext: true, geoLocationContext: true) - - let trackerData = TrackerData(appId: "anAppId", trackerNamespace: "aNamespace") - trackerData.subject = subject - trackerData.devicePlatform = .general - trackerData.base64Encoded = false - trackerData.sessionContext = true - trackerData.foregroundTimeout = 300 - trackerData.backgroundTimeout = 150 - - let event = Structured(category: "Category", action: "Action") - let trackerEvent = TrackerEvent(event: event, state: nil) - let stateManager = StateManager() - - let payloadBuilder = TrackerPayloadBuilder() - var payload = payloadBuilder.payload(event: trackerEvent, tracker: trackerData, stateManager: stateManager) - - var payloadDict = payload!.dictionary - - XCTAssertEqual(payloadDict[kSPPlatform] as? String, devicePlatformToString(.general)) - XCTAssertEqual(payloadDict[kSPAppId] as? String, "anAppId") - XCTAssertEqual(payloadDict[kSPNamespace] as? String, "aNamespace") - - // Test setting variables to new values - - trackerData.devicePlatform = .desktop - trackerData.appId = "newAppId" - trackerData.trackerNamespace = "newNamespace" - - payload = payloadBuilder.payload(event: trackerEvent, tracker: trackerData, stateManager: stateManager) - payloadDict = payload!.dictionary - - XCTAssertEqual(payloadDict[kSPPlatform] as? String, "pc") - XCTAssertEqual(payloadDict[kSPAppId] as? String, "newAppId") - XCTAssertEqual(payloadDict[kSPNamespace] as? String, "newNamespace") - } -} -//#pragma clang diagnostic pop diff --git a/Tests/Utils/MockEventStore.swift b/Tests/Utils/MockEventStore.swift index 9c5053e60..eb7a93600 100644 --- a/Tests/Utils/MockEventStore.swift +++ b/Tests/Utils/MockEventStore.swift @@ -26,16 +26,12 @@ class MockEventStore: NSObject, EventStore { } func addEvent(_ payload: Payload) { - objc_sync_enter(self) lastInsertedRow += 1 logVerbose(message: "Add \(payload)") db[Int64(lastInsertedRow)] = payload - objc_sync_exit(self) } func removeEvent(withId storeId: Int64) -> Bool { - objc_sync_enter(self) - defer { objc_sync_exit(self) } logVerbose(message: "Remove \(storeId)") return db.removeValue(forKey: storeId) != nil } @@ -49,22 +45,16 @@ class MockEventStore: NSObject, EventStore { } func removeAllEvents() -> Bool { - objc_sync_enter(self) db.removeAll() lastInsertedRow = -1 - objc_sync_exit(self) return true } func count() -> UInt { - objc_sync_enter(self) - defer { objc_sync_exit(self) } return UInt(db.count) } func emittableEvents(withQueryLimit queryLimit: UInt) -> [EmitterEvent] { - objc_sync_enter(self) - defer { objc_sync_exit(self) } var eventIds: [Int64] = [] var events: [EmitterEvent] = [] for (key, obj) in db { diff --git a/Tests/Utils/MockTimer.swift b/Tests/Utils/MockTimer.swift index 47da12c45..78b622478 100644 --- a/Tests/Utils/MockTimer.swift +++ b/Tests/Utils/MockTimer.swift @@ -12,21 +12,27 @@ // language governing permissions and limitations there under. import Foundation +@testable import SnowplowTracker -class MockTimer: Timer { +class MockTimer: InternalQueueTimer { - var block: ((Timer) -> Void)! + var block: (() -> Void) + + init(block: @escaping () -> Void) { + self.block = block + } static var currentTimer: MockTimer! - override func fire() { - block(self) + func fire() { + InternalQueue.sync { + block() + } } - override open class func scheduledTimer(withTimeInterval interval: TimeInterval, - repeats: Bool, - block: @escaping (Timer) -> Void) -> Timer { - let mockTimer = MockTimer() + static func startTimer(_ interval: TimeInterval, + _ block: @escaping () -> Void) -> InternalQueueTimer { + let mockTimer = MockTimer(block: block) mockTimer.block = block MockTimer.currentTimer = mockTimer