-
Notifications
You must be signed in to change notification settings - Fork 93
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Add API to decorate link with user/session info
- Loading branch information
Showing
6 changed files
with
321 additions
and
1 deletion.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
36 changes: 36 additions & 0 deletions
36
Sources/Snowplow/Tracker/CrossDeviceParameterConfiguration.swift
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,36 @@ | ||
// 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 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 | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,113 @@ | ||
// 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 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 testArgs() { | ||
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 epoch = "\\d{13}" | ||
let appId = tracker.appId.toBase64() | ||
let subjectUserId = tracker.subject!.userId!.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 getTracker() -> TrackerController { | ||
let networkConnection = MockNetworkConnection(requestOption: .post, statusCode: 200) | ||
let emitterConfig = EmitterConfiguration() | ||
emitterConfig.eventStore = MockEventStore() | ||
emitterConfig.bufferOption = .single | ||
let networkConfig = NetworkConfiguration(networkConnection: networkConnection) | ||
|
||
return createTracker(networkConfig: networkConfig, emitterConfig: emitterConfig) | ||
} | ||
|
||
private func createTracker(networkConfig: NetworkConfiguration, emitterConfig: EmitterConfiguration) -> TrackerController { | ||
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])! | ||
} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,19 @@ | ||
import Foundation | ||
|
||
extension String { | ||
func toBase64() -> String { | ||
var encoded = Data(self.utf8).base64EncodedString() | ||
// 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 | ||
} | ||
} |