From 6fcf905e0809fd00f22a6fa2e0c7f80db18e8b86 Mon Sep 17 00:00:00 2001 From: Greg Leonard <45019882+greg-el@users.noreply.github.com> Date: Wed, 13 Sep 2023 16:33:08 +0100 Subject: [PATCH] Add API to decorate link with user/session info --- Sources/Core/Tracker/Tracker.swift | 75 ++++++++ .../Core/Tracker/TrackerControllerImpl.swift | 8 + .../Controllers/TrackerController.swift | 33 ++++ .../CrossDeviceParameterConfiguration.swift | 47 +++++ Sources/Snowplow/Utils/Stringb64.swift | 34 ++++ Tests/TestLinkDecorator.swift | 181 ++++++++++++++++++ 6 files changed, 378 insertions(+) create mode 100644 Sources/Snowplow/Tracker/CrossDeviceParameterConfiguration.swift create mode 100644 Sources/Snowplow/Utils/Stringb64.swift create mode 100644 Tests/TestLinkDecorator.swift diff --git a/Sources/Core/Tracker/Tracker.swift b/Sources/Core/Tracker/Tracker.swift index ee9dffa75..c9555aeee 100644 --- a/Sources/Core/Tracker/Tracker.swift +++ b/Sources/Core/Tracker/Tracker.swift @@ -385,7 +385,82 @@ class Tracker: NSObject { emitter.resumeTimer() session?.startChecker() } + + /// - Returns: The associated value of the extended parameter if enabled and set in the tracker, an empty string if the parameter is not enabled, or nil if parameter is enabled without an aviliable value + func tryExtendedParameter(_ extendedParameterSet: Bool, _ value: String?) -> String? { + switch (extendedParameterSet, value) { + case (true, .some(let val)): + return val + case (true, .none): + return nil + case (false, _): + return "" + } + } + + func decorateLinkErrorTemplate(_ extendedParameterName: String) -> String { + "Cannot decorate link: \(extendedParameterName) has been requested in CrossDeviceParameterConfiguration, but it is not set." + } + + @objc func decorateLink(_ url: URL, extendedParameters: CrossDeviceParameterConfiguration? = nil) -> URL? { + var userId: String + switch self.session { + case .none: + logError(message: "\(url) could not be decorated as session.userId is nil") + return nil + case .some(let s): + userId = s.userId + } + + let extendedParameters = extendedParameters ?? CrossDeviceParameterConfiguration() + + guard let sessionId = tryExtendedParameter(extendedParameters.sessionId, self.session?.state?.sessionId) else { + logError(message: "\(decorateLinkErrorTemplate("sessionId")) Ensure an event has been tracked to generate a session before calling this method.") + return nil + } + + guard let sourceId = tryExtendedParameter(extendedParameters.sourceId, self.appId) else { + logError(message: decorateLinkErrorTemplate("appId")) + return nil + } + + guard let sourcePlatform = tryExtendedParameter(extendedParameters.sourcePlatform, devicePlatformToString(self.devicePlatform)) else { + logError(message: decorateLinkErrorTemplate("sourcePlatform")) + return nil + } + + guard let subjectUserId = tryExtendedParameter(extendedParameters.subjectUserId, self.subject?.userId) else { + logError(message: "\(decorateLinkErrorTemplate("subjectUserId")) Ensure SubjectConfiguration.userId has been passed set on your tracker.") + return nil + } + + let reason = extendedParameters.reason ?? "" + + let spParameters = [ + userId, + String(Int(Date().timeIntervalSince1970 * 1000)), + sessionId, + subjectUserId.toBase64(), + sourceId.toBase64(), + sourcePlatform, + reason.toBase64() + ].joined(separator: ".").trimmingCharacters(in: ["."]) + + var components = URLComponents(url: url, resolvingAgainstBaseURL: false) + let spQueryParam = URLQueryItem(name: "_sp", value: spParameters) + + if let index = components?.queryItems?.firstIndex(where: { $0.name == "_sp" }) { + // Replace the old query item with the new one + components?.queryItems?[index] = spQueryParam + } else { + let queryItems = components?.queryItems ?? [] + components?.queryItems = queryItems + [spQueryParam] + } + + return components?.url + } + // MARK: - Notifications management @objc func receiveScreenViewNotification(_ notification: Notification) { diff --git a/Sources/Core/Tracker/TrackerControllerImpl.swift b/Sources/Core/Tracker/TrackerControllerImpl.swift index afc0a5f90..3f90e7520 100644 --- a/Sources/Core/Tracker/TrackerControllerImpl.swift +++ b/Sources/Core/Tracker/TrackerControllerImpl.swift @@ -73,6 +73,14 @@ class TrackerControllerImpl: Controller, TrackerController { func track(_ event: Event) -> UUID? { return tracker.track(event) } + + func decorateLink(_ url: URL) -> URL? { + return tracker.decorateLink(url) + } + + func decorateLink(_ url: URL, extendedParameters: CrossDeviceParameterConfiguration) -> URL? { + return tracker.decorateLink(url, extendedParameters: extendedParameters) + } // MARK: - Properties' setters and getters diff --git a/Sources/Snowplow/Controllers/TrackerController.swift b/Sources/Snowplow/Controllers/TrackerController.swift index 3c5fcdf4c..b81ed240b 100644 --- a/Sources/Snowplow/Controllers/TrackerController.swift +++ b/Sources/Snowplow/Controllers/TrackerController.swift @@ -75,4 +75,37 @@ public protocol TrackerController: TrackerConfigurationProtocol { /// The tracker will start tracking again. @objc func resume() + /// Adds user and session information to a URL. + /// + /// For example, calling decorateLink on `appSchema://path/to/page` will return: + /// + /// `appSchema://path/to/page?_sp=domainUserId.timestamp.sessionId..sourceId` + /// + /// Filled by this method: + /// - `domainUserId`: Value of ``SessionController.userId`` + /// - `timestamp`: ms precision epoch timestamp + /// - `sessionId`: Value of ``SessionController.sessionId`` + /// - `sourceId`: Value of ``Tracker.appId`` + /// + /// - Parameter uri The URI to add the query string to + /// + /// - Returns Optional URL + /// - null if ``SnowplowTracker/SessionController/userId`` is null from `sessionContext(false)` being passed in ``TrackerConfiguration`` + /// - otherwise, decorated URL + @objc + func decorateLink(_ url: URL) -> URL? + /// Adds user and session information to a URL. + /// + /// For example, calling decorateLink on `appSchema://path/to/page` with all extended parameters enabled will return: + /// + /// `appSchema://path/to/page?_sp=domainUserId.timestamp.sessionId.subjectUserId.sourceId.platform.reason` + /// + /// - Parameter url The URL to add the query string to + /// - Parameter extendedParameters Any optional parameters to include in the query string. + /// + /// - Returns Optional URL + /// - null if ``SnowplowTracker/SessionController/userId`` is null from `sessionContext(false)` being passed in ``TrackerConfiguration`` + /// - otherwise, decorated URL + @objc + func decorateLink(_ url: URL, extendedParameters: CrossDeviceParameterConfiguration) -> URL? } diff --git a/Sources/Snowplow/Tracker/CrossDeviceParameterConfiguration.swift b/Sources/Snowplow/Tracker/CrossDeviceParameterConfiguration.swift new file mode 100644 index 000000000..ac226555a --- /dev/null +++ b/Sources/Snowplow/Tracker/CrossDeviceParameterConfiguration.swift @@ -0,0 +1,47 @@ +// 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 + +/// +/// Configuration object for ``TrackerController/decorateLink`` +/// +/// Whether to include the following values when decorating a link: +/// - `sessionId`: Value of ``SessionController.sessionId`` +/// - `subjectUserId`: Value of ``Subject.userId`` +/// - `sourceId`: Value of ``Tracker.appId`` +/// - `platform`: Value of ``Tracker.platform`` +/// - `reason`: Optional identifier/information for cross-navigation +/// +/// ``sourceId`` and ``sessionId`` are enabled by default. +public class CrossDeviceParameterConfiguration : NSObject { + var sessionId: Bool + var subjectUserId: Bool + var sourceId: Bool + var sourcePlatform: Bool + var reason: String? + + init( + sessionId: Bool = true, + subjectUserId: Bool = false, + sourceId: Bool = true, + sourcePlatform: Bool = false, + reason: String? = nil + ) { + self.sessionId = sessionId + self.subjectUserId = subjectUserId + self.sourceId = sourceId + self.sourcePlatform = sourcePlatform + self.reason = reason + } +} diff --git a/Sources/Snowplow/Utils/Stringb64.swift b/Sources/Snowplow/Utils/Stringb64.swift new file mode 100644 index 000000000..ad0f69fca --- /dev/null +++ b/Sources/Snowplow/Utils/Stringb64.swift @@ -0,0 +1,34 @@ + +// 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 + +public extension String { + func toBase64(urlSafe: Bool = true) -> String { + var encoded = Data(self.utf8).base64EncodedString() + if urlSafe { + // We need URL safe with no padding. Since there is no built-in way to do this, we transform + // the encoded payload to make it URL safe by replacing chars that are different in the URL-safe + // alphabet. Namely, 62 is - instead of +, and 63 _ instead of /. + // See: https://tools.ietf.org/html/rfc4648#section-5 + encoded = encoded + .replacingOccurrences(of: "/", with: "_") + .replacingOccurrences(of: "+", with: "-") + + // There is also no padding since the length is implicitly known. + encoded = encoded.trimmingCharacters(in: CharacterSet(charactersIn: "=")) + } + return encoded + } +} diff --git a/Tests/TestLinkDecorator.swift b/Tests/TestLinkDecorator.swift new file mode 100644 index 000000000..d17f75748 --- /dev/null +++ b/Tests/TestLinkDecorator.swift @@ -0,0 +1,181 @@ +// 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 XCTest +@testable import SnowplowTracker + +class TestLinkDecorator: XCTestCase { + let epoch = "\\d{13}" + + let replacements = [".", "/", "?"] + func matches(for regex: String, in text: String) { + var regex = "^\(regex)$" + + do { + for replacement in replacements { + regex = regex.replacingOccurrences(of: replacement, with: "\\" + replacement) + } + let pattern = try NSRegularExpression(pattern: regex) + let nsString = text as NSString + let results = pattern.matches(in: text, range: NSRange(location: 0, length: nsString.length)) + let fullMatch = results.map { nsString.substring(with: $0.range)} + if (fullMatch.count == 0) { + XCTFail("URL does not match pattern:\n\(text)\n\(regex)") + } + XCTAssertEqual(fullMatch.count, 1) + } catch let error { + print("invalid regex: \(error.localizedDescription)") + } + } + + func testParameterConfiguration() { + let tracker = getTracker() + let _ = tracker.track(ScreenView(name: "test")) + + let link = URL(string: "https://example.com")! + let userId = tracker.session!.userId! + let sessionId = tracker.session!.sessionId! + let subjectUserId = tracker.subject!.userId!.toBase64() + let appId = tracker.appId.toBase64() + let platform = devicePlatformToString(tracker.devicePlatform)! + let reason = "reason" + let reasonb64 = reason.toBase64() + + let testCases = [ + // All false + ( + config: CrossDeviceParameterConfiguration(sessionId: false, sourceId: false), + expected: "https://example.com?_sp=\(userId).\(epoch)" + ) + , + // Default + ( + config: CrossDeviceParameterConfiguration(), + expected: "https://example.com?_sp=\(userId).\(epoch).\(sessionId)..\(appId)" + ), + ( + config: CrossDeviceParameterConfiguration(subjectUserId: true), + expected: "https://example.com?_sp=\(userId).\(epoch).\(sessionId).\(subjectUserId).\(appId)" + ), + ( + config: CrossDeviceParameterConfiguration(subjectUserId: true, sourcePlatform: true), + expected: "https://example.com?_sp=\(userId).\(epoch).\(sessionId).\(subjectUserId).\(appId).\(platform)" + ), + ( + config: CrossDeviceParameterConfiguration(subjectUserId: true, sourcePlatform: true, reason: reason), + expected: "https://example.com?_sp=\(userId).\(epoch).\(sessionId).\(subjectUserId).\(appId).\(platform).\(reasonb64)" + ), + ( + config: CrossDeviceParameterConfiguration(sessionId: false, sourceId: false, reason: reason), + expected: "https://example.com?_sp=\(userId).\(epoch).....\(reasonb64)" + ) + ] + + for (config, expected) in testCases { + matches( + for: expected, + in: tracker.decorateLink(link, extendedParameters: config)!.absoluteString + ) + } + } + + func testWithExistingSpQueryParameter() { + let tracker = getTracker() + let link = URL(string: "https://example.com?_sp=test")! + let userId = tracker.session!.userId! + + let result = tracker.decorateLink(link, extendedParameters: CrossDeviceParameterConfiguration(sessionId: false, sourceId: false))! + + matches(for: "https://example.com?_sp=\(userId).\(epoch)", in: result.absoluteString) + } + + func testWithOtherQueryParameters() { + let tracker = getTracker() + let link = URL(string: "https://example.com?a=a&b=b")! + let userId = tracker.session!.userId! + + let result = tracker.decorateLink(link, extendedParameters: CrossDeviceParameterConfiguration(sessionId: false, sourceId: false))! + + matches(for: "https://example.com?a=a&b=b&_sp=\(userId).\(epoch)", in: result.absoluteString) + } + + func testExistingSpQueryParameterInMiddleOfOtherQueryParameters() { + let tracker = getTracker() + let link = URL(string: "https://example.com?a=a&_sp=test&b=b")! + let userId = tracker.session!.userId! + + let result = tracker.decorateLink(link, extendedParameters: CrossDeviceParameterConfiguration(sessionId: false, sourceId: false))! + + matches(for: "https://example.com?a=a&_sp=\(userId).\(epoch)&b=b", in: result.absoluteString) + } + + func testErrors() { + let tracker = getTrackerNoSessionOrSubject() + let link = URL(string: "https://example.com")! + + let testCases = [ + // sessionId wont be set + CrossDeviceParameterConfiguration(sessionId: true), + // Subject.userId wont be set + CrossDeviceParameterConfiguration(sessionId: false, subjectUserId: true), + ] + + for config in testCases { + let result = tracker.decorateLink(link, extendedParameters: config) + XCTAssertNil(result) + } + } + + + + func getTracker() -> TrackerController { + let networkConnection = MockNetworkConnection(requestOption: .post, statusCode: 200) + + let emitterConfig = EmitterConfiguration() + emitterConfig.eventStore = MockEventStore() + emitterConfig.bufferOption = .single + + let networkConfig = NetworkConfiguration(networkConnection: networkConnection) + + let trackerConfig = TrackerConfiguration().sessionContext(true) + trackerConfig.installAutotracking = false + trackerConfig.screenViewAutotracking = false + trackerConfig.lifecycleAutotracking = false + + let subjectConfig = SubjectConfiguration().userId("userId") + + let namespace = "testEmitter" + String(describing: Int.random(in: 0..<100)) + return Snowplow.createTracker(namespace: namespace, + network: networkConfig, + configurations: [trackerConfig, emitterConfig, subjectConfig])! + } + + private func getTrackerNoSessionOrSubject() -> TrackerController { + let emitterConfig = EmitterConfiguration() + emitterConfig.eventStore = MockEventStore() + emitterConfig.bufferOption = .single + + let networkConfig = NetworkConfiguration(networkConnection: MockNetworkConnection(requestOption: .post, statusCode: 200)) + + let trackerConfig = TrackerConfiguration() + trackerConfig.installAutotracking = false + trackerConfig.screenViewAutotracking = false + trackerConfig.lifecycleAutotracking = false + + let namespace = "testEmitter" + String(describing: Int.random(in: 0..<100)) + return Snowplow.createTracker(namespace: namespace, + network: networkConfig, + configurations: [trackerConfig, emitterConfig])! + } + +}