From 216a00e6f78f40d13ee31ccc3c9179f2deb0bfca Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 13:05:41 +0100 Subject: [PATCH 01/49] implement experiment manager --- .../ExperimentCohortsManager.swift | 130 ++++++++ .../PrivacyConfigurationData.swift | 23 ++ .../ExperimentCohortsManagerTests.swift | 287 ++++++++++++++++++ .../PrivacyConfigurationDataTests.swift | 3 + .../Resources/privacy-config-example.json | 17 +- 5 files changed, 459 insertions(+), 1 deletion(-) create mode 100644 Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift create mode 100644 Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift new file mode 100644 index 000000000..74d648e0c --- /dev/null +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -0,0 +1,130 @@ +// +// ExperimentCohortsManager.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import Foundation + +struct ExperimentSubfeature { + let subfeatureID: SubfeatureID + let cohorts: [PrivacyConfigurationData.Cohort] +} + +typealias CohortID = String +typealias SubfeatureID = String + +struct ExperimentData: Codable { + let cohort: String + let enrollmentDate: Date +} + +typealias Experiments = [String: ExperimentData] + +protocol ExperimentCohortsManaging { + /// Retrieves the cohort ID associated with the specified subfeature. + /// - Parameter subfeature: The experiment subfeature for which the cohort ID is needed. + /// - Returns: The cohort ID as a `String` if one exists; otherwise, returns `nil`. + func cohort(for subfeatureID: SubfeatureID) -> CohortID? + + /// Retrieves the enrollment date for the specified subfeature. + /// - Parameter subfeatureID: The experiment subfeature for which the enrollment date is needed. + /// - Returns: The `Date` of enrollment if one exists; otherwise, returns `nil`. + func enrolmentDate(for subfeatureID: SubfeatureID) -> Date? + + /// Assigns a cohort to the given subfeature based on defined weights and saves it to UserDefaults. + /// - Parameter subfeature: The experiment subfeature to assign a cohort for. + /// - Returns: The name of the assigned cohort, or `nil` if no cohort could be assigned. + func assignCohort(for subfeature: ExperimentSubfeature) -> CohortID? + + /// Removes the assigned cohort data for the specified subfeature. + /// - Parameter subfeature: The experiment subfeature for which the cohort data should be removed. + func removeCohort(for subfeatureID: SubfeatureID) +} + +struct ExperimentCohortsManager: ExperimentCohortsManaging { + + private let userDefaults: UserDefaults + private let decoder = JSONDecoder() + private let encoder = JSONEncoder() + private let queue = DispatchQueue(label: "com.experimentManager.queue") + private let experimentsDataKey = "ExperimentsData" + private let randomizer: (Range) -> Double + + init(userDefaults: UserDefaults = UserDefaults.standard, randomizer: @escaping (Range) -> Double) { + self.userDefaults = userDefaults + self.randomizer = randomizer + encoder.dateEncodingStrategy = .secondsSince1970 + decoder.dateDecodingStrategy = .secondsSince1970 + } + + func cohort(for subfeatureID: SubfeatureID) -> CohortID? { + guard let experiments = getExperimentData() else { return nil } + return experiments[subfeatureID]?.cohort + } + + func enrolmentDate(for subfeatureID: SubfeatureID) -> Date? { + guard let experiments = getExperimentData() else { return nil } + return experiments[subfeatureID]?.enrollmentDate + } + + func assignCohort(for subfeature: ExperimentSubfeature) -> CohortID? { + let cohorts = subfeature.cohorts + let totalWeight = cohorts.reduce(0, { $0 + $1.weight }) + guard totalWeight > 0 else { return nil } + + let randomValue = randomizer(0.. Experiments? { + queue.sync { + guard let savedData = userDefaults.data(forKey: experimentsDataKey) else { return nil } + return try? decoder.decode(Experiments.self, from: savedData) + } + } + + private func saveExperimentData(_ experiments: Experiments) { + queue.sync { + if let encodedData = try? encoder.encode(experiments) { + userDefaults.set(encodedData, forKey: experimentsDataKey) + } + } + } + + private func saveCohort(_ cohort: CohortID, in experimentID: SubfeatureID) { + var experiments = getExperimentData() ?? Experiments() + let experimentData = ExperimentData(cohort: cohort, enrollmentDate: Date()) + experiments[experimentID] = experimentData + saveExperimentData(experiments) + } +} + diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift index 813c87503..eb9267505 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift @@ -36,6 +36,20 @@ public struct PrivacyConfigurationData { static public let enabled = "enabled" } + public struct Cohort { + public let name: String + public let weight: Int + + public init?(json: [String: Any]) { + guard let name = json["name"] as? String, + let weightValue = json["weight"] as? Int else { + return nil + } + + self.name = name + self.weight = weightValue + } + } public let features: [FeatureName: PrivacyFeature] public let trackerAllowlist: TrackerAllowlist public let unprotectedTemporary: [ExceptionEntry] @@ -121,6 +135,7 @@ public struct PrivacyConfigurationData { case state case minSupportedVersion case rollout + case cohorts } public struct Rollout: Hashable { @@ -157,6 +172,7 @@ public struct PrivacyConfigurationData { public let state: FeatureState public let minSupportedVersion: FeatureSupportedVersion? public let rollout: Rollout? + public let cohorts: [Cohort]? public init?(json: [String: Any]) { guard let state = json[CodingKeys.state.rawValue] as? String else { @@ -171,6 +187,13 @@ public struct PrivacyConfigurationData { } else { self.rollout = nil } + + if let cohortData = json[CodingKeys.cohorts.rawValue] as? [[String: Any]] { + let parsedCohorts = cohortData.compactMap { Cohort(json: $0) } + self.cohorts = parsedCohorts.isEmpty ? nil : parsedCohorts + } else { + self.cohorts = nil + } } } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift new file mode 100644 index 000000000..f3e68a2d6 --- /dev/null +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -0,0 +1,287 @@ +// +// ExperimentCohortsManagerTests.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import XCTest +@testable import BrowserServicesKit + +final class ExperimentCohortsManagerTests: XCTestCase { + + var mockUserDefaults: UserDefaults! + var experimentCohortsManager: ExperimentCohortsManager! + + let subfeatureName1 = "TestSubfeature1" + var expectedDate1: Date! + var experimentData1: ExperimentData! + + let subfeatureName2 = "TestSubfeature2" + var expectedDate2: Date! + var experimentData2: ExperimentData! + + let encoder: JSONEncoder = { + let encoder = JSONEncoder() + encoder.dateEncodingStrategy = .secondsSince1970 + return encoder + }() + + override func setUp() { + super.setUp() + mockUserDefaults = UserDefaults(suiteName: "com.test.ExperimentCohortsManagerTests") + mockUserDefaults.removePersistentDomain(forName: "com.test.ExperimentCohortsManagerTests") + + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { _ in 50.0 } + ) + + expectedDate1 = Date() + experimentData1 = ExperimentData(cohort: "TestCohort1", enrollmentDate: expectedDate1) + + expectedDate2 = Date().addingTimeInterval(60) // Second subfeature with a different date + experimentData2 = ExperimentData(cohort: "TestCohort2", enrollmentDate: expectedDate2) + } + + override func tearDown() { + mockUserDefaults.removePersistentDomain(forName: "com.test.ExperimentCohortsManagerTests") + mockUserDefaults = nil + experimentCohortsManager = nil + expectedDate1 = nil + experimentData1 = nil + expectedDate2 = nil + experimentData2 = nil + super.tearDown() + } + + private func saveExperimentData(_ data: [String: ExperimentData]) { + if let encodedData = try? encoder.encode(data) { + mockUserDefaults.set(encodedData, forKey: "ExperimentsData") + } + } + + + func testCohortReturnsCohortIDIfExistsForMultipleSubfeatures() { + // GIVEN + saveExperimentData([subfeatureName1: experimentData1, subfeatureName2: experimentData2]) + + // WHEN + let result1 = experimentCohortsManager.cohort(for: subfeatureName1) + let result2 = experimentCohortsManager.cohort(for: subfeatureName2) + + // THEN + XCTAssertEqual(result1, experimentData1.cohort) + XCTAssertEqual(result2, experimentData2.cohort) + } + + func testEnrolmentDateReturnsCorrectDateIfExists() { + // GIVEN + saveExperimentData([subfeatureName1: experimentData1]) + + // WHEN + let result1 = experimentCohortsManager.enrolmentDate(for: subfeatureName1) + let result2 = experimentCohortsManager.enrolmentDate(for: subfeatureName2) + + // THEN + let timeDifference1 = abs(expectedDate1.timeIntervalSince(result1 ?? Date())) + + XCTAssertLessThanOrEqual(timeDifference1, 1.0, "Expected enrollment date for subfeatureName1 to match at the second level") + XCTAssertNil(result2) + } + + func testCohortReturnsNilIfCohortDoesNotExist() { + // GIVEN + let subfeatureName = "TestSubfeature" + + // WHEN + let result = experimentCohortsManager.cohort(for: subfeatureName) + + // THEN + XCTAssertNil(result) + } + + func testEnrolmentDateReturnsNilIfDateDoesNotExist() { + // GIVEN + let subfeatureName = "TestSubfeature" + + // WHEN + let result = experimentCohortsManager.enrolmentDate(for: subfeatureName) + + // THEN + XCTAssertNil(result) + } + + func testRemoveCohortSuccessfullyRemovesData() { + // GIVEN + saveExperimentData([subfeatureName1: experimentData1]) + let initialData = mockUserDefaults.data(forKey: "ExperimentsData") + XCTAssertNotNil(initialData, "Expected initial data to be saved in UserDefaults.") + + // WHEN + experimentCohortsManager.removeCohort(for: subfeatureName1) + + // THEN + if let remainingData = mockUserDefaults.data(forKey: "ExperimentsData") { + let decoder = JSONDecoder() + let experiments = try? decoder.decode(Experiments.self, from: remainingData) + XCTAssertNil(experiments?[subfeatureName1]) + } + } + + func testRemoveCohortDoesNothingIfSubfeatureDoesNotExist() { + // GIVEN + saveExperimentData([subfeatureName1: experimentData1, subfeatureName2: experimentData2]) + let initialData = mockUserDefaults.data(forKey: "ExperimentsData") + XCTAssertNotNil(initialData, "Expected initial data to be saved in UserDefaults.") + + // WHEN + experimentCohortsManager.removeCohort(for: "someOtherSubfeature") + + // THEN + if let remainingData = mockUserDefaults.data(forKey: "ExperimentsData") { + let decoder = JSONDecoder() + let experiments = try? decoder.decode(Experiments.self, from: remainingData) + XCTAssertNotNil(experiments?[subfeatureName1]) + XCTAssertNotNil(experiments?[subfeatureName2]) + } + } + + func testAssignCohortReturnsNilIfNoCohorts() { + // GIVEN + let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: []) + + // WHEN + let result = experimentCohortsManager.assignCohort(for: subfeature) + + // THEN + XCTAssertNil(result) + } + + func testAssignCohortReturnsNilIfAllWeightsAreZero() { + // GIVEN + let jsonCohort1: [String: Any] = ["name": "TestCohort", "weight": 0] + let jsonCohort2: [String: Any] = ["name": "TestCohort", "weight": 0] + let cohorts = [ + PrivacyConfigurationData.Cohort(json: jsonCohort1)!, + PrivacyConfigurationData.Cohort(json: jsonCohort2)! + ] + let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: cohorts) + + // WHEN + let result = experimentCohortsManager.assignCohort(for: subfeature) + + // THEN + XCTAssertNil(result) + } + + func testAssignCohortSelectsCorrectCohortBasedOnWeight() { + // Cohort1 has weight 1, Cohort2 has weight 3 + // Total weight is 1 + 3 = 4 + let jsonCohort1: [String: Any] = ["name": "Cohort1", "weight": 1] + let jsonCohort2: [String: Any] = ["name": "Cohort2", "weight": 3] + let cohorts = [ + PrivacyConfigurationData.Cohort(json: jsonCohort1)!, + PrivacyConfigurationData.Cohort(json: jsonCohort2)! + ] + let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: cohorts) + let expectedTotalWeight = 4.0 + + // Use a custom randomizer to verify the range + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { range in + // Assert that the range lower bound is 0 + XCTAssertEqual(range.lowerBound, 0.0) + // Assert that the range upper bound is the total weight + XCTAssertEqual(range.upperBound, expectedTotalWeight) + return 0.0 + } + ) + + // Test case where random value is at the very start of Cohort1's range (0) + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { _ in 0.0 } + ) + let resultStartOfCohort1 = experimentCohortsManager.assignCohort(for: subfeature) + XCTAssertEqual(resultStartOfCohort1, "Cohort1") + + // Test case where random value is at the end of Cohort1's range (0.9) + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { _ in 0.9 } + ) + let resultEndOfCohort1 = experimentCohortsManager.assignCohort(for: subfeature) + XCTAssertEqual(resultEndOfCohort1, "Cohort1") + + // Test case where random value is at the start of Cohort2's range (1.00 to 4) + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { _ in 1.00 } + ) + let resultStartOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) + XCTAssertEqual(resultStartOfCohort2, "Cohort2") + + // Test case where random value falls exactly within Cohort2's range (2.5) + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { _ in 2.5 } + ) + let resultMiddleOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) + XCTAssertEqual(resultMiddleOfCohort2, "Cohort2") + + // Test case where random value is at the end of Cohort2's range (4) + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { _ in 3.9 } + ) + let resultEndOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) + XCTAssertEqual(resultEndOfCohort2, "Cohort2") + } + + func testAssignCohortWithSingleCohortAlwaysSelectsThatCohort() { + // GIVEN + let jsonCohort1: [String: Any] = ["name": "Cohort1", "weight": 1] + let cohorts = [ + PrivacyConfigurationData.Cohort(json: jsonCohort1)! + ] + let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: cohorts) + let expectedTotalWeight = 1.0 + + // Use a custom randomizer to verify the range + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { range in + // Assert that the range lower bound is 0 + XCTAssertEqual(range.lowerBound, 0.0) + // Assert that the range upper bound is the total weight + XCTAssertEqual(range.upperBound, expectedTotalWeight) + return 0.0 + } + ) + + // WHEN + experimentCohortsManager = ExperimentCohortsManager( + userDefaults: mockUserDefaults, + randomizer: { range in Double.random(in: range)} + ) + let result = experimentCohortsManager.assignCohort(for: subfeature) + + // THEN + XCTAssertEqual(result, "Cohort1") + } + +} diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift index f4406afef..fac814b1c 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift @@ -65,6 +65,9 @@ class PrivacyConfigurationDataTests: XCTestCase { XCTAssertEqual(subfeatures["disabledSubfeature"]?.state, "disabled") XCTAssertEqual(subfeatures["minSupportedSubfeature"]?.minSupportedVersion, "1.36.0") XCTAssertEqual(subfeatures["enabledSubfeature"]?.state, "enabled") + XCTAssertEqual(subfeatures["enabledSubfeature"]?.cohorts?.count, 3) + XCTAssertEqual(subfeatures["enabledSubfeature"]?.cohorts?[0].name, "myExperimentControl") + XCTAssertEqual(subfeatures["enabledSubfeature"]?.cohorts?[0].weight, 1) XCTAssertEqual(subfeatures["internalSubfeature"]?.state, "internal") } else { XCTFail("Could not parse subfeatures") diff --git a/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json b/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json index 2728eaf55..3fd5be5a5 100644 --- a/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json +++ b/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json @@ -170,7 +170,22 @@ "minSupportedVersion": "1.36.0" }, "enabledSubfeature": { - "state": "enabled" + "state": "enabled", + "description": "A description of the sub-feature", + "cohorts": [ + { + "name": "myExperimentControl", + "weight": 1 + }, + { + "name": "myExperimentBlue", + "weight": 1 + }, + { + "name": "myExperimentRed", + "weight": 1 + } + ] }, "internalSubfeature": { "state": "internal" From 914a5ed08996a96426e302ed9afac4147ff7f254 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 13:32:49 +0100 Subject: [PATCH 02/49] fix lint issues --- .../PrivacyConfig/ExperimentCohortsManager.swift | 2 -- .../PrivacyConfig/ExperimentCohortsManagerTests.swift | 1 - 2 files changed, 3 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index 74d648e0c..db137f414 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -1,6 +1,5 @@ // // ExperimentCohortsManager.swift -// DuckDuckGo // // Copyright © 2024 DuckDuckGo. All rights reserved. // @@ -127,4 +126,3 @@ struct ExperimentCohortsManager: ExperimentCohortsManaging { saveExperimentData(experiments) } } - diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index f3e68a2d6..e6b5a9734 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -1,6 +1,5 @@ // // ExperimentCohortsManagerTests.swift -// DuckDuckGo // // Copyright © 2024 DuckDuckGo. All rights reserved. // From 93247a3c241c7d4259a7cbe7b999a4f69db95ae2 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 13:47:57 +0100 Subject: [PATCH 03/49] fix linting issue --- .../PrivacyConfig/ExperimentCohortsManagerTests.swift | 1 - 1 file changed, 1 deletion(-) diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index e6b5a9734..fd1ad68eb 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -72,7 +72,6 @@ final class ExperimentCohortsManagerTests: XCTestCase { } } - func testCohortReturnsCohortIDIfExistsForMultipleSubfeatures() { // GIVEN saveExperimentData([subfeatureName1: experimentData1, subfeatureName2: experimentData2]) From 5d530f29cc3357bfe236a4b7c32a895ac3344ce3 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 15:00:48 +0100 Subject: [PATCH 04/49] wrap UserDefaults --- .../ExperimentCohortsManager.swift | 19 +++++-- .../ExperimentCohortsManagerTests.swift | 57 +++++++++++-------- 2 files changed, 47 insertions(+), 29 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index db137f414..6f21016c8 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -33,6 +33,11 @@ struct ExperimentData: Codable { typealias Experiments = [String: ExperimentData] +protocol ExperimentDataStoring { + func data(forKey defaultName: String) -> Data? + func set(_ value: Any?, forKey defaultName: String) +} + protocol ExperimentCohortsManaging { /// Retrieves the cohort ID associated with the specified subfeature. /// - Parameter subfeature: The experiment subfeature for which the cohort ID is needed. @@ -56,15 +61,15 @@ protocol ExperimentCohortsManaging { struct ExperimentCohortsManager: ExperimentCohortsManaging { - private let userDefaults: UserDefaults + private let store: ExperimentDataStoring private let decoder = JSONDecoder() private let encoder = JSONEncoder() private let queue = DispatchQueue(label: "com.experimentManager.queue") - private let experimentsDataKey = "ExperimentsData" private let randomizer: (Range) -> Double + private let experimentsDataKey = "ExperimentsData" - init(userDefaults: UserDefaults = UserDefaults.standard, randomizer: @escaping (Range) -> Double) { - self.userDefaults = userDefaults + init(store: ExperimentDataStoring = UserDefaults.standard, randomizer: @escaping (Range) -> Double) { + self.store = store self.randomizer = randomizer encoder.dateEncodingStrategy = .secondsSince1970 decoder.dateDecodingStrategy = .secondsSince1970 @@ -106,7 +111,7 @@ struct ExperimentCohortsManager: ExperimentCohortsManaging { private func getExperimentData() -> Experiments? { queue.sync { - guard let savedData = userDefaults.data(forKey: experimentsDataKey) else { return nil } + guard let savedData = store.data(forKey: experimentsDataKey) else { return nil } return try? decoder.decode(Experiments.self, from: savedData) } } @@ -114,7 +119,7 @@ struct ExperimentCohortsManager: ExperimentCohortsManaging { private func saveExperimentData(_ experiments: Experiments) { queue.sync { if let encodedData = try? encoder.encode(experiments) { - userDefaults.set(encodedData, forKey: experimentsDataKey) + store.set(encodedData, forKey: experimentsDataKey) } } } @@ -126,3 +131,5 @@ struct ExperimentCohortsManager: ExperimentCohortsManaging { saveExperimentData(experiments) } } + +extension UserDefaults: ExperimentDataStoring {} diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index fd1ad68eb..3a134e2dd 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -21,7 +21,7 @@ import XCTest final class ExperimentCohortsManagerTests: XCTestCase { - var mockUserDefaults: UserDefaults! + var mockStore: MockExperimentDataStore! var experimentCohortsManager: ExperimentCohortsManager! let subfeatureName1 = "TestSubfeature1" @@ -40,11 +40,9 @@ final class ExperimentCohortsManagerTests: XCTestCase { override func setUp() { super.setUp() - mockUserDefaults = UserDefaults(suiteName: "com.test.ExperimentCohortsManagerTests") - mockUserDefaults.removePersistentDomain(forName: "com.test.ExperimentCohortsManagerTests") - + mockStore = MockExperimentDataStore() experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { _ in 50.0 } ) @@ -56,8 +54,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { } override func tearDown() { - mockUserDefaults.removePersistentDomain(forName: "com.test.ExperimentCohortsManagerTests") - mockUserDefaults = nil + mockStore = nil experimentCohortsManager = nil expectedDate1 = nil experimentData1 = nil @@ -68,7 +65,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { private func saveExperimentData(_ data: [String: ExperimentData]) { if let encodedData = try? encoder.encode(data) { - mockUserDefaults.set(encodedData, forKey: "ExperimentsData") + mockStore.dataToReturn = encodedData } } @@ -125,14 +122,13 @@ final class ExperimentCohortsManagerTests: XCTestCase { func testRemoveCohortSuccessfullyRemovesData() { // GIVEN saveExperimentData([subfeatureName1: experimentData1]) - let initialData = mockUserDefaults.data(forKey: "ExperimentsData") - XCTAssertNotNil(initialData, "Expected initial data to be saved in UserDefaults.") // WHEN experimentCohortsManager.removeCohort(for: subfeatureName1) // THEN - if let remainingData = mockUserDefaults.data(forKey: "ExperimentsData") { + + if let remainingData = mockStore.dataSaved { let decoder = JSONDecoder() let experiments = try? decoder.decode(Experiments.self, from: remainingData) XCTAssertNil(experiments?[subfeatureName1]) @@ -142,14 +138,12 @@ final class ExperimentCohortsManagerTests: XCTestCase { func testRemoveCohortDoesNothingIfSubfeatureDoesNotExist() { // GIVEN saveExperimentData([subfeatureName1: experimentData1, subfeatureName2: experimentData2]) - let initialData = mockUserDefaults.data(forKey: "ExperimentsData") - XCTAssertNotNil(initialData, "Expected initial data to be saved in UserDefaults.") // WHEN experimentCohortsManager.removeCohort(for: "someOtherSubfeature") // THEN - if let remainingData = mockUserDefaults.data(forKey: "ExperimentsData") { + if let remainingData = mockStore.dataSaved { let decoder = JSONDecoder() let experiments = try? decoder.decode(Experiments.self, from: remainingData) XCTAssertNotNil(experiments?[subfeatureName1]) @@ -199,7 +193,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { // Use a custom randomizer to verify the range experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { range in // Assert that the range lower bound is 0 XCTAssertEqual(range.lowerBound, 0.0) @@ -211,7 +205,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { // Test case where random value is at the very start of Cohort1's range (0) experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { _ in 0.0 } ) let resultStartOfCohort1 = experimentCohortsManager.assignCohort(for: subfeature) @@ -219,7 +213,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { // Test case where random value is at the end of Cohort1's range (0.9) experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { _ in 0.9 } ) let resultEndOfCohort1 = experimentCohortsManager.assignCohort(for: subfeature) @@ -227,7 +221,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { // Test case where random value is at the start of Cohort2's range (1.00 to 4) experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { _ in 1.00 } ) let resultStartOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) @@ -235,7 +229,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { // Test case where random value falls exactly within Cohort2's range (2.5) experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { _ in 2.5 } ) let resultMiddleOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) @@ -243,14 +237,14 @@ final class ExperimentCohortsManagerTests: XCTestCase { // Test case where random value is at the end of Cohort2's range (4) experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { _ in 3.9 } ) let resultEndOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) XCTAssertEqual(resultEndOfCohort2, "Cohort2") } - func testAssignCohortWithSingleCohortAlwaysSelectsThatCohort() { + func testAssignCohortWithSingleCohortAlwaysSelectsThatCohort() throws { // GIVEN let jsonCohort1: [String: Any] = ["name": "Cohort1", "weight": 1] let cohorts = [ @@ -261,7 +255,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { // Use a custom randomizer to verify the range experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { range in // Assert that the range lower bound is 0 XCTAssertEqual(range.lowerBound, 0.0) @@ -273,13 +267,30 @@ final class ExperimentCohortsManagerTests: XCTestCase { // WHEN experimentCohortsManager = ExperimentCohortsManager( - userDefaults: mockUserDefaults, + store: mockStore, randomizer: { range in Double.random(in: range)} ) let result = experimentCohortsManager.assignCohort(for: subfeature) + let savedData = try XCTUnwrap(mockStore.dataSaved) // THEN XCTAssertEqual(result, "Cohort1") + let decodedSavedData = try XCTUnwrap(JSONDecoder().decode(Experiments.self, from: savedData)) + XCTAssertEqual(cohorts[0].name, decodedSavedData[subfeature.subfeatureID]?.cohort) + } + +} + +class MockExperimentDataStore: ExperimentDataStoring { + var dataToReturn: Data? + var dataSaved: Data? + + func data(forKey defaultName: String) -> Data? { + dataToReturn + } + + func set(_ value: Any?, forKey defaultName: String) { + dataSaved = value as? Data } } From 2dcef1efe1cbf601fe790db786a74b06b73fefb8 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 15:03:05 +0100 Subject: [PATCH 05/49] fix linting --- .../PrivacyConfig/ExperimentCohortsManagerTests.swift | 3 +-- 1 file changed, 1 insertion(+), 2 deletions(-) diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index 3a134e2dd..6c1a1ef4c 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -127,7 +127,6 @@ final class ExperimentCohortsManagerTests: XCTestCase { experimentCohortsManager.removeCohort(for: subfeatureName1) // THEN - if let remainingData = mockStore.dataSaved { let decoder = JSONDecoder() let experiments = try? decoder.decode(Experiments.self, from: remainingData) @@ -288,7 +287,7 @@ class MockExperimentDataStore: ExperimentDataStoring { func data(forKey defaultName: String) -> Data? { dataToReturn } - + func set(_ value: Any?, forKey defaultName: String) { dataSaved = value as? Data } From 5aec88df61e668ba36ea46c39ceffeb1d820fd0e Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 16:22:35 +0100 Subject: [PATCH 06/49] refactor --- .../ExperimentCohortsManager.swift | 30 +---- .../PrivacyConfig/ExperimentsDataStore.swift | 60 ++++++++++ .../ExperimentCohortsManagerTests.swift | 59 +++------- .../ExperimentsDataStoreTests.swift | 105 ++++++++++++++++++ 4 files changed, 186 insertions(+), 68 deletions(-) create mode 100644 Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift create mode 100644 Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index 6f21016c8..ba167a0e4 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -26,18 +26,13 @@ struct ExperimentSubfeature { typealias CohortID = String typealias SubfeatureID = String -struct ExperimentData: Codable { +struct ExperimentData: Codable, Equatable { let cohort: String let enrollmentDate: Date } typealias Experiments = [String: ExperimentData] -protocol ExperimentDataStoring { - func data(forKey defaultName: String) -> Data? - func set(_ value: Any?, forKey defaultName: String) -} - protocol ExperimentCohortsManaging { /// Retrieves the cohort ID associated with the specified subfeature. /// - Parameter subfeature: The experiment subfeature for which the cohort ID is needed. @@ -59,20 +54,16 @@ protocol ExperimentCohortsManaging { func removeCohort(for subfeatureID: SubfeatureID) } -struct ExperimentCohortsManager: ExperimentCohortsManaging { +class ExperimentCohortsManager: ExperimentCohortsManaging { - private let store: ExperimentDataStoring - private let decoder = JSONDecoder() - private let encoder = JSONEncoder() + private var store: ExperimentsDataStoring private let queue = DispatchQueue(label: "com.experimentManager.queue") private let randomizer: (Range) -> Double private let experimentsDataKey = "ExperimentsData" - init(store: ExperimentDataStoring = UserDefaults.standard, randomizer: @escaping (Range) -> Double) { + init(store: ExperimentsDataStoring = ExperimentsDataStore(), randomizer: @escaping (Range) -> Double) { self.store = store self.randomizer = randomizer - encoder.dateEncodingStrategy = .secondsSince1970 - decoder.dateDecodingStrategy = .secondsSince1970 } func cohort(for subfeatureID: SubfeatureID) -> CohortID? { @@ -110,18 +101,11 @@ struct ExperimentCohortsManager: ExperimentCohortsManaging { } private func getExperimentData() -> Experiments? { - queue.sync { - guard let savedData = store.data(forKey: experimentsDataKey) else { return nil } - return try? decoder.decode(Experiments.self, from: savedData) - } + return store.experiments } private func saveExperimentData(_ experiments: Experiments) { - queue.sync { - if let encodedData = try? encoder.encode(experiments) { - store.set(encodedData, forKey: experimentsDataKey) - } - } + store.experiments = experiments } private func saveCohort(_ cohort: CohortID, in experimentID: SubfeatureID) { @@ -131,5 +115,3 @@ struct ExperimentCohortsManager: ExperimentCohortsManaging { saveExperimentData(experiments) } } - -extension UserDefaults: ExperimentDataStoring {} diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift new file mode 100644 index 000000000..849f9168b --- /dev/null +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift @@ -0,0 +1,60 @@ +// +// ExperimentsDataStore.swift +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import Foundation + +protocol ExperimentsDataStoring { + var experiments: Experiments? { get set } +} + +protocol LocalDataStoring { + func data(forKey defaultName: String) -> Data? + func set(_ value: Any?, forKey defaultName: String) +} + +struct ExperimentsDataStore: ExperimentsDataStoring { + private let localDataStoring: LocalDataStoring + private let experimentsDataKey = "ExperimentsData" + private let queue = DispatchQueue(label: "com.experimentManager.queue") + private let decoder = JSONDecoder() + private let encoder = JSONEncoder() + + init(localDataStoring: LocalDataStoring = UserDefaults.standard) { + self.localDataStoring = localDataStoring + encoder.dateEncodingStrategy = .secondsSince1970 + decoder.dateDecodingStrategy = .secondsSince1970 + } + + var experiments: Experiments? { + get { + queue.sync { + guard let savedData = localDataStoring.data(forKey: experimentsDataKey) else { return nil } + return try? decoder.decode(Experiments.self, from: savedData) + } + } + set { + queue.sync { + if let encodedData = try? encoder.encode(newValue) { + localDataStoring.set(encodedData, forKey: experimentsDataKey) + } + } + } + } +} + +extension UserDefaults: LocalDataStoring {} diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index 6c1a1ef4c..bc7f787cf 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -25,11 +25,9 @@ final class ExperimentCohortsManagerTests: XCTestCase { var experimentCohortsManager: ExperimentCohortsManager! let subfeatureName1 = "TestSubfeature1" - var expectedDate1: Date! var experimentData1: ExperimentData! let subfeatureName2 = "TestSubfeature2" - var expectedDate2: Date! var experimentData2: ExperimentData! let encoder: JSONEncoder = { @@ -46,32 +44,24 @@ final class ExperimentCohortsManagerTests: XCTestCase { randomizer: { _ in 50.0 } ) - expectedDate1 = Date() + let expectedDate1 = Date() experimentData1 = ExperimentData(cohort: "TestCohort1", enrollmentDate: expectedDate1) - expectedDate2 = Date().addingTimeInterval(60) // Second subfeature with a different date + let expectedDate2 = Date().addingTimeInterval(60) experimentData2 = ExperimentData(cohort: "TestCohort2", enrollmentDate: expectedDate2) } override func tearDown() { mockStore = nil experimentCohortsManager = nil - expectedDate1 = nil experimentData1 = nil - expectedDate2 = nil experimentData2 = nil super.tearDown() } - private func saveExperimentData(_ data: [String: ExperimentData]) { - if let encodedData = try? encoder.encode(data) { - mockStore.dataToReturn = encodedData - } - } - func testCohortReturnsCohortIDIfExistsForMultipleSubfeatures() { // GIVEN - saveExperimentData([subfeatureName1: experimentData1, subfeatureName2: experimentData2]) + mockStore.experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] // WHEN let result1 = experimentCohortsManager.cohort(for: subfeatureName1) @@ -84,14 +74,14 @@ final class ExperimentCohortsManagerTests: XCTestCase { func testEnrolmentDateReturnsCorrectDateIfExists() { // GIVEN - saveExperimentData([subfeatureName1: experimentData1]) + mockStore.experiments = [subfeatureName1: experimentData1] // WHEN let result1 = experimentCohortsManager.enrolmentDate(for: subfeatureName1) let result2 = experimentCohortsManager.enrolmentDate(for: subfeatureName2) // THEN - let timeDifference1 = abs(expectedDate1.timeIntervalSince(result1 ?? Date())) + let timeDifference1 = abs(experimentData1.enrollmentDate.timeIntervalSince(result1 ?? Date())) XCTAssertLessThanOrEqual(timeDifference1, 1.0, "Expected enrollment date for subfeatureName1 to match at the second level") XCTAssertNil(result2) @@ -119,35 +109,28 @@ final class ExperimentCohortsManagerTests: XCTestCase { XCTAssertNil(result) } - func testRemoveCohortSuccessfullyRemovesData() { + func testRemoveCohortSuccessfullyRemovesData() throws { // GIVEN - saveExperimentData([subfeatureName1: experimentData1]) + mockStore.experiments = [subfeatureName1: experimentData1] // WHEN experimentCohortsManager.removeCohort(for: subfeatureName1) // THEN - if let remainingData = mockStore.dataSaved { - let decoder = JSONDecoder() - let experiments = try? decoder.decode(Experiments.self, from: remainingData) - XCTAssertNil(experiments?[subfeatureName1]) - } + let experiments = try XCTUnwrap(mockStore.experiments) + XCTAssertTrue(experiments.isEmpty) } func testRemoveCohortDoesNothingIfSubfeatureDoesNotExist() { // GIVEN - saveExperimentData([subfeatureName1: experimentData1, subfeatureName2: experimentData2]) + let expectedExperiments: Experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] + mockStore.experiments = expectedExperiments // WHEN experimentCohortsManager.removeCohort(for: "someOtherSubfeature") // THEN - if let remainingData = mockStore.dataSaved { - let decoder = JSONDecoder() - let experiments = try? decoder.decode(Experiments.self, from: remainingData) - XCTAssertNotNil(experiments?[subfeatureName1]) - XCTAssertNotNil(experiments?[subfeatureName2]) - } + XCTAssertEqual( mockStore.experiments, expectedExperiments) } func testAssignCohortReturnsNilIfNoCohorts() { @@ -270,26 +253,14 @@ final class ExperimentCohortsManagerTests: XCTestCase { randomizer: { range in Double.random(in: range)} ) let result = experimentCohortsManager.assignCohort(for: subfeature) - let savedData = try XCTUnwrap(mockStore.dataSaved) // THEN XCTAssertEqual(result, "Cohort1") - let decodedSavedData = try XCTUnwrap(JSONDecoder().decode(Experiments.self, from: savedData)) - XCTAssertEqual(cohorts[0].name, decodedSavedData[subfeature.subfeatureID]?.cohort) + XCTAssertEqual(cohorts[0].name, mockStore.experiments?[subfeature.subfeatureID]?.cohort) } } -class MockExperimentDataStore: ExperimentDataStoring { - var dataToReturn: Data? - var dataSaved: Data? - - func data(forKey defaultName: String) -> Data? { - dataToReturn - } - - func set(_ value: Any?, forKey defaultName: String) { - dataSaved = value as? Data - } - +class MockExperimentDataStore: ExperimentsDataStoring { + var experiments: Experiments? = nil } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift new file mode 100644 index 000000000..0466155b0 --- /dev/null +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift @@ -0,0 +1,105 @@ +// +// ExperimentsDataStoreTests.swift +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import XCTest +@testable import BrowserServicesKit + +final class ExperimentsDataStoreTests: XCTestCase { + + let subfeatureName1 = "TestSubfeature1" + var expectedDate1: Date! + var experimentData1: ExperimentData! + + let subfeatureName2 = "TestSubfeature2" + var expectedDate2: Date! + var experimentData2: ExperimentData! + + var mockDataStore: MockLocalDataStore! + var experimentsDataStore: ExperimentsDataStore! + let testExperimentKey = "ExperimentsData" + + override func setUp() { + super.setUp() + mockDataStore = MockLocalDataStore() + experimentsDataStore = ExperimentsDataStore(localDataStoring: mockDataStore) + } + + override func tearDown() { + mockDataStore = nil + experimentsDataStore = nil + super.tearDown() + } + + func testExperimentsGetReturnsDecodedExperiments() { + // GIVEN + let experimentData1 = ExperimentData(cohort: "TestCohort1", enrollmentDate: Date()) + let experimentData2 = ExperimentData(cohort: "TestCohort2", enrollmentDate: Date()) + let experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] + + let encoder = JSONEncoder() + encoder.dateEncodingStrategy = .secondsSince1970 + let encodedData = try? encoder.encode(experiments) + mockDataStore.data = encodedData + + // WHEN + let result = experimentsDataStore.experiments + + // THEN + let timeDifference1 = abs(experimentData1.enrollmentDate.timeIntervalSince(result?[subfeatureName1]?.enrollmentDate ?? Date())) + let timeDifference2 = abs(experimentData2.enrollmentDate.timeIntervalSince(result?[subfeatureName2]?.enrollmentDate ?? Date())) + XCTAssertEqual(result?[subfeatureName1]?.cohort, experimentData1.cohort) + XCTAssertLessThanOrEqual(timeDifference1, 1.0) + + XCTAssertEqual(result?[subfeatureName2]?.cohort, experimentData2.cohort) + XCTAssertLessThanOrEqual(timeDifference2, 1.0) + } + + func testExperimentsSetEncodesAndStoresData() throws { + // GIVEN + let experimentData1 = ExperimentData(cohort: "TestCohort1", enrollmentDate: Date()) + let experimentData2 = ExperimentData(cohort: "TestCohort2", enrollmentDate: Date()) + let experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] + + // WHEN + experimentsDataStore.experiments = experiments + + // THEN + let storedData = try XCTUnwrap(mockDataStore.data) + let decoder = JSONDecoder() + decoder.dateDecodingStrategy = .secondsSince1970 + let decodedExperiments = try? decoder.decode(Experiments.self, from: storedData) + let timeDifference1 = abs(experimentData1.enrollmentDate.timeIntervalSince(decodedExperiments?[subfeatureName1]?.enrollmentDate ?? Date())) + let timeDifference2 = abs(experimentData2.enrollmentDate.timeIntervalSince(decodedExperiments?[subfeatureName2]?.enrollmentDate ?? Date())) + XCTAssertEqual(decodedExperiments?[subfeatureName1]?.cohort, experimentData1.cohort) + XCTAssertLessThanOrEqual(timeDifference1, 1.0) + XCTAssertEqual(decodedExperiments?[subfeatureName2]?.cohort, experimentData2.cohort) + XCTAssertLessThanOrEqual(timeDifference2, 1.0) + } +} + +class MockLocalDataStore: LocalDataStoring { + var data: Data? + + func data(forKey defaultName: String) -> Data? { + return data + } + + func set(_ value: Any?, forKey defaultName: String) { + data = value as? Data + } +} From b188d6718c0c6fda5a760031d4ea57c120dae91c Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 16:24:48 +0100 Subject: [PATCH 07/49] fix linting --- .../PrivacyConfig/ExperimentCohortsManager.swift | 2 +- .../PrivacyConfig/ExperimentCohortsManagerTests.swift | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index ba167a0e4..3974d2d45 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -54,7 +54,7 @@ protocol ExperimentCohortsManaging { func removeCohort(for subfeatureID: SubfeatureID) } -class ExperimentCohortsManager: ExperimentCohortsManaging { +final class ExperimentCohortsManager: ExperimentCohortsManaging { private var store: ExperimentsDataStoring private let queue = DispatchQueue(label: "com.experimentManager.queue") diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index bc7f787cf..25be855d4 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -262,5 +262,5 @@ final class ExperimentCohortsManagerTests: XCTestCase { } class MockExperimentDataStore: ExperimentsDataStoring { - var experiments: Experiments? = nil + var experiments: Experiments? } From 1054923d7b71487d95d8abd5b80526ecc5ac9766 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 17:58:11 +0100 Subject: [PATCH 08/49] address some comments --- .../ExperimentCohortsManager.swift | 36 +++++++++---------- .../PrivacyConfigurationData.swift | 4 +-- .../ExperimentCohortsManagerTests.swift | 30 ++++++++-------- 3 files changed, 34 insertions(+), 36 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index 3974d2d45..0288de356 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -42,24 +42,26 @@ protocol ExperimentCohortsManaging { /// Retrieves the enrollment date for the specified subfeature. /// - Parameter subfeatureID: The experiment subfeature for which the enrollment date is needed. /// - Returns: The `Date` of enrollment if one exists; otherwise, returns `nil`. - func enrolmentDate(for subfeatureID: SubfeatureID) -> Date? + func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? /// Assigns a cohort to the given subfeature based on defined weights and saves it to UserDefaults. /// - Parameter subfeature: The experiment subfeature to assign a cohort for. /// - Returns: The name of the assigned cohort, or `nil` if no cohort could be assigned. - func assignCohort(for subfeature: ExperimentSubfeature) -> CohortID? + func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? /// Removes the assigned cohort data for the specified subfeature. /// - Parameter subfeature: The experiment subfeature for which the cohort data should be removed. - func removeCohort(for subfeatureID: SubfeatureID) + func removeCohort(from subfeatureID: SubfeatureID) } final class ExperimentCohortsManager: ExperimentCohortsManaging { private var store: ExperimentsDataStoring - private let queue = DispatchQueue(label: "com.experimentManager.queue") private let randomizer: (Range) -> Double - private let experimentsDataKey = "ExperimentsData" + + var experiments: Experiments? { + store.experiments + } init(store: ExperimentsDataStoring = ExperimentsDataStore(), randomizer: @escaping (Range) -> Double) { self.store = store @@ -67,16 +69,16 @@ final class ExperimentCohortsManager: ExperimentCohortsManaging { } func cohort(for subfeatureID: SubfeatureID) -> CohortID? { - guard let experiments = getExperimentData() else { return nil } + guard let experiments = experiments else { return nil } return experiments[subfeatureID]?.cohort } - func enrolmentDate(for subfeatureID: SubfeatureID) -> Date? { - guard let experiments = getExperimentData() else { return nil } + func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? { + guard let experiments = experiments else { return nil } return experiments[subfeatureID]?.enrollmentDate } - func assignCohort(for subfeature: ExperimentSubfeature) -> CohortID? { + func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? { let cohorts = subfeature.cohorts let totalWeight = cohorts.reduce(0, { $0 + $1.weight }) guard totalWeight > 0 else { return nil } @@ -94,24 +96,20 @@ final class ExperimentCohortsManager: ExperimentCohortsManaging { return nil } - func removeCohort(for subfeatureID: SubfeatureID) { - guard var experiments = getExperimentData() else { return } + func removeCohort(from subfeatureID: SubfeatureID) { + guard var experiments = experiments else { return } experiments.removeValue(forKey: subfeatureID) - saveExperimentData(experiments) - } - - private func getExperimentData() -> Experiments? { - return store.experiments + saveExperiment(experiments) } - private func saveExperimentData(_ experiments: Experiments) { + private func saveExperiment(_ experiments: Experiments) { store.experiments = experiments } private func saveCohort(_ cohort: CohortID, in experimentID: SubfeatureID) { - var experiments = getExperimentData() ?? Experiments() + var experiments = experiments ?? Experiments() let experimentData = ExperimentData(cohort: cohort, enrollmentDate: Date()) experiments[experimentID] = experimentData - saveExperimentData(experiments) + saveExperiment(experiments) } } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift index eb9267505..a7673d64c 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift @@ -42,12 +42,12 @@ public struct PrivacyConfigurationData { public init?(json: [String: Any]) { guard let name = json["name"] as? String, - let weightValue = json["weight"] as? Int else { + let weight = json["weight"] as? Int else { return nil } self.name = name - self.weight = weightValue + self.weight = weight } } public let features: [FeatureName: PrivacyFeature] diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index 25be855d4..518249560 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -72,13 +72,13 @@ final class ExperimentCohortsManagerTests: XCTestCase { XCTAssertEqual(result2, experimentData2.cohort) } - func testEnrolmentDateReturnsCorrectDateIfExists() { + func testEnrollmentDateReturnsCorrectDateIfExists() { // GIVEN mockStore.experiments = [subfeatureName1: experimentData1] // WHEN - let result1 = experimentCohortsManager.enrolmentDate(for: subfeatureName1) - let result2 = experimentCohortsManager.enrolmentDate(for: subfeatureName2) + let result1 = experimentCohortsManager.enrollmentDate(for: subfeatureName1) + let result2 = experimentCohortsManager.enrollmentDate(for: subfeatureName2) // THEN let timeDifference1 = abs(experimentData1.enrollmentDate.timeIntervalSince(result1 ?? Date())) @@ -98,12 +98,12 @@ final class ExperimentCohortsManagerTests: XCTestCase { XCTAssertNil(result) } - func testEnrolmentDateReturnsNilIfDateDoesNotExist() { + func testEnrollmentDateReturnsNilIfDateDoesNotExist() { // GIVEN let subfeatureName = "TestSubfeature" // WHEN - let result = experimentCohortsManager.enrolmentDate(for: subfeatureName) + let result = experimentCohortsManager.enrollmentDate(for: subfeatureName) // THEN XCTAssertNil(result) @@ -114,7 +114,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { mockStore.experiments = [subfeatureName1: experimentData1] // WHEN - experimentCohortsManager.removeCohort(for: subfeatureName1) + experimentCohortsManager.removeCohort(from: subfeatureName1) // THEN let experiments = try XCTUnwrap(mockStore.experiments) @@ -127,7 +127,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { mockStore.experiments = expectedExperiments // WHEN - experimentCohortsManager.removeCohort(for: "someOtherSubfeature") + experimentCohortsManager.removeCohort(from: "someOtherSubfeature") // THEN XCTAssertEqual( mockStore.experiments, expectedExperiments) @@ -138,7 +138,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: []) // WHEN - let result = experimentCohortsManager.assignCohort(for: subfeature) + let result = experimentCohortsManager.assignCohort(to: subfeature) // THEN XCTAssertNil(result) @@ -155,7 +155,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: cohorts) // WHEN - let result = experimentCohortsManager.assignCohort(for: subfeature) + let result = experimentCohortsManager.assignCohort(to: subfeature) // THEN XCTAssertNil(result) @@ -190,7 +190,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { store: mockStore, randomizer: { _ in 0.0 } ) - let resultStartOfCohort1 = experimentCohortsManager.assignCohort(for: subfeature) + let resultStartOfCohort1 = experimentCohortsManager.assignCohort(to: subfeature) XCTAssertEqual(resultStartOfCohort1, "Cohort1") // Test case where random value is at the end of Cohort1's range (0.9) @@ -198,7 +198,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { store: mockStore, randomizer: { _ in 0.9 } ) - let resultEndOfCohort1 = experimentCohortsManager.assignCohort(for: subfeature) + let resultEndOfCohort1 = experimentCohortsManager.assignCohort(to: subfeature) XCTAssertEqual(resultEndOfCohort1, "Cohort1") // Test case where random value is at the start of Cohort2's range (1.00 to 4) @@ -206,7 +206,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { store: mockStore, randomizer: { _ in 1.00 } ) - let resultStartOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) + let resultStartOfCohort2 = experimentCohortsManager.assignCohort(to: subfeature) XCTAssertEqual(resultStartOfCohort2, "Cohort2") // Test case where random value falls exactly within Cohort2's range (2.5) @@ -214,7 +214,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { store: mockStore, randomizer: { _ in 2.5 } ) - let resultMiddleOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) + let resultMiddleOfCohort2 = experimentCohortsManager.assignCohort(to: subfeature) XCTAssertEqual(resultMiddleOfCohort2, "Cohort2") // Test case where random value is at the end of Cohort2's range (4) @@ -222,7 +222,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { store: mockStore, randomizer: { _ in 3.9 } ) - let resultEndOfCohort2 = experimentCohortsManager.assignCohort(for: subfeature) + let resultEndOfCohort2 = experimentCohortsManager.assignCohort(to: subfeature) XCTAssertEqual(resultEndOfCohort2, "Cohort2") } @@ -252,7 +252,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { store: mockStore, randomizer: { range in Double.random(in: range)} ) - let result = experimentCohortsManager.assignCohort(for: subfeature) + let result = experimentCohortsManager.assignCohort(to: subfeature) // THEN XCTAssertEqual(result, "Cohort1") From c05aa96ea141104efb4e12a30df9c3ad996ad563 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 8 Nov 2024 18:01:43 +0100 Subject: [PATCH 09/49] minor refactor --- .../PrivacyConfig/ExperimentCohortsManager.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index 0288de356..271b934d0 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -80,7 +80,7 @@ final class ExperimentCohortsManager: ExperimentCohortsManaging { func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? { let cohorts = subfeature.cohorts - let totalWeight = cohorts.reduce(0, { $0 + $1.weight }) + let totalWeight = cohorts.map(\.weight).reduce(0, +) guard totalWeight > 0 else { return nil } let randomValue = randomizer(0.. Date: Fri, 8 Nov 2024 18:19:50 +0100 Subject: [PATCH 10/49] use Constants enum --- .../PrivacyConfig/ExperimentsDataStore.swift | 9 ++++++--- 1 file changed, 6 insertions(+), 3 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift index 849f9168b..338c575d7 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift @@ -28,8 +28,11 @@ protocol LocalDataStoring { } struct ExperimentsDataStore: ExperimentsDataStoring { + + private enum Constants { + static let experimentsDataKey = "ExperimentsData" + } private let localDataStoring: LocalDataStoring - private let experimentsDataKey = "ExperimentsData" private let queue = DispatchQueue(label: "com.experimentManager.queue") private let decoder = JSONDecoder() private let encoder = JSONEncoder() @@ -43,14 +46,14 @@ struct ExperimentsDataStore: ExperimentsDataStoring { var experiments: Experiments? { get { queue.sync { - guard let savedData = localDataStoring.data(forKey: experimentsDataKey) else { return nil } + guard let savedData = localDataStoring.data(forKey: Constants.experimentsDataKey) else { return nil } return try? decoder.decode(Experiments.self, from: savedData) } } set { queue.sync { if let encodedData = try? encoder.encode(newValue) { - localDataStoring.set(encodedData, forKey: experimentsDataKey) + localDataStoring.set(encodedData, forKey: Constants.experimentsDataKey) } } } From e862ad8640793604f006dc3bf78d20b337af4b0a Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Mon, 11 Nov 2024 13:14:36 +0100 Subject: [PATCH 11/49] add Targets and configurations --- .../AppPrivacyConfiguration.swift | 16 ++ .../PrivacyConfig/PrivacyConfiguration.swift | 1 + .../PrivacyConfigurationData.swift | 35 +++- .../PrivacyConfigurationManager.swift | 5 + .../AppPrivacyConfigurationTests.swift | 180 ++++++++++++++++++ .../PrivacyConfigurationDataTests.swift | 3 + .../Resources/privacy-config-example.json | 10 + 7 files changed, 248 insertions(+), 2 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 89e64093d..3c07a7918 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -33,6 +33,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { private let locallyUnprotected: DomainsProtectionStore private let internalUserDecider: InternalUserDecider private let userDefaults: UserDefaults + private let locale: Locale private let installDate: Date? public init(data: PrivacyConfigurationData, @@ -40,12 +41,14 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { localProtection: DomainsProtectionStore, internalUserDecider: InternalUserDecider, userDefaults: UserDefaults = UserDefaults(), + locale: Locale = Locale.current, installDate: Date? = nil) { self.data = data self.identifier = identifier self.locallyUnprotected = localProtection self.internalUserDecider = internalUserDecider self.userDefaults = userDefaults + self.locale = locale self.installDate = installDate } @@ -193,9 +196,11 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { public func stateFor(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + // Check parent feature state let parentState = stateFor(featureKey: subfeature.parent, versionProvider: versionProvider) guard case .enabled = parentState else { return parentState } + // Check sub-feature state let subfeatures = subfeatures(for: subfeature.parent) let subfeatureData = subfeatures[subfeature.rawValue] @@ -210,6 +215,12 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { default: return .disabled(.disabledInConfig) } + // Check Targets + // It should not be wrapped in an array and will be removed at some point + if let target = subfeatureData?.targets?.first, !matchTarget(target: target){ + return .disabled(.targetDoesNotMatch) + } + // Handle Rollouts if let rollout = subfeatureData?.rollout, !isRolloutEnabled(subfeature: subfeature, rolloutSteps: rollout.steps, randomizer: randomizer) { @@ -219,6 +230,11 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { return .enabled } + private func matchTarget(target: PrivacyConfigurationData.PrivacyFeature.Feature.Target) -> Bool{ + return target.localeCountry == locale.regionCode && + target.localeLanguage == locale.languageCode + } + private func subfeatures(for feature: PrivacyFeature) -> PrivacyConfigurationData.PrivacyFeature.Features { return data.features[feature.rawValue]?.features ?? [:] } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift index 37d5ace68..a5b63ea30 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift @@ -30,6 +30,7 @@ public enum PrivacyConfigurationFeatureDisabledReason: Equatable { case tooOldInstallation case limitedToInternalUsers case stillInRollout + case targetDoesNotMatch } public protocol PrivacyConfiguration { diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift index a7673d64c..069cda914 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift @@ -136,6 +136,8 @@ public struct PrivacyConfigurationData { case minSupportedVersion case rollout case cohorts + case targets + case config } public struct Rollout: Hashable { @@ -169,10 +171,27 @@ public struct PrivacyConfigurationData { } } + public struct Target { + enum CodingKeys: String { + case localeCountry + case localeLanguage + } + + public let localeCountry: String + public let localeLanguage: String + + public init(json: [String: Any]) { + self.localeCountry = json[CodingKeys.localeCountry.rawValue] as? String ?? "" + self.localeLanguage = json[CodingKeys.localeLanguage.rawValue] as? String ?? "" + } + } + public let state: FeatureState public let minSupportedVersion: FeatureSupportedVersion? public let rollout: Rollout? public let cohorts: [Cohort]? + public let targets: [Target]? + public let config: [String : String]? public init?(json: [String: Any]) { guard let state = json[CodingKeys.state.rawValue] as? String else { @@ -190,9 +209,21 @@ public struct PrivacyConfigurationData { if let cohortData = json[CodingKeys.cohorts.rawValue] as? [[String: Any]] { let parsedCohorts = cohortData.compactMap { Cohort(json: $0) } - self.cohorts = parsedCohorts.isEmpty ? nil : parsedCohorts + cohorts = parsedCohorts.isEmpty ? nil : parsedCohorts + } else { + cohorts = nil + } + + if let targetData = json[CodingKeys.targets.rawValue] as? [[String: Any]] { + targets = targetData.compactMap { Target(json: $0) } + } else { + targets = nil + } + + if let configData = json[CodingKeys.config.rawValue] as? [String: String] { + config = configData } else { - self.cohorts = nil + config = nil } } } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index 219a01613..33e60f898 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -55,6 +55,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { private let localProtection: DomainsProtectionStore private let errorReporting: EventMapping? private let installDate: Date? + private let locale: Locale public let internalUserDecider: InternalUserDecider @@ -110,12 +111,14 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { localProtection: DomainsProtectionStore, errorReporting: EventMapping? = nil, internalUserDecider: InternalUserDecider, + locale: Locale = Locale.current, installDate: Date? = nil ) { self.embeddedDataProvider = embeddedDataProvider self.localProtection = localProtection self.errorReporting = errorReporting self.internalUserDecider = internalUserDecider + self.locale = locale self.installDate = installDate reload(etag: fetchedETag, data: fetchedData) @@ -127,6 +130,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { identifier: fetchedData.etag, localProtection: localProtection, internalUserDecider: internalUserDecider, + locale: locale, installDate: installDate) } @@ -134,6 +138,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { identifier: embeddedConfigData.etag, localProtection: localProtection, internalUserDecider: internalUserDecider, + locale: locale, installDate: installDate) } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift index 252a7b866..0929b7c44 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift @@ -907,6 +907,186 @@ class AppPrivacyConfigurationTests: XCTestCase { XCTAssertEqual(configAfterUpdate.stateFor(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:)), .disabled(.disabledInConfig)) } + let exampleSubfeatureEnabledWithTarget = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "targets": [ + { + "localeCountry": "US", + "localeLanguage": "fr" + } + ] + } + } + } + } + } + """.data(using: .utf8)! + + func testWhenCheckingSubfeatureStateWithSubfeatureEnabledWhenTargetMatches_SubfeatureShouldBeEnabled() { + let mockEmbeddedData = MockEmbeddedDataProvider(data: exampleSubfeatureEnabledWithTarget, etag: "test") + let mockInternalUserStore = MockInternalUserStoring() + let locale = Locale(identifier: "fr_US") + + let manager = PrivacyConfigurationManager(fetchedETag: nil, + fetchedData: nil, + embeddedDataProvider: mockEmbeddedData, + localProtection: MockDomainsProtectionStore(), + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + locale: locale) + let config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) + } + + func testWhenCheckingSubfeatureStateWithSubfeatureEnabledWhenRegionDoesNotMatches_SubfeatureShouldBeDisabled() { + let mockEmbeddedData = MockEmbeddedDataProvider(data: exampleSubfeatureEnabledWithTarget, etag: "test") + let mockInternalUserStore = MockInternalUserStoring() + let locale = Locale(identifier: "fr_FR") + + let manager = PrivacyConfigurationManager(fetchedETag: nil, + fetchedData: nil, + embeddedDataProvider: mockEmbeddedData, + localProtection: MockDomainsProtectionStore(), + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + locale: locale) + let config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .disabled(.targetDoesNotMatch)) + } + + func testWhenCheckingSubfeatureStateWithSubfeatureEnabledWhenLanguageDoesNotMatches_SubfeatureShouldBeDisabled() { + let mockEmbeddedData = MockEmbeddedDataProvider(data: exampleSubfeatureEnabledWithTarget, etag: "test") + let mockInternalUserStore = MockInternalUserStoring() + let locale = Locale(identifier: "it_US") + + let manager = PrivacyConfigurationManager(fetchedETag: nil, + fetchedData: nil, + embeddedDataProvider: mockEmbeddedData, + localProtection: MockDomainsProtectionStore(), + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + locale: locale) + let config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .disabled(.targetDoesNotMatch)) + } + + let exampleSubfeatureDisabledWithTarget = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "disabled", + "targets": [ + { + "localeCountry": "US", + "localeLanguage": "fr" + } + ] + } + } + } + } + } + """.data(using: .utf8)! + + func testWhenCheckingSubfeatureStateWithSubfeatureDisabledWhenTargetMatches_SubfeatureShouldBeDisabled() { + let mockEmbeddedData = MockEmbeddedDataProvider(data: exampleSubfeatureDisabledWithTarget, etag: "test") + let mockInternalUserStore = MockInternalUserStoring() + let locale = Locale(identifier: "fr_US") + + let manager = PrivacyConfigurationManager(fetchedETag: nil, + fetchedData: nil, + embeddedDataProvider: mockEmbeddedData, + localProtection: MockDomainsProtectionStore(), + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + locale: locale) + let config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .disabled(.disabledInConfig)) + } + + let exampleSubfeatureEnabledWithRolloutAndTarget = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "targets": [ + { + "localeCountry": "US", + "localeLanguage": "fr" + } + ], + "rollout": { + "steps": [{ + "percent": 5.0 + }] + } + } + } + } + } + } + """.data(using: .utf8)! + + func testWhenCheckingSubfeatureStateWithSubfeatureEnabledAndTargetMatchesWhenNotInRollout_SubfeatureShouldBeDisabled() { + let mockEmbeddedData = MockEmbeddedDataProvider(data: exampleSubfeatureEnabledWithRolloutAndTarget, etag: "test") + let mockInternalUserStore = MockInternalUserStoring() + let locale = Locale(identifier: "fr_US") + mockRandomValue = 7.0 + clearRolloutData(feature: "autofill", subFeature: "credentialsSaving") + + let manager = PrivacyConfigurationManager(fetchedETag: nil, + fetchedData: nil, + embeddedDataProvider: mockEmbeddedData, + localProtection: MockDomainsProtectionStore(), + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + locale: locale) + let config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .disabled(.stillInRollout)) + } + + func testWhenCheckingSubfeatureStateWithSubfeatureEnabledAndTargetMatchesWhenInRollout_SubfeatureShouldBeEnabled() { + let mockEmbeddedData = MockEmbeddedDataProvider(data: exampleSubfeatureEnabledWithRolloutAndTarget, etag: "test") + let mockInternalUserStore = MockInternalUserStoring() + let locale = Locale(identifier: "fr_US") + mockRandomValue = 2.0 + clearRolloutData(feature: "autofill", subFeature: "credentialsSaving") + + let manager = PrivacyConfigurationManager(fetchedETag: nil, + fetchedData: nil, + embeddedDataProvider: mockEmbeddedData, + localProtection: MockDomainsProtectionStore(), + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + locale: locale) + let config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) + } + let exampleEnabledSubfeatureWithRollout = """ { diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift index fac814b1c..3b4a997f1 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift @@ -68,6 +68,9 @@ class PrivacyConfigurationDataTests: XCTestCase { XCTAssertEqual(subfeatures["enabledSubfeature"]?.cohorts?.count, 3) XCTAssertEqual(subfeatures["enabledSubfeature"]?.cohorts?[0].name, "myExperimentControl") XCTAssertEqual(subfeatures["enabledSubfeature"]?.cohorts?[0].weight, 1) + XCTAssertEqual(subfeatures["enabledSubfeature"]?.targets?[0].localeCountry, "US") + XCTAssertEqual(subfeatures["enabledSubfeature"]?.targets?[0].localeLanguage, "fr") + XCTAssertEqual(subfeatures["enabledSubfeature"]?.config, ["foo" : "foo/value", "bar": "bar/value"]) XCTAssertEqual(subfeatures["internalSubfeature"]?.state, "internal") } else { XCTFail("Could not parse subfeatures") diff --git a/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json b/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json index 3fd5be5a5..8523bc0fe 100644 --- a/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json +++ b/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json @@ -171,6 +171,16 @@ }, "enabledSubfeature": { "state": "enabled", + "targets": [ + { + "localeCountry": "US", + "localeLanguage": "fr" + } + ], + "config": { + "foo": "foo/value", + "bar": "bar/value" + }, "description": "A description of the sub-feature", "cohorts": [ { From 6090a1513982a6b2ba614176d07fa4b2af7a0989 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Mon, 11 Nov 2024 13:32:10 +0100 Subject: [PATCH 12/49] fix linting issues --- .../PrivacyConfig/PrivacyConfigurationData.swift | 2 +- .../PrivacyConfig/PrivacyConfigurationManager.swift | 2 +- .../PrivacyConfig/AppPrivacyConfigurationTests.swift | 4 ++-- .../PrivacyConfig/PrivacyConfigurationDataTests.swift | 2 +- 4 files changed, 5 insertions(+), 5 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift index 069cda914..42df09e91 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift @@ -191,7 +191,7 @@ public struct PrivacyConfigurationData { public let rollout: Rollout? public let cohorts: [Cohort]? public let targets: [Target]? - public let config: [String : String]? + public let config: [String: String]? public init?(json: [String: Any]) { guard let state = json[CodingKeys.state.rawValue] as? String else { diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index 33e60f898..004729e70 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -138,7 +138,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { identifier: embeddedConfigData.etag, localProtection: localProtection, internalUserDecider: internalUserDecider, - locale: locale, + locale: locale, installDate: installDate) } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift index 0929b7c44..9388b14e9 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift @@ -1064,7 +1064,7 @@ class AppPrivacyConfigurationTests: XCTestCase { locale: locale) let config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .disabled(.stillInRollout)) } @@ -1083,7 +1083,7 @@ class AppPrivacyConfigurationTests: XCTestCase { locale: locale) let config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift index 3b4a997f1..88119434b 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift @@ -70,7 +70,7 @@ class PrivacyConfigurationDataTests: XCTestCase { XCTAssertEqual(subfeatures["enabledSubfeature"]?.cohorts?[0].weight, 1) XCTAssertEqual(subfeatures["enabledSubfeature"]?.targets?[0].localeCountry, "US") XCTAssertEqual(subfeatures["enabledSubfeature"]?.targets?[0].localeLanguage, "fr") - XCTAssertEqual(subfeatures["enabledSubfeature"]?.config, ["foo" : "foo/value", "bar": "bar/value"]) + XCTAssertEqual(subfeatures["enabledSubfeature"]?.config, ["foo": "foo/value", "bar": "bar/value"]) XCTAssertEqual(subfeatures["internalSubfeature"]?.state, "internal") } else { XCTFail("Could not parse subfeatures") From 14f92e927fbd455bf40040c6aa1f9a55312ced50 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Tue, 12 Nov 2024 12:04:48 +0100 Subject: [PATCH 13/49] initial implementation --- .../AppPrivacyConfiguration.swift | 65 +++++++++++++++++-- .../ExperimentCohortsManager.swift | 25 +++---- .../PrivacyConfig/ExperimentsDataStore.swift | 10 +-- .../PrivacyConfig/PrivacyConfiguration.swift | 13 ++-- .../UserContentControllerTests.swift | 8 +-- .../AppPrivacyConfigurationTests.swift | 14 ++-- ...SubscriptionFeatureAvailabilityTests.swift | 4 +- Tests/DDGSyncTests/Mocks/Mocks.swift | 8 +-- 8 files changed, 98 insertions(+), 49 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 3c07a7918..5a55c2ba9 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -34,6 +34,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { private let internalUserDecider: InternalUserDecider private let userDefaults: UserDefaults private let locale: Locale + private let experimentManager: ExperimentCohortsManaging private let installDate: Date? public init(data: PrivacyConfigurationData, @@ -42,6 +43,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { internalUserDecider: InternalUserDecider, userDefaults: UserDefaults = UserDefaults(), locale: Locale = Locale.current, + experimentManager: ExperimentCohortsManaging = ExperimentCohortsManager(), installDate: Date? = nil) { self.data = data self.identifier = identifier @@ -49,6 +51,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { self.internalUserDecider = internalUserDecider self.userDefaults = userDefaults self.locale = locale + self.experimentManager = experimentManager self.installDate = installDate } @@ -183,10 +186,11 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } public func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, + cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool { - - switch stateFor(subfeature, versionProvider: versionProvider, randomizer: randomizer) { + + switch stateFor(subfeature, cohortID: cohortID, versionProvider: versionProvider, randomizer: randomizer) { case .enabled: return true case .disabled: @@ -194,18 +198,26 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } } - public func stateFor(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + private func isParentFeatureEnabed(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider) -> PrivacyConfigurationFeatureState { + return stateFor(featureKey: subfeature.parent, versionProvider: versionProvider) + } + + public func stateFor(_ subfeature: any PrivacySubfeature, + cohortID: CohortID?, + versionProvider: AppVersionProvider, + randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - // Check parent feature state + // Step 1: Check parent feature state let parentState = stateFor(featureKey: subfeature.parent, versionProvider: versionProvider) guard case .enabled = parentState else { return parentState } - // Check sub-feature state + // Step 2: Retrieve subfeature data and check version let subfeatures = subfeatures(for: subfeature.parent) let subfeatureData = subfeatures[subfeature.rawValue] let satisfiesMinVersion = satisfiesMinVersion(subfeatureData?.minSupportedVersion, versionProvider: versionProvider) + // Step 3: Check sub-feature state switch subfeatureData?.state { case PrivacyConfigurationData.State.enabled: guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } @@ -215,6 +227,48 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { default: return .disabled(.disabledInConfig) } + // Step 4: Check if a cohort was passed in the func + // If no corhort passed check for Target and Rollout + guard let passedCohort = cohortID else { + return checkTargetAndRollouts(subfeatureData, subfeature: subfeature, randomizer: randomizer) + } + + // Step 5: Verify there are cohorts in the subfeature data + // If not remove cohort (in case it was previously assigned) + // and check for Target and Rollout + guard let cohorts = subfeatureData?.cohorts else { + experimentManager.removeCohort(from: subfeature.rawValue) + return checkTargetAndRollouts(subfeatureData, subfeature: subfeature, randomizer: randomizer) + } + + // Step 6: Verify there the cohorts in the subfeature contain the cohort passed in the func + // If not remove cohort (in case it was previously assigned) before proceeding + if !cohorts.contains(where: { $0.name == passedCohort + }) { + experimentManager.removeCohort(from: subfeature.rawValue) + } + + // Step 7: Check if a cohort was already assigned + // If so check if it matches the one passed in the func and return .enable or disabled accordingly + if let assignedCohort = experimentManager.cohort(for: subfeature.rawValue) { + return (assignedCohort == passedCohort) ? .enabled : .disabled(.experimentCohortDoesNotMatch) + } + + // Step 8: check Target and Rollout + // if disabled return .disabled otherwise continue + let targetAndRolloutState = checkTargetAndRollouts(subfeatureData, subfeature: subfeature, randomizer: randomizer) + if targetAndRolloutState != .enabled { + return targetAndRolloutState + } + + // Step 9: Assign cohort and check if they match + let newAssignedCohort = experimentManager.assignCohort(to: ExperimentSubfeature(subfeatureID: subfeature.rawValue, cohorts: cohorts)) + return (newAssignedCohort == passedCohort) ? .enabled : .disabled(.experimentCohortDoesNotMatch) + } + + private func checkTargetAndRollouts(_ subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature?, + subfeature: any PrivacySubfeature, + randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { // Check Targets // It should not be wrapped in an array and will be removed at some point if let target = subfeatureData?.targets?.first, !matchTarget(target: target){ @@ -230,6 +284,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { return .enabled } + private func matchTarget(target: PrivacyConfigurationData.PrivacyFeature.Feature.Target) -> Bool{ return target.localeCountry == locale.regionCode && target.localeLanguage == locale.languageCode diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index 271b934d0..19f8a5355 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -18,22 +18,22 @@ import Foundation -struct ExperimentSubfeature { +public struct ExperimentSubfeature { let subfeatureID: SubfeatureID let cohorts: [PrivacyConfigurationData.Cohort] } -typealias CohortID = String -typealias SubfeatureID = String +public typealias CohortID = String +public typealias SubfeatureID = String -struct ExperimentData: Codable, Equatable { +public struct ExperimentData: Codable, Equatable { let cohort: String let enrollmentDate: Date } -typealias Experiments = [String: ExperimentData] +public typealias Experiments = [String: ExperimentData] -protocol ExperimentCohortsManaging { +public protocol ExperimentCohortsManaging { /// Retrieves the cohort ID associated with the specified subfeature. /// - Parameter subfeature: The experiment subfeature for which the cohort ID is needed. /// - Returns: The cohort ID as a `String` if one exists; otherwise, returns `nil`. @@ -54,7 +54,7 @@ protocol ExperimentCohortsManaging { func removeCohort(from subfeatureID: SubfeatureID) } -final class ExperimentCohortsManager: ExperimentCohortsManaging { +public final class ExperimentCohortsManager: ExperimentCohortsManaging { private var store: ExperimentsDataStoring private let randomizer: (Range) -> Double @@ -63,22 +63,23 @@ final class ExperimentCohortsManager: ExperimentCohortsManaging { store.experiments } - init(store: ExperimentsDataStoring = ExperimentsDataStore(), randomizer: @escaping (Range) -> Double) { + public init(store: ExperimentsDataStoring = ExperimentsDataStore(), + randomizer: @escaping (Range) -> Double = Double.random(in:)) { self.store = store self.randomizer = randomizer } - func cohort(for subfeatureID: SubfeatureID) -> CohortID? { + public func cohort(for subfeatureID: SubfeatureID) -> CohortID? { guard let experiments = experiments else { return nil } return experiments[subfeatureID]?.cohort } - func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? { + public func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? { guard let experiments = experiments else { return nil } return experiments[subfeatureID]?.enrollmentDate } - func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? { + public func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? { let cohorts = subfeature.cohorts let totalWeight = cohorts.map(\.weight).reduce(0, +) guard totalWeight > 0 else { return nil } @@ -96,7 +97,7 @@ final class ExperimentCohortsManager: ExperimentCohortsManaging { return nil } - func removeCohort(from subfeatureID: SubfeatureID) { + public func removeCohort(from subfeatureID: SubfeatureID) { guard var experiments = experiments else { return } experiments.removeValue(forKey: subfeatureID) saveExperiment(experiments) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift index 338c575d7..bb43ab426 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift @@ -18,16 +18,16 @@ import Foundation -protocol ExperimentsDataStoring { +public protocol ExperimentsDataStoring { var experiments: Experiments? { get set } } -protocol LocalDataStoring { +public protocol LocalDataStoring { func data(forKey defaultName: String) -> Data? func set(_ value: Any?, forKey defaultName: String) } -struct ExperimentsDataStore: ExperimentsDataStoring { +public struct ExperimentsDataStore: ExperimentsDataStoring { private enum Constants { static let experimentsDataKey = "ExperimentsData" @@ -37,13 +37,13 @@ struct ExperimentsDataStore: ExperimentsDataStoring { private let decoder = JSONDecoder() private let encoder = JSONEncoder() - init(localDataStoring: LocalDataStoring = UserDefaults.standard) { + public init(localDataStoring: LocalDataStoring = UserDefaults.standard) { self.localDataStoring = localDataStoring encoder.dateEncodingStrategy = .secondsSince1970 decoder.dateDecodingStrategy = .secondsSince1970 } - var experiments: Experiments? { + public var experiments: Experiments? { get { queue.sync { guard let savedData = localDataStoring.data(forKey: Constants.experimentsDataKey) else { return nil } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift index a5b63ea30..94304aa27 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift @@ -31,6 +31,7 @@ public enum PrivacyConfigurationFeatureDisabledReason: Equatable { case limitedToInternalUsers case stillInRollout case targetDoesNotMatch + case experimentCohortDoesNotMatch } public protocol PrivacyConfiguration { @@ -57,8 +58,8 @@ public protocol PrivacyConfiguration { func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool func stateFor(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> PrivacyConfigurationFeatureState - func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool - func stateFor(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState + func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool + func stateFor(_ subfeature: any PrivacySubfeature, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState /// Domains for which given PrivacyFeature is disabled. /// @@ -114,11 +115,11 @@ public extension PrivacyConfiguration { return stateFor(featureKey: featureKey, versionProvider: AppVersionProvider()) } - func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, randomizer: (Range) -> Double = Double.random(in:)) -> Bool { - return isSubfeatureEnabled(subfeature, versionProvider: AppVersionProvider(), randomizer: randomizer) + func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, cohortID: CohortID? = nil, randomizer: (Range) -> Double = Double.random(in:)) -> Bool { + return isSubfeatureEnabled(subfeature, cohortID: cohortID, versionProvider: AppVersionProvider(), randomizer: randomizer) } - func stateFor(_ subfeature: any PrivacySubfeature, randomizer: (Range) -> Double = Double.random(in:)) -> PrivacyConfigurationFeatureState { - return stateFor(subfeature, versionProvider: AppVersionProvider(), randomizer: randomizer) + func stateFor(_ subfeature: any PrivacySubfeature, cohortID: CohortID? = nil, randomizer: (Range) -> Double = Double.random(in:)) -> PrivacyConfigurationFeatureState { + return stateFor(subfeature, cohortID: cohortID, versionProvider: AppVersionProvider(), randomizer: randomizer) } } diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift b/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift index e1cb475cd..78d527308 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift @@ -318,15 +318,11 @@ class PrivacyConfigurationMock: PrivacyConfiguration { return .enabled } - func isSubfeatureEnabled( - _ subfeature: any PrivacySubfeature, - versionProvider: AppVersionProvider, - randomizer: (Range) -> Double - ) -> Bool { + func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { true } - func stateFor(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { return .enabled } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift index 9388b14e9..e3be10a1f 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift @@ -586,13 +586,13 @@ class AppPrivacyConfigurationTests: XCTestCase { let config = manager.privacyConfig let oldVersionProvider = MockAppVersionProvider(appVersion: "1.35.0") - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: oldVersionProvider, randomizer: Double.random(in:))) - XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, versionProvider: oldVersionProvider, randomizer: Double.random(in:)), .disabled(.appVersionNotSupported)) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: oldVersionProvider, randomizer: Double.random(in:))) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: oldVersionProvider, randomizer: Double.random(in:)), .disabled(.appVersionNotSupported)) let currentVersionProvider = MockAppVersionProvider(appVersion: "1.36.0") - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: currentVersionProvider, randomizer: Double.random(in:))) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: currentVersionProvider, randomizer: Double.random(in:))) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) let futureVersionProvider = MockAppVersionProvider(appVersion: "2.16.0") - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: futureVersionProvider, randomizer: Double.random(in:))) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: futureVersionProvider, randomizer: Double.random(in:))) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) } @@ -661,12 +661,12 @@ class AppPrivacyConfigurationTests: XCTestCase { let oldVersionProvider = MockAppVersionProvider(appVersion: "1.35.0") XCTAssertFalse(config.isEnabled(featureKey: .autofill, versionProvider: oldVersionProvider)) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: oldVersionProvider, randomizer: Double.random(in:))) - XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, versionProvider: oldVersionProvider, randomizer: Double.random(in:)), .disabled(.appVersionNotSupported)) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: oldVersionProvider, randomizer: Double.random(in:))) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: oldVersionProvider, randomizer: Double.random(in:)), .disabled(.appVersionNotSupported)) let currentVersionProvider = MockAppVersionProvider(appVersion: "1.36.0") XCTAssertTrue(config.isEnabled(featureKey: .autofill, versionProvider: currentVersionProvider)) - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: currentVersionProvider, randomizer: Double.random(in:))) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: currentVersionProvider, randomizer: Double.random(in:))) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) } diff --git a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift index a85dadc1e..0334f4f7a 100644 --- a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift +++ b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift @@ -221,11 +221,11 @@ class MockPrivacyConfiguration: PrivacyConfiguration { var isSubfeatureEnabledCheck: ((any PrivacySubfeature) -> Bool)? - func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool { + func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { isSubfeatureEnabledCheck?(subfeature) ?? false } - func stateFor(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { if isSubfeatureEnabledCheck?(subfeature) == true { return .enabled } diff --git a/Tests/DDGSyncTests/Mocks/Mocks.swift b/Tests/DDGSyncTests/Mocks/Mocks.swift index 013123c6d..b65b8abdb 100644 --- a/Tests/DDGSyncTests/Mocks/Mocks.swift +++ b/Tests/DDGSyncTests/Mocks/Mocks.swift @@ -169,15 +169,11 @@ class MockPrivacyConfiguration: PrivacyConfiguration { return .enabled } - func isSubfeatureEnabled( - _ subfeature: any PrivacySubfeature, - versionProvider: AppVersionProvider, - randomizer: (Range) -> Double - ) -> Bool { + func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { true } - func stateFor(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { return .enabled } From 8bda08cea171d7838fe0810de9fab550c64b8d12 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Thu, 14 Nov 2024 13:06:43 +0100 Subject: [PATCH 14/49] add tests --- .../AppPrivacyConfiguration.swift | 153 ++-- .../ExperimentCohortsManager.swift | 13 +- .../Features/PrivacyFeature.swift | 2 +- .../PrivacyConfigurationData.swift | 8 +- .../PrivacyConfigurationManager.swift | 7 +- ...dPrivacyConfigurationExperimentTests.swift | 749 ++++++++++++++++++ .../ExperimentCohortsManagerTests.swift | 12 +- .../ExperimentsDataStoreTests.swift | 8 +- 8 files changed, 862 insertions(+), 90 deletions(-) create mode 100644 Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 5a55c2ba9..49d5c87ac 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -36,6 +36,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { private let locale: Locale private let experimentManager: ExperimentCohortsManaging private let installDate: Date? + private let experimentManagerQueue = DispatchQueue(label: "com.experimentManager.queue") public init(data: PrivacyConfigurationData, identifier: String, @@ -198,96 +199,110 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } } - private func isParentFeatureEnabed(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider) -> PrivacyConfigurationFeatureState { - return stateFor(featureKey: subfeature.parent, versionProvider: versionProvider) - } - - public func stateFor(_ subfeature: any PrivacySubfeature, + public func stateFor(experiment: ExperimentData, + parentID: String, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - - // Step 1: Check parent feature state - let parentState = stateFor(featureKey: subfeature.parent, versionProvider: versionProvider) + let parentFeature = PrivacyFeature(rawValue: parentID)! + let parentState = stateFor(featureKey: parentFeature, versionProvider: versionProvider) guard case .enabled = parentState else { return parentState } + let subfeatures = subfeatures(for: parentFeature) + let subfeatureData = subfeatures[subfeatureID] + subfeatureData. - // Step 2: Retrieve subfeature data and check version - let subfeatures = subfeatures(for: subfeature.parent) - let subfeatureData = subfeatures[subfeature.rawValue] - - let satisfiesMinVersion = satisfiesMinVersion(subfeatureData?.minSupportedVersion, versionProvider: versionProvider) - - // Step 3: Check sub-feature state - switch subfeatureData?.state { - case PrivacyConfigurationData.State.enabled: - guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } - case PrivacyConfigurationData.State.internal: - guard internalUserDecider.isInternalUser else { return .disabled(.limitedToInternalUsers) } - guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } - default: return .disabled(.disabledInConfig) - } + } - // Step 4: Check if a cohort was passed in the func - // If no corhort passed check for Target and Rollout - guard let passedCohort = cohortID else { - return checkTargetAndRollouts(subfeatureData, subfeature: subfeature, randomizer: randomizer) - } + public func getAllActiveExperiments() -> Experiments { - // Step 5: Verify there are cohorts in the subfeature data - // If not remove cohort (in case it was previously assigned) - // and check for Target and Rollout - guard let cohorts = subfeatureData?.cohorts else { - experimentManager.removeCohort(from: subfeature.rawValue) - return checkTargetAndRollouts(subfeatureData, subfeature: subfeature, randomizer: randomizer) - } + } - // Step 6: Verify there the cohorts in the subfeature contain the cohort passed in the func - // If not remove cohort (in case it was previously assigned) before proceeding - if !cohorts.contains(where: { $0.name == passedCohort - }) { - experimentManager.removeCohort(from: subfeature.rawValue) - } + public func stateFor(_ subfeature: any PrivacySubfeature, + cohortID: CohortID?, + versionProvider: AppVersionProvider, + randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + return experimentManagerQueue.sync { + // Step 1: Check parent feature state + let parentState = stateFor(featureKey: subfeature.parent, versionProvider: versionProvider) + guard case .enabled = parentState else { return parentState } + + // Step 2: Retrieve subfeature data and check version + let subfeatures = subfeatures(for: subfeature.parent) + let subfeatureData = subfeatures[subfeature.rawValue] + + let satisfiesMinVersion = satisfiesMinVersion(subfeatureData?.minSupportedVersion, versionProvider: versionProvider) + + // Step 3: Check sub-feature state + switch subfeatureData?.state { + case PrivacyConfigurationData.State.enabled: + guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } + case PrivacyConfigurationData.State.internal: + guard internalUserDecider.isInternalUser else { return .disabled(.limitedToInternalUsers) } + guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } + default: return .disabled(.disabledInConfig) + } - // Step 7: Check if a cohort was already assigned - // If so check if it matches the one passed in the func and return .enable or disabled accordingly - if let assignedCohort = experimentManager.cohort(for: subfeature.rawValue) { - return (assignedCohort == passedCohort) ? .enabled : .disabled(.experimentCohortDoesNotMatch) - } + // // Step 4: Handle Rollouts + if let rollout = subfeatureData?.rollout, + !isRolloutEnabled(subfeature: subfeature, rolloutSteps: rollout.steps, randomizer: randomizer) { + return .disabled(.stillInRollout) + } - // Step 8: check Target and Rollout - // if disabled return .disabled otherwise continue - let targetAndRolloutState = checkTargetAndRollouts(subfeatureData, subfeature: subfeature, randomizer: randomizer) - if targetAndRolloutState != .enabled { - return targetAndRolloutState + // Step 5: Check if a cohort was passed in the func + // If no corhort passed check for Target and Rollout + guard let passedCohort = cohortID else { + return checkTargets(subfeatureData) + } + + // Step 6: Verify there are cohorts in the subfeature data + // If not remove cohort (in case it was previously assigned) + // and check for Target and Rollout + guard let cohorts = subfeatureData?.cohorts else { + experimentManager.removeCohort(from: subfeature.rawValue) + return .disabled(.experimentCohortDoesNotMatch) + } + + // Step 7: Verify there the cohorts in the subfeature contain the cohort passed in the func + // If not remove cohort (in case it was previously assigned) before proceeding + if !cohorts.contains(where: { $0.name == passedCohort + }) { + experimentManager.removeCohort(from: subfeature.rawValue) + } + + // Step 8: Check if a cohort was already assigned + // If so check if it matches the one passed in the func and return .enable or disabled accordingly + if let assignedCohort = experimentManager.cohort(for: subfeature.rawValue) { + return (assignedCohort == passedCohort) ? .enabled : .disabled(.experimentCohortDoesNotMatch) + } + + // Step 9: check Target and Rollout + // if disabled return .disabled otherwise continue + let targetsState = checkTargets(subfeatureData) + if targetsState != .enabled { + return targetsState + } + + // Step 10: Assign cohort and check if they match + let newAssignedCohort = experimentManager.assignCohort(to: ExperimentSubfeature(parentID: subfeature.parent.rawValue, subfeatureID: subfeature.rawValue, cohorts: cohorts)) + return (newAssignedCohort == passedCohort) ? .enabled : .disabled(.experimentCohortDoesNotMatch) } - - // Step 9: Assign cohort and check if they match - let newAssignedCohort = experimentManager.assignCohort(to: ExperimentSubfeature(subfeatureID: subfeature.rawValue, cohorts: cohorts)) - return (newAssignedCohort == passedCohort) ? .enabled : .disabled(.experimentCohortDoesNotMatch) } - private func checkTargetAndRollouts(_ subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature?, - subfeature: any PrivacySubfeature, - randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + private func checkTargets(_ subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature?) -> PrivacyConfigurationFeatureState { // Check Targets - // It should not be wrapped in an array and will be removed at some point - if let target = subfeatureData?.targets?.first, !matchTarget(target: target){ + if let targets = subfeatureData?.targets, !matchTargets(targets: targets){ return .disabled(.targetDoesNotMatch) } - // Handle Rollouts - if let rollout = subfeatureData?.rollout, - !isRolloutEnabled(subfeature: subfeature, rolloutSteps: rollout.steps, randomizer: randomizer) { - return .disabled(.stillInRollout) - } - return .enabled } - private func matchTarget(target: PrivacyConfigurationData.PrivacyFeature.Feature.Target) -> Bool{ - return target.localeCountry == locale.regionCode && - target.localeLanguage == locale.languageCode + private func matchTargets(targets: [PrivacyConfigurationData.PrivacyFeature.Feature.Target]) -> Bool { + return targets.contains { target in + (target.localeCountry == nil || target.localeCountry == locale.regionCode) && + (target.localeLanguage == nil || target.localeLanguage == locale.languageCode) + } } private func subfeatures(for feature: PrivacyFeature) -> PrivacyConfigurationData.PrivacyFeature.Features { diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index 19f8a5355..5872e03ad 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -19,16 +19,19 @@ import Foundation public struct ExperimentSubfeature { + let parentID: ParentFeatureID let subfeatureID: SubfeatureID let cohorts: [PrivacyConfigurationData.Cohort] } public typealias CohortID = String public typealias SubfeatureID = String +public typealias ParentFeatureID = String public struct ExperimentData: Codable, Equatable { - let cohort: String - let enrollmentDate: Date + public let parentID: String + public let cohort: String + public let enrollmentDate: Date } public typealias Experiments = [String: ExperimentData] @@ -90,7 +93,7 @@ public final class ExperimentCohortsManager: ExperimentCohortsManaging { for cohort in cohorts { cumulativeWeight += Double(cohort.weight) if randomValue < cumulativeWeight { - saveCohort(cohort.name, in: subfeature.subfeatureID) + saveCohort(cohort.name, in: subfeature.subfeatureID, parentID: subfeature.parentID) return cohort.name } } @@ -107,9 +110,9 @@ public final class ExperimentCohortsManager: ExperimentCohortsManaging { store.experiments = experiments } - private func saveCohort(_ cohort: CohortID, in experimentID: SubfeatureID) { + private func saveCohort(_ cohort: CohortID, in experimentID: SubfeatureID, parentID: ParentFeatureID) { var experiments = experiments ?? Experiments() - let experimentData = ExperimentData(cohort: cohort, enrollmentDate: Date()) + let experimentData = ExperimentData(parentID: parentID, cohort: cohort, enrollmentDate: Date()) experiments[experimentID] = experimentData saveExperiment(experiments) } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/Features/PrivacyFeature.swift b/Sources/BrowserServicesKit/PrivacyConfig/Features/PrivacyFeature.swift index f21c9dc12..027b210c2 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/Features/PrivacyFeature.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/Features/PrivacyFeature.swift @@ -19,7 +19,7 @@ import Foundation /// Features whose `rawValue` should be the key to access their corresponding `PrivacyConfigurationData.PrivacyFeature` object -public enum PrivacyFeature: String { +public enum PrivacyFeature: String, CaseIterable { case contentBlocking case duckPlayer case fingerprintingTemporaryStorage diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift index 42df09e91..2d67ef5e4 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift @@ -177,12 +177,12 @@ public struct PrivacyConfigurationData { case localeLanguage } - public let localeCountry: String - public let localeLanguage: String + public let localeCountry: String? + public let localeLanguage: String? public init(json: [String: Any]) { - self.localeCountry = json[CodingKeys.localeCountry.rawValue] as? String ?? "" - self.localeLanguage = json[CodingKeys.localeLanguage.rawValue] as? String ?? "" + self.localeCountry = json[CodingKeys.localeCountry.rawValue] as? String + self.localeLanguage = json[CodingKeys.localeLanguage.rawValue] as? String } } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index 004729e70..7b216b04b 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -56,6 +56,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { private let errorReporting: EventMapping? private let installDate: Date? private let locale: Locale + private let experimentCohortManager: ExperimentCohortsManaging public let internalUserDecider: InternalUserDecider @@ -112,12 +113,14 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { errorReporting: EventMapping? = nil, internalUserDecider: InternalUserDecider, locale: Locale = Locale.current, + experimentCohortManager: ExperimentCohortsManaging = ExperimentCohortsManager(), installDate: Date? = nil ) { self.embeddedDataProvider = embeddedDataProvider self.localProtection = localProtection self.errorReporting = errorReporting self.internalUserDecider = internalUserDecider + self.experimentCohortManager = experimentCohortManager self.locale = locale self.installDate = installDate @@ -131,6 +134,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { localProtection: localProtection, internalUserDecider: internalUserDecider, locale: locale, + experimentManager: experimentCohortManager, installDate: installDate) } @@ -138,7 +142,8 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { identifier: embeddedConfigData.etag, localProtection: localProtection, internalUserDecider: internalUserDecider, - locale: locale, + locale: locale, + experimentManager: experimentCohortManager, installDate: installDate) } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift new file mode 100644 index 000000000..ebd15b267 --- /dev/null +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift @@ -0,0 +1,749 @@ +// +// AddPrivacyConfigurationExperimentTests.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import XCTest +import BrowserServicesKit + +final class AddPrivacyConfigurationExperimentTests: XCTestCase { + + var featureJson: Data = "{}".data(using: .utf8)! + var mockEmbeddedData: MockEmbeddedDataProvider! + var mockStore: MockExperimentDataStore! + var experimentManager: ExperimentCohortsManager! + var manager: PrivacyConfigurationManager! + var locale: Locale! + + let subfeatureName = "credentialsSaving" + + + override func setUp() { + locale = Locale(identifier: "fr_US") + mockEmbeddedData = MockEmbeddedDataProvider(data: featureJson, etag: "test") + let mockInternalUserStore = MockInternalUserStoring() + mockStore = MockExperimentDataStore() + experimentManager = ExperimentCohortsManager(store: mockStore) + manager = PrivacyConfigurationManager(fetchedETag: nil, + fetchedData: nil, + embeddedDataProvider: mockEmbeddedData, + localProtection: MockDomainsProtectionStore(), + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + locale: locale, + experimentCohortManager: experimentManager) + } + + override func tearDown() { + featureJson = "".data(using: .utf8)! + mockEmbeddedData = nil + mockStore = nil + experimentManager = nil + manager = nil + } + + + func testCohortOnlyAssignedWhenCallingStateForSubfeature() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + let config = manager.privacyConfig + + // we haven't called isEnabled yet, so cohorts should not be yet assigned + XCTAssertNil(mockStore.experiments) + XCTAssertNil(experimentManager.cohort(for: subfeatureName)) + + // we call isEnabled() without cohort, cohort should not be assigned either + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) + XCTAssertNil(mockStore.experiments) + XCTAssertNil(experimentManager.cohort(for: subfeatureName)) + + // we call isEnabled(cohort), then we should assign cohort + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, cohortID: "blue"), .disabled(.experimentCohortDoesNotMatch)) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + } + + func testRemoveAllCohortsRemotelyRemovesAssignedCohort() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + var config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // remove blue cohort + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // remove all remaining cohorts + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2 + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) + XCTAssertNil(experimentManager.cohort(for: subfeatureName)) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + } + + func testDisablingFeatureDisablesCohort() { + // Initially subfeature for both cohorts is disabled + var config = manager.privacyConfig + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(mockStore.experiments) + XCTAssertNil(experimentManager.cohort(for: subfeatureName)) + + // When features with cohort the cohort with weight 1 is enabled + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // If the subfeature is then disabled isSubfeatureEnabled should return false + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "disabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // If the subfeature is parent feature disabled isSubfeatureEnabled should return false + featureJson = + """ + { + "features": { + "autofill": { + "state": "disabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + } + + func testCohortsAndTargetsInteraction() { + func featureJson(country: String, language: String) -> Data { + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeLanguage": "\(language)", + "localeCountry": "\(country)" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + } + manager.reload(etag: "", data: featureJson(country: "FR", language: "fr")) + var config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(mockStore.experiments) + XCTAssertNil(experimentManager.cohort(for: subfeatureName)) + + manager.reload(etag: "", data: featureJson(country: "US", language: "en")) + config = manager.privacyConfig + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(mockStore.experiments) + XCTAssertNil(experimentManager.cohort(for: subfeatureName)) + + manager.reload(etag: "", data: featureJson(country: "US", language: "fr")) + config = manager.privacyConfig + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // once cohort is assigned, changing targets shall not affect feature state + manager.reload(etag: "", data: featureJson(country: "IT", language: "it")) + config = manager.privacyConfig + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + let featureJson2 = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "FR" + } + ], + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson2) + config = manager.privacyConfig + + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) + XCTAssertNil(experimentManager.cohort(for: subfeatureName)) + + // re-populate experiment to re-assign new cohort, should not be assigned as it has wrong targets + manager.reload(etag: "", data: featureJson(country: "IT", language: "it")) + config = manager.privacyConfig + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) + XCTAssertNil(experimentManager.cohort(for: subfeatureName)) + } + + func testChangeRemoteCohortsAfterAssignmentShouldNoop() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + var config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // changing targets should not change cohort assignment + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "IT" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // changing cohort weight should not change current assignment + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 0 + }, + { + "name": "blue", + "weight": 1 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // adding cohorts should not change current assignment + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 1 + }, + { + "name": "red", + "weight": 1 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + } + + func testEnrollmentDate() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "", data: featureJson) + var config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertTrue(mockStore.experiments?.isEmpty ?? true) + XCTAssertNil(experimentManager.cohort(for: subfeatureName), "control") + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + let currentTime = Date().timeIntervalSince1970 + let enrollmentTime = mockStore.experiments?[subfeatureName]?.enrollmentDate.timeIntervalSince1970 + + XCTAssertNotNil(enrollmentTime) + if let enrollmentTime = enrollmentTime { + let tolerance: TimeInterval = 60 // 1 minute in seconds + XCTAssertEqual(currentTime, enrollmentTime, accuracy: tolerance) + } + } + + func testRollbackCohortExperiments() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "rollout": { + "steps": [ + { + "percent": 100 + } + ] + }, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "foo", data: featureJson) + var config = manager.privacyConfig + clearRolloutData(feature: "autofill", subFeature: "credentialsSaving") + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "rollout": { + "steps": [ + { + "percent": 0 + } + ] + }, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "foo", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + } + + func clearRolloutData(feature: String, subFeature: String) { + UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).enabled") + UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).lastRolloutCount") + } +} diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index 518249560..b4bdb88a1 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -45,10 +45,10 @@ final class ExperimentCohortsManagerTests: XCTestCase { ) let expectedDate1 = Date() - experimentData1 = ExperimentData(cohort: "TestCohort1", enrollmentDate: expectedDate1) + experimentData1 = ExperimentData(parentID: "TestParent", cohort: "TestCohort1", enrollmentDate: expectedDate1) let expectedDate2 = Date().addingTimeInterval(60) - experimentData2 = ExperimentData(cohort: "TestCohort2", enrollmentDate: expectedDate2) + experimentData2 = ExperimentData(parentID: "TestParent", cohort: "TestCohort2", enrollmentDate: expectedDate2) } override func tearDown() { @@ -135,7 +135,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { func testAssignCohortReturnsNilIfNoCohorts() { // GIVEN - let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: []) + let subfeature = ExperimentSubfeature(parentID: "parent", subfeatureID: subfeatureName1, cohorts: []) // WHEN let result = experimentCohortsManager.assignCohort(to: subfeature) @@ -152,7 +152,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { PrivacyConfigurationData.Cohort(json: jsonCohort1)!, PrivacyConfigurationData.Cohort(json: jsonCohort2)! ] - let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: cohorts) + let subfeature = ExperimentSubfeature(parentID: "parent", subfeatureID: subfeatureName1, cohorts: cohorts) // WHEN let result = experimentCohortsManager.assignCohort(to: subfeature) @@ -170,7 +170,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { PrivacyConfigurationData.Cohort(json: jsonCohort1)!, PrivacyConfigurationData.Cohort(json: jsonCohort2)! ] - let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: cohorts) + let subfeature = ExperimentSubfeature(parentID: "parent", subfeatureID: subfeatureName1, cohorts: cohorts) let expectedTotalWeight = 4.0 // Use a custom randomizer to verify the range @@ -232,7 +232,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { let cohorts = [ PrivacyConfigurationData.Cohort(json: jsonCohort1)! ] - let subfeature = ExperimentSubfeature(subfeatureID: subfeatureName1, cohorts: cohorts) + let subfeature = ExperimentSubfeature(parentID: "parent", subfeatureID: subfeatureName1, cohorts: cohorts) let expectedTotalWeight = 1.0 // Use a custom randomizer to verify the range diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift index 0466155b0..e5deb4f60 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift @@ -47,8 +47,8 @@ final class ExperimentsDataStoreTests: XCTestCase { func testExperimentsGetReturnsDecodedExperiments() { // GIVEN - let experimentData1 = ExperimentData(cohort: "TestCohort1", enrollmentDate: Date()) - let experimentData2 = ExperimentData(cohort: "TestCohort2", enrollmentDate: Date()) + let experimentData1 = ExperimentData(parentID: "parent", cohort: "TestCohort1", enrollmentDate: Date()) + let experimentData2 = ExperimentData(parentID: "parent", cohort: "TestCohort2", enrollmentDate: Date()) let experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] let encoder = JSONEncoder() @@ -71,8 +71,8 @@ final class ExperimentsDataStoreTests: XCTestCase { func testExperimentsSetEncodesAndStoresData() throws { // GIVEN - let experimentData1 = ExperimentData(cohort: "TestCohort1", enrollmentDate: Date()) - let experimentData2 = ExperimentData(cohort: "TestCohort2", enrollmentDate: Date()) + let experimentData1 = ExperimentData(parentID: "parent", cohort: "TestCohort1", enrollmentDate: Date()) + let experimentData2 = ExperimentData(parentID: "parent", cohort: "TestCohort2", enrollmentDate: Date()) let experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] // WHEN From 56c54a777fb2339997d3e79974c50028a2cf5cd5 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 15 Nov 2024 11:39:08 +0100 Subject: [PATCH 15/49] refactor --- .../AppPrivacyConfiguration.swift | 126 +++++------ .../ExperimentCohortsManager.swift | 80 ++++--- .../PrivacyConfig/PrivacyConfiguration.swift | 8 + .../PrivacyConfigurationData.swift | 9 + .../PrivacyConfigurationManager.swift | 2 +- .../ContentBlocker/WebViewTestHelper.swift | 12 +- .../GPC/GPCTests.swift | 3 +- ...dPrivacyConfigurationExperimentTests.swift | 201 ++++++++++++++++- .../ExperimentCohortsManagerTests.swift | 213 ++++++------------ .../ExperimentsDataStoreTests.swift | 4 +- .../PrivacyConfigurationDataTests.swift | 1 + .../PrivacyConfigurationReferenceTests.swift | 3 +- .../Resources/privacy-config-example.json | 5 +- 13 files changed, 419 insertions(+), 248 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index c99e89a4e..30f33784a 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -36,7 +36,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { private let locale: Locale private let experimentManager: ExperimentCohortsManaging private let installDate: Date? - private let experimentManagerQueue = DispatchQueue(label: "com.experimentManager.queue") + static let experimentManagerQueue = DispatchQueue(label: "com.experimentManager.queue") public init(data: PrivacyConfigurationData, identifier: String, @@ -44,7 +44,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { internalUserDecider: InternalUserDecider, userDefaults: UserDefaults = UserDefaults(), locale: Locale = Locale.current, - experimentManager: ExperimentCohortsManaging = ExperimentCohortsManager(), + experimentManager: ExperimentCohortsManaging, installDate: Date? = nil) { self.data = data self.identifier = identifier @@ -144,13 +144,14 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } } - private func isRolloutEnabled(subfeature: any PrivacySubfeature, + private func isRolloutEnabled(subfeatureID: SubfeatureID, + parentID: ParentFeatureID, rolloutSteps: [PrivacyConfigurationData.PrivacyFeature.Feature.RolloutStep], randomizer: (Range) -> Double) -> Bool { // Empty rollouts should be default enabled guard !rolloutSteps.isEmpty else { return true } - let defsPrefix = "config.\(subfeature.parent.rawValue).\(subfeature.rawValue)" + let defsPrefix = "config.\(parentID).\(subfeatureID)" if userDefaults.bool(forKey: "\(defsPrefix).\(Constants.enabledKey)") { return true } @@ -199,41 +200,55 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } } -// public func stateFor(experiment: ExperimentData, -// parentID: String, -// cohortID: CohortID?, -// versionProvider: AppVersionProvider, -// randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { -// let parentFeature = PrivacyFeature(rawValue: parentID)! -// let parentState = stateFor(featureKey: parentFeature, versionProvider: versionProvider) -// guard case .enabled = parentState else { return parentState } -// let subfeatures = subfeatures(for: parentFeature) -// let subfeatureData = subfeatures[subfeatureID] -// subfeatureData. -// -// } -// public func getAllActiveExperiments() -> Experiments { -// -// } + public func getAllActiveExperiments(versionProvider: AppVersionProvider, + randomizer: (Range) -> Double) -> Experiments { + Self.experimentManagerQueue.sync { + guard let assignedExperiments = experimentManager.experiments else { return [:] } + var experiments: Experiments = [:] + for (key, value) in assignedExperiments { + if stateFor(subfeatureID: key, experimentData: value, versionProvider: versionProvider, randomizer: randomizer) == .enabled { + experiments[key] = value + } + } + return experiments + } + } - public func stateFor(_ subfeature: any PrivacySubfeature, + private func stateFor(subfeatureID: SubfeatureID, experimentData: ExperimentData, versionProvider: AppVersionProvider, + randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + guard let parentFeature = PrivacyFeature(rawValue: experimentData.parentID) else { return .disabled(.featureMissing) } + let subfeatures = subfeatures(for: parentFeature) + guard let subfeatureData = subfeatures[subfeatureID] else { return .disabled(.featureMissing) } + return stateFor(parentFeature: parentFeature, subfeatureData: subfeatureData, subfeatureID: subfeatureID, cohortID: experimentData.cohort, versionProvider: versionProvider, randomizer: randomizer) + } + + public func stateFor(_ subfeature: any PrivacySubfeature, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - return experimentManagerQueue.sync { + let subfeatures = subfeatures(for: subfeature.parent) + guard let subfeatureData = subfeatures[subfeature.rawValue] else { return .disabled(.featureMissing) } + + return stateFor(parentFeature: subfeature.parent, subfeatureData: subfeatureData, subfeatureID: subfeature.rawValue, cohortID: cohortID, versionProvider: versionProvider, randomizer: randomizer) + } + + private func stateFor(parentFeature: PrivacyFeature, + subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature, + subfeatureID: SubfeatureID, + cohortID: CohortID?, + versionProvider: AppVersionProvider, + randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + Self.experimentManagerQueue.sync { // Step 1: Check parent feature state - let parentState = stateFor(featureKey: subfeature.parent, versionProvider: versionProvider) + let parentState = stateFor(featureKey: parentFeature, versionProvider: versionProvider) guard case .enabled = parentState else { return parentState } - // Step 2: Retrieve subfeature data and check version - let subfeatures = subfeatures(for: subfeature.parent) - let subfeatureData = subfeatures[subfeature.rawValue] - - let satisfiesMinVersion = satisfiesMinVersion(subfeatureData?.minSupportedVersion, versionProvider: versionProvider) + // Step 2: Check version + let satisfiesMinVersion = satisfiesMinVersion(subfeatureData.minSupportedVersion, versionProvider: versionProvider) // Step 3: Check sub-feature state - switch subfeatureData?.state { + switch subfeatureData.state { case PrivacyConfigurationData.State.enabled: guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } case PrivacyConfigurationData.State.internal: @@ -241,50 +256,33 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } default: return .disabled(.disabledInConfig) } - - // // Step 4: Handle Rollouts - if let rollout = subfeatureData?.rollout, - !isRolloutEnabled(subfeature: subfeature, rolloutSteps: rollout.steps, randomizer: randomizer) { + + // Step 4: Handle Rollouts + if let rollout = subfeatureData.rollout, + !isRolloutEnabled(subfeatureID: subfeatureID, parentID: parentFeature.rawValue, rolloutSteps: rollout.steps, randomizer: randomizer) { return .disabled(.stillInRollout) } - + // Step 5: Check if a cohort was passed in the func // If no corhort passed check for Target and Rollout guard let passedCohort = cohortID else { return checkTargets(subfeatureData) } - // Step 6: Verify there are cohorts in the subfeature data - // If not remove cohort (in case it was previously assigned) - // and check for Target and Rollout - guard let cohorts = subfeatureData?.cohorts else { - experimentManager.removeCohort(from: subfeature.rawValue) - return .disabled(.experimentCohortDoesNotMatch) - } - - // Step 7: Verify there the cohorts in the subfeature contain the cohort passed in the func - // If not remove cohort (in case it was previously assigned) before proceeding - if !cohorts.contains(where: { $0.name == passedCohort - }) { - experimentManager.removeCohort(from: subfeature.rawValue) - } - - // Step 8: Check if a cohort was already assigned - // If so check if it matches the one passed in the func and return .enable or disabled accordingly - if let assignedCohort = experimentManager.cohort(for: subfeature.rawValue) { - return (assignedCohort == passedCohort) ? .enabled : .disabled(.experimentCohortDoesNotMatch) - } - - // Step 9: check Target and Rollout - // if disabled return .disabled otherwise continue + // Step 6: Verify there the cohort + // Check if cohort assigned and matches passed cohort + // If cohort not assigned + // Tries to assign if matching target + // Check if cohort assigned and matches passed cohort + let cohorts = subfeatureData.cohorts ?? [] let targetsState = checkTargets(subfeatureData) - if targetsState != .enabled { - return targetsState + let assignedCohortResponse = experimentManager.cohort(for: ExperimentSubfeature(parentID: parentFeature.rawValue, subfeatureID: subfeatureID, cohorts: cohorts), assignIfEnabled: targetsState == .enabled) + let possibleDisabledReason: PrivacyConfigurationFeatureDisabledReason = assignedCohortResponse.didAttemptAssignment && targetsState != .enabled ? .targetDoesNotMatch : .experimentCohortDoesNotMatch + if let assignedCohort = assignedCohortResponse.cohortID { + return (assignedCohort == passedCohort) ? .enabled : .disabled(possibleDisabledReason) + } else { + return .disabled(possibleDisabledReason) } - - // Step 10: Assign cohort and check if they match - let newAssignedCohort = experimentManager.assignCohort(to: ExperimentSubfeature(parentID: subfeature.parent.rawValue, subfeatureID: subfeature.rawValue, cohorts: cohorts)) - return (newAssignedCohort == passedCohort) ? .enabled : .disabled(.experimentCohortDoesNotMatch) } } @@ -293,11 +291,9 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { if let targets = subfeatureData?.targets, !matchTargets(targets: targets){ return .disabled(.targetDoesNotMatch) } - return .enabled } - private func matchTargets(targets: [PrivacyConfigurationData.PrivacyFeature.Feature.Target]) -> Bool { return targets.contains { target in (target.localeCountry == nil || target.localeCountry == locale.regionCode) && diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index 78b404c53..2173160f8 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -37,48 +37,59 @@ public struct ExperimentData: Codable, Equatable { public typealias Experiments = [String: ExperimentData] public protocol ExperimentCohortsManaging { - /// Retrieves the cohort ID associated with the specified subfeature. - /// - Parameter subfeatureID: The name of the experiment subfeature for which the cohort ID is needed. - /// - Returns: The cohort ID as a `String` if one exists; otherwise, returns `nil`. - func cohort(for subfeatureID: SubfeatureID) -> CohortID? - - /// Retrieves the enrollment date for the specified subfeature. - /// - Parameter subfeatureID: The name of the experiment subfeature for which the enrollment date is needed. - /// - Returns: The `Date` of enrollment if one exists; otherwise, returns `nil`. - func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? - - /// Assigns a cohort to the given subfeature based on defined weights and saves it to UserDefaults. - /// - Parameter subfeature: The ExperimentSubfeature to which a cohort needs to be assigned to. - /// - Returns: The name of the assigned cohort, or `nil` if no cohort could be assigned. - func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? - - /// Removes the assigned cohort data for the specified subfeature. - /// - Parameter subfeatureID: The name of the experiment subfeature for which the cohort data should be removed. - func removeCohort(from subfeatureID: SubfeatureID) + /// Retrieves all the experiments a user is enrolled into + var experiments: Experiments? { get } + + /// Retrieves the assigned cohort for a given experiment subfeature, or attempts to assign a new cohort if none is currently assigned + /// and `assignIfEnabled` is set to true. If a cohort is already assigned but does not match any valid cohorts for the experiment, + /// the cohort will be removed. + /// + /// - Parameters: + /// - experiment: The `ExperimentSubfeature` for which to retrieve, assign, or remove a cohort. This subfeature includes + /// relevant identifiers and potential cohorts that may be assigned. + /// - assignIfEnabled: A Boolean value that determines whether a new cohort should be assigned if none is currently assigned. + /// If `true`, the function will attempt to assign a cohort from the available options; otherwise, it will only check for existing assignments. + /// + /// - Returns: A tuple containing: + /// - `cohortID`: The identifier of the assigned cohort if one exists, or `nil` if no cohort was assigned, if assignment failed, or if the cohort was removed. + /// - `didAttemptAssignment`: A Boolean indicating whether an assignment attempt was made. This will be `true` if `assignIfEnabled` + /// is `true` and no cohort was previously assigned, and `false` otherwise. + func cohort(for experiment: ExperimentSubfeature, assignIfEnabled: Bool) -> (cohortID: CohortID?, didAttemptAssignment: Bool) } -public final class ExperimentCohortsManager: ExperimentCohortsManaging { +public class ExperimentCohortsManager: ExperimentCohortsManaging { private var store: ExperimentsDataStoring private let randomizer: (Range) -> Double + private let queue = DispatchQueue(label: "com.ExperimentCohortsManager.queue") + public var experiments: Experiments? { + get { + queue.sync { + store.experiments + } + } + } - public init(store: ExperimentsDataStoring = ExperimentsDataStore(), randomizer: @escaping (Range) -> Double = Double.random(in:)) { + public init(store: ExperimentsDataStoring, randomizer: @escaping (Range) -> Double = Double.random(in:)) { self.store = store self.randomizer = randomizer } - public func cohort(for subfeatureID: SubfeatureID) -> CohortID? { - guard let experiments = store.experiments else { return nil } - return experiments[subfeatureID]?.cohort - } + public func cohort(for experiment: ExperimentSubfeature, assignIfEnabled: Bool) -> (cohortID: CohortID?, didAttemptAssignment: Bool) { + queue.sync { + let assignedCohort = cohort(for: experiment.subfeatureID) + if experiment.cohorts.contains(where: { $0.name == assignedCohort }) { + return (assignedCohort, false) + } else { + removeCohort(from: experiment.subfeatureID) + } - public func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? { - guard let experiments = store.experiments else { return nil } - return experiments[subfeatureID]?.enrollmentDate + return assignIfEnabled ? (assignCohort(to: experiment), true) : (nil, true) + } } - public func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? { + private func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? { let cohorts = subfeature.cohorts let totalWeight = cohorts.map(\.weight).reduce(0, +) guard totalWeight > 0 else { return nil } @@ -96,7 +107,17 @@ public final class ExperimentCohortsManager: ExperimentCohortsManaging { return nil } - public func removeCohort(from subfeatureID: SubfeatureID) { + func cohort(for subfeatureID: SubfeatureID) -> CohortID? { + guard let experiments = store.experiments else { return nil } + return experiments[subfeatureID]?.cohort + } + + private func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? { + guard let experiments = store.experiments else { return nil } + return experiments[subfeatureID]?.enrollmentDate + } + + private func removeCohort(from subfeatureID: SubfeatureID) { guard var experiments = store.experiments else { return } experiments.removeValue(forKey: subfeatureID) store.experiments = experiments @@ -108,4 +129,5 @@ public final class ExperimentCohortsManager: ExperimentCohortsManaging { experiments[experimentID] = experimentData store.experiments = experiments } + } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift index 94304aa27..9131566e0 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift @@ -104,6 +104,10 @@ public protocol PrivacyConfiguration { func userEnabledProtection(forDomain: String) /// Adds given domain to locally unprotected list. func userDisabledProtection(forDomain: String) + + /// Gives the list of all the active experiments an user is enrolled in + func getAllActiveExperiments(versionProvider: AppVersionProvider, + randomizer: (Range) -> Double) -> Experiments } public extension PrivacyConfiguration { @@ -122,4 +126,8 @@ public extension PrivacyConfiguration { func stateFor(_ subfeature: any PrivacySubfeature, cohortID: CohortID? = nil, randomizer: (Range) -> Double = Double.random(in:)) -> PrivacyConfigurationFeatureState { return stateFor(subfeature, cohortID: cohortID, versionProvider: AppVersionProvider(), randomizer: randomizer) } + + func getAllActiveExperiments(versionProvider: AppVersionProvider = AppVersionProvider(), randomizer: (Range) -> Double = Double.random(in:)) -> Experiments { + return getAllActiveExperiments(versionProvider: versionProvider, randomizer: randomizer) + } } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift index ec2e2ca20..f6f91567f 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationData.swift @@ -137,6 +137,7 @@ public struct PrivacyConfigurationData { case rollout case cohorts case targets + case settings } public struct Rollout: Hashable { @@ -190,6 +191,7 @@ public struct PrivacyConfigurationData { public let rollout: Rollout? public let cohorts: [Cohort]? public let targets: [Target]? + public let settings: String? public init?(json: [String: Any]) { guard let state = json[CodingKeys.state.rawValue] as? String else { @@ -217,6 +219,13 @@ public struct PrivacyConfigurationData { } else { targets = nil } + + if let settingsData = json[CodingKeys.settings.rawValue] { + let jsonData = try? JSONSerialization.data(withJSONObject: settingsData, options: []) + settings = jsonData != nil ? String(data: jsonData!, encoding: .utf8) : nil + } else { + settings = nil + } } } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index 7b216b04b..6fa567e3a 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -113,7 +113,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { errorReporting: EventMapping? = nil, internalUserDecider: InternalUserDecider, locale: Locale = Locale.current, - experimentCohortManager: ExperimentCohortsManaging = ExperimentCohortsManager(), + experimentCohortManager: ExperimentCohortsManaging = ExperimentCohortsManager(store: ExperimentsDataStore()), installDate: Date? = nil ) { self.embeddedDataProvider = embeddedDataProvider diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift index 31370ce4a..bced88b71 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift @@ -197,7 +197,7 @@ final class WebKitTestHelper { return AppPrivacyConfiguration(data: privacyData, identifier: "", localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), experimentManager: MockExperimentCohortsManager()) } static func prepareContentBlockingRules(trackerData: TrackerData, @@ -225,3 +225,13 @@ final class WebKitTestHelper { } } } + +class MockExperimentCohortsManager: ExperimentCohortsManaging { + var experiments: BrowserServicesKit.Experiments? + + func cohort(for experiment: BrowserServicesKit.ExperimentSubfeature, assignIfEnabled: Bool) -> (cohortID: BrowserServicesKit.CohortID?, didAttemptAssignment: Bool) { + return (nil, true) + } + + +} diff --git a/Tests/BrowserServicesKitTests/GPC/GPCTests.swift b/Tests/BrowserServicesKitTests/GPC/GPCTests.swift index 1eee4af76..8ee3ec2e2 100644 --- a/Tests/BrowserServicesKitTests/GPC/GPCTests.swift +++ b/Tests/BrowserServicesKitTests/GPC/GPCTests.swift @@ -42,7 +42,8 @@ final class GPCTests: XCTestCase { appConfig = AppPrivacyConfiguration(data: privacyData, identifier: "", localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + experimentManager: MockExperimentCohortsManager()) } func testWhenGPCEnableDomainIsHttpThenISGPCEnabledTrue() { diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift index ebd15b267..3ecc0eb9e 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift @@ -18,7 +18,7 @@ // import XCTest -import BrowserServicesKit +@testable import BrowserServicesKit final class AddPrivacyConfigurationExperimentTests: XCTestCase { @@ -200,6 +200,79 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) } + func testRemoveAssignedCohortsRemotelyRemovesAssignedCohortAndTriesToReassign() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "2", data: featureJson) + var config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // remove blue cohort + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "red", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "2", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "red")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "red") + } + func testDisablingFeatureDisablesCohort() { // Initially subfeature for both cohorts is disabled var config = manager.privacyConfig @@ -625,7 +698,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - var config = manager.privacyConfig + let config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertTrue(mockStore.experiments?.isEmpty ?? true) @@ -742,6 +815,130 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } + func testCohortEnabledAndStopEnrollmentAndRhenRollBack() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "foo", data: featureJson) + var config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // Stop enrollment, should keep assigned cohorts + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 0 + }, + { + "name": "blue", + "weight": 1 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "foo", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") + + // remove control, should re-allocate to blue + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "blue", + "weight": 1 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "foo", data: featureJson) + config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "blue") + } + + func clearRolloutData(feature: String, subFeature: String) { UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).enabled") UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).lastRolloutCount") diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift index b4bdb88a1..0a5114d50 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift @@ -21,6 +21,11 @@ import XCTest final class ExperimentCohortsManagerTests: XCTestCase { + let cohort1 = PrivacyConfigurationData.Cohort(json: ["name": "Cohort1", "weight": 1])! + let cohort2 = PrivacyConfigurationData.Cohort(json: ["name": "Cohort2", "weight": 0])! + let cohort3 = PrivacyConfigurationData.Cohort(json: ["name": "Cohort3", "weight": 2])! + let cohort4 = PrivacyConfigurationData.Cohort(json: ["name": "Cohort4", "weight": 0])! + var mockStore: MockExperimentDataStore! var experimentCohortsManager: ExperimentCohortsManager! @@ -30,6 +35,12 @@ final class ExperimentCohortsManagerTests: XCTestCase { let subfeatureName2 = "TestSubfeature2" var experimentData2: ExperimentData! + let subfeatureName3 = "TestSubfeature3" + var experimentData3: ExperimentData! + + let subfeatureName4 = "TestSubfeature4" + var experimentData4: ExperimentData! + let encoder: JSONEncoder = { let encoder = JSONEncoder() encoder.dateEncodingStrategy = .secondsSince1970 @@ -40,15 +51,20 @@ final class ExperimentCohortsManagerTests: XCTestCase { super.setUp() mockStore = MockExperimentDataStore() experimentCohortsManager = ExperimentCohortsManager( - store: mockStore, - randomizer: { _ in 50.0 } + store: mockStore ) let expectedDate1 = Date() - experimentData1 = ExperimentData(parentID: "TestParent", cohort: "TestCohort1", enrollmentDate: expectedDate1) + experimentData1 = ExperimentData(parentID: "TestParent", cohort: cohort1.name, enrollmentDate: expectedDate1) let expectedDate2 = Date().addingTimeInterval(60) - experimentData2 = ExperimentData(parentID: "TestParent", cohort: "TestCohort2", enrollmentDate: expectedDate2) + experimentData2 = ExperimentData(parentID: "TestParent", cohort: cohort2.name, enrollmentDate: expectedDate2) + + let expectedDate3 = Date() + experimentData3 = ExperimentData(parentID: "TestParent", cohort: cohort3.name, enrollmentDate: expectedDate3) + + let expectedDate4 = Date().addingTimeInterval(60) + experimentData4 = ExperimentData(parentID: "TestParent", cohort: cohort4.name, enrollmentDate: expectedDate4) } override func tearDown() { @@ -59,206 +75,117 @@ final class ExperimentCohortsManagerTests: XCTestCase { super.tearDown() } - func testCohortReturnsCohortIDIfExistsForMultipleSubfeatures() { + func testExperimentReturnAssignedExperiments() { // GIVEN mockStore.experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] // WHEN - let result1 = experimentCohortsManager.cohort(for: subfeatureName1) - let result2 = experimentCohortsManager.cohort(for: subfeatureName2) + let experiments = experimentCohortsManager.experiments // THEN - XCTAssertEqual(result1, experimentData1.cohort) - XCTAssertEqual(result2, experimentData2.cohort) + XCTAssertEqual(experiments?.count, 2) + XCTAssertEqual(experiments?[subfeatureName1], experimentData1) + XCTAssertEqual(experiments?[subfeatureName2], experimentData2) + XCTAssertNil(experiments?[subfeatureName3]) } - func testEnrollmentDateReturnsCorrectDateIfExists() { + func testCohortReturnsCohortIDIfExistsForMultipleSubfeatures() { // GIVEN - mockStore.experiments = [subfeatureName1: experimentData1] + mockStore.experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] // WHEN - let result1 = experimentCohortsManager.enrollmentDate(for: subfeatureName1) - let result2 = experimentCohortsManager.enrollmentDate(for: subfeatureName2) + let result1 = experimentCohortsManager.cohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort1, cohort2]), assignIfEnabled: false).cohortID + let result2 = experimentCohortsManager.cohort(for: ExperimentSubfeature(parentID: experimentData2.parentID, subfeatureID: subfeatureName2, cohorts: [cohort2, cohort3]), assignIfEnabled: false).cohortID // THEN - let timeDifference1 = abs(experimentData1.enrollmentDate.timeIntervalSince(result1 ?? Date())) - - XCTAssertLessThanOrEqual(timeDifference1, 1.0, "Expected enrollment date for subfeatureName1 to match at the second level") - XCTAssertNil(result2) + XCTAssertEqual(result1, experimentData1.cohort) + XCTAssertEqual(result2, experimentData2.cohort) } - func testCohortReturnsNilIfCohortDoesNotExist() { + func testCohortAssignIfEnabledWhenNoCohortExists() { // GIVEN - let subfeatureName = "TestSubfeature" + mockStore.experiments = [:] + let cohorts = [cohort1, cohort2] + let experiment = ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: cohorts) // WHEN - let result = experimentCohortsManager.cohort(for: subfeatureName) + let result = experimentCohortsManager.cohort(for: experiment, assignIfEnabled: true) // THEN - XCTAssertNil(result) + XCTAssertNotNil(result.cohortID) + XCTAssertTrue(result.didAttemptAssignment) + XCTAssertEqual(result.cohortID, experimentData1.cohort) } - func testEnrollmentDateReturnsNilIfDateDoesNotExist() { + func testCohortDoesNotAssignIfAssignIfEnabledIsFalse() { // GIVEN - let subfeatureName = "TestSubfeature" + mockStore.experiments = [:] + let cohorts = [cohort1, cohort2] + let experiment = ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: cohorts) // WHEN - let result = experimentCohortsManager.enrollmentDate(for: subfeatureName) + let result = experimentCohortsManager.cohort(for: experiment, assignIfEnabled: false) // THEN - XCTAssertNil(result) + XCTAssertNil(result.cohortID) + XCTAssertTrue(result.didAttemptAssignment) } - func testRemoveCohortSuccessfullyRemovesData() throws { + func testCohortDoesNotAssignIfAssignIfEnabledIsTrueButNoCohortsAvailable() { // GIVEN - mockStore.experiments = [subfeatureName1: experimentData1] + mockStore.experiments = [:] + let experiment = ExperimentSubfeature(parentID: "TestParent", subfeatureID: "NonExistentSubfeature", cohorts: []) // WHEN - experimentCohortsManager.removeCohort(from: subfeatureName1) + let result = experimentCohortsManager.cohort(for: experiment, assignIfEnabled: true) // THEN - let experiments = try XCTUnwrap(mockStore.experiments) - XCTAssertTrue(experiments.isEmpty) + XCTAssertNil(result.cohortID) + XCTAssertTrue(result.didAttemptAssignment) } - func testRemoveCohortDoesNothingIfSubfeatureDoesNotExist() { + func testCohortReassignsCohortIfAssignedCohortDoesNotExistAndAssignIfEnabledIsTrue() { // GIVEN - let expectedExperiments: Experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] - mockStore.experiments = expectedExperiments + mockStore.experiments = [subfeatureName1: experimentData1] // WHEN - experimentCohortsManager.removeCohort(from: "someOtherSubfeature") + let result1 = experimentCohortsManager.cohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), assignIfEnabled: true).cohortID // THEN - XCTAssertEqual( mockStore.experiments, expectedExperiments) + XCTAssertEqual(result1, experimentData3.cohort) } - func testAssignCohortReturnsNilIfNoCohorts() { + func testCohortDoesNotReassignsCohortIfAssignedCohortDoesNotExistAndAssignIfEnabledIsTrue() { // GIVEN - let subfeature = ExperimentSubfeature(parentID: "parent", subfeatureID: subfeatureName1, cohorts: []) + mockStore.experiments = [subfeatureName1: experimentData1] // WHEN - let result = experimentCohortsManager.assignCohort(to: subfeature) + let result1 = experimentCohortsManager.cohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), assignIfEnabled: false).cohortID // THEN - XCTAssertNil(result) + XCTAssertNil(result1) } - func testAssignCohortReturnsNilIfAllWeightsAreZero() { + func testCohortAssignsBasedOnWeight() { // GIVEN - let jsonCohort1: [String: Any] = ["name": "TestCohort", "weight": 0] - let jsonCohort2: [String: Any] = ["name": "TestCohort", "weight": 0] - let cohorts = [ - PrivacyConfigurationData.Cohort(json: jsonCohort1)!, - PrivacyConfigurationData.Cohort(json: jsonCohort2)! - ] - let subfeature = ExperimentSubfeature(parentID: "parent", subfeatureID: subfeatureName1, cohorts: cohorts) - - // WHEN - let result = experimentCohortsManager.assignCohort(to: subfeature) - - // THEN - XCTAssertNil(result) - } - - func testAssignCohortSelectsCorrectCohortBasedOnWeight() { - // Cohort1 has weight 1, Cohort2 has weight 3 - // Total weight is 1 + 3 = 4 - let jsonCohort1: [String: Any] = ["name": "Cohort1", "weight": 1] - let jsonCohort2: [String: Any] = ["name": "Cohort2", "weight": 3] - let cohorts = [ - PrivacyConfigurationData.Cohort(json: jsonCohort1)!, - PrivacyConfigurationData.Cohort(json: jsonCohort2)! - ] - let subfeature = ExperimentSubfeature(parentID: "parent", subfeatureID: subfeatureName1, cohorts: cohorts) - let expectedTotalWeight = 4.0 - - // Use a custom randomizer to verify the range - experimentCohortsManager = ExperimentCohortsManager( - store: mockStore, - randomizer: { range in - // Assert that the range lower bound is 0 - XCTAssertEqual(range.lowerBound, 0.0) - // Assert that the range upper bound is the total weight - XCTAssertEqual(range.upperBound, expectedTotalWeight) - return 0.0 - } - ) - - // Test case where random value is at the very start of Cohort1's range (0) - experimentCohortsManager = ExperimentCohortsManager( - store: mockStore, - randomizer: { _ in 0.0 } - ) - let resultStartOfCohort1 = experimentCohortsManager.assignCohort(to: subfeature) - XCTAssertEqual(resultStartOfCohort1, "Cohort1") - - // Test case where random value is at the end of Cohort1's range (0.9) - experimentCohortsManager = ExperimentCohortsManager( - store: mockStore, - randomizer: { _ in 0.9 } - ) - let resultEndOfCohort1 = experimentCohortsManager.assignCohort(to: subfeature) - XCTAssertEqual(resultEndOfCohort1, "Cohort1") - - // Test case where random value is at the start of Cohort2's range (1.00 to 4) - experimentCohortsManager = ExperimentCohortsManager( - store: mockStore, - randomizer: { _ in 1.00 } - ) - let resultStartOfCohort2 = experimentCohortsManager.assignCohort(to: subfeature) - XCTAssertEqual(resultStartOfCohort2, "Cohort2") + let experiment = ExperimentSubfeature(parentID: experimentData3.parentID, subfeatureID: subfeatureName3, cohorts: [cohort3, cohort4]) - // Test case where random value falls exactly within Cohort2's range (2.5) - experimentCohortsManager = ExperimentCohortsManager( - store: mockStore, - randomizer: { _ in 2.5 } - ) - let resultMiddleOfCohort2 = experimentCohortsManager.assignCohort(to: subfeature) - XCTAssertEqual(resultMiddleOfCohort2, "Cohort2") + let randomizer: (Range) -> Double = { range in + return 1.5 + } - // Test case where random value is at the end of Cohort2's range (4) experimentCohortsManager = ExperimentCohortsManager( store: mockStore, - randomizer: { _ in 3.9 } - ) - let resultEndOfCohort2 = experimentCohortsManager.assignCohort(to: subfeature) - XCTAssertEqual(resultEndOfCohort2, "Cohort2") - } - - func testAssignCohortWithSingleCohortAlwaysSelectsThatCohort() throws { - // GIVEN - let jsonCohort1: [String: Any] = ["name": "Cohort1", "weight": 1] - let cohorts = [ - PrivacyConfigurationData.Cohort(json: jsonCohort1)! - ] - let subfeature = ExperimentSubfeature(parentID: "parent", subfeatureID: subfeatureName1, cohorts: cohorts) - let expectedTotalWeight = 1.0 - - // Use a custom randomizer to verify the range - experimentCohortsManager = ExperimentCohortsManager( - store: mockStore, - randomizer: { range in - // Assert that the range lower bound is 0 - XCTAssertEqual(range.lowerBound, 0.0) - // Assert that the range upper bound is the total weight - XCTAssertEqual(range.upperBound, expectedTotalWeight) - return 0.0 - } + randomizer: randomizer ) // WHEN - experimentCohortsManager = ExperimentCohortsManager( - store: mockStore, - randomizer: { range in Double.random(in: range)} - ) - let result = experimentCohortsManager.assignCohort(to: subfeature) + let result = experimentCohortsManager.cohort(for: experiment, assignIfEnabled: true) // THEN - XCTAssertEqual(result, "Cohort1") - XCTAssertEqual(cohorts[0].name, mockStore.experiments?[subfeature.subfeatureID]?.cohort) + XCTAssertEqual(result.cohortID, experimentData3.cohort) + XCTAssertTrue(result.didAttemptAssignment) } - } class MockExperimentDataStore: ExperimentsDataStoring { diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift index e5deb4f60..8816daec2 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift @@ -72,7 +72,7 @@ final class ExperimentsDataStoreTests: XCTestCase { func testExperimentsSetEncodesAndStoresData() throws { // GIVEN let experimentData1 = ExperimentData(parentID: "parent", cohort: "TestCohort1", enrollmentDate: Date()) - let experimentData2 = ExperimentData(parentID: "parent", cohort: "TestCohort2", enrollmentDate: Date()) + let experimentData2 = ExperimentData(parentID: "parent2", cohort: "TestCohort2", enrollmentDate: Date()) let experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] // WHEN @@ -86,8 +86,10 @@ final class ExperimentsDataStoreTests: XCTestCase { let timeDifference1 = abs(experimentData1.enrollmentDate.timeIntervalSince(decodedExperiments?[subfeatureName1]?.enrollmentDate ?? Date())) let timeDifference2 = abs(experimentData2.enrollmentDate.timeIntervalSince(decodedExperiments?[subfeatureName2]?.enrollmentDate ?? Date())) XCTAssertEqual(decodedExperiments?[subfeatureName1]?.cohort, experimentData1.cohort) + XCTAssertEqual(decodedExperiments?[subfeatureName1]?.parentID, experimentData1.parentID) XCTAssertLessThanOrEqual(timeDifference1, 1.0) XCTAssertEqual(decodedExperiments?[subfeatureName2]?.cohort, experimentData2.cohort) + XCTAssertEqual(decodedExperiments?[subfeatureName2]?.parentID, experimentData2.parentID) XCTAssertLessThanOrEqual(timeDifference2, 1.0) } } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift index 720393246..08ecbec85 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationDataTests.swift @@ -70,6 +70,7 @@ class PrivacyConfigurationDataTests: XCTestCase { XCTAssertEqual(subfeatures["enabledSubfeature"]?.cohorts?[0].weight, 1) XCTAssertEqual(subfeatures["enabledSubfeature"]?.targets?[0].localeCountry, "US") XCTAssertEqual(subfeatures["enabledSubfeature"]?.targets?[0].localeLanguage, "fr") + XCTAssertEqual(subfeatures["enabledSubfeature"]?.settings, "{\"foo\":\"foo\\/value\",\"bar\":\"bar\\/value\"}") XCTAssertEqual(subfeatures["internalSubfeature"]?.state, "internal") } else { XCTFail("Could not parse subfeatures") diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift index 2eed811a6..81c01d9bb 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift @@ -42,7 +42,8 @@ final class PrivacyConfigurationReferenceTests: XCTestCase { let privacyConfiguration = AppPrivacyConfiguration(data: privacyConfigurationData, identifier: UUID().uuidString, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + experimentManager: MockExperimentCohortsManager()) for test in testConfig.tests { if test.exceptPlatforms.contains(.macosBrowser) || test.exceptPlatforms.contains(.iosBrowser) { os_log("Skipping test %@", test.name) diff --git a/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json b/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json index 31046de83..735c3914f 100644 --- a/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json +++ b/Tests/BrowserServicesKitTests/Resources/privacy-config-example.json @@ -171,19 +171,16 @@ }, "enabledSubfeature": { "state": "enabled", -<<<<<<< HEAD "targets": [ { "localeCountry": "US", "localeLanguage": "fr" } ], - "config": { + "settings": { "foo": "foo/value", "bar": "bar/value" }, -======= ->>>>>>> main "description": "A description of the sub-feature", "cohorts": [ { From 86b7c97c55a963dfa9b3a13db6d8017a5e291756 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 15 Nov 2024 12:46:06 +0100 Subject: [PATCH 16/49] getAllActiveExperiments tests --- .../AppPrivacyConfiguration.swift | 147 +++++---- ...dPrivacyConfigurationExperimentTests.swift | 284 ++++++++++++++++++ 2 files changed, 371 insertions(+), 60 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 30f33784a..4fc56d76c 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -204,85 +204,112 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { public func getAllActiveExperiments(versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Experiments { Self.experimentManagerQueue.sync { - guard let assignedExperiments = experimentManager.experiments else { return [:] } - var experiments: Experiments = [:] - for (key, value) in assignedExperiments { - if stateFor(subfeatureID: key, experimentData: value, versionProvider: versionProvider, randomizer: randomizer) == .enabled { - experiments[key] = value - } - } - return experiments - } + guard let assignedExperiments = experimentManager.experiments else { return [:] } + return assignedExperiments.filter { key, value in + stateFor(subfeatureID: key, experimentData: value, versionProvider: versionProvider, randomizer: randomizer) == .enabled + } + } } private func stateFor(subfeatureID: SubfeatureID, experimentData: ExperimentData, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - guard let parentFeature = PrivacyFeature(rawValue: experimentData.parentID) else { return .disabled(.featureMissing) } - let subfeatures = subfeatures(for: parentFeature) - guard let subfeatureData = subfeatures[subfeatureID] else { return .disabled(.featureMissing) } - return stateFor(parentFeature: parentFeature, subfeatureData: subfeatureData, subfeatureID: subfeatureID, cohortID: experimentData.cohort, versionProvider: versionProvider, randomizer: randomizer) + guard let parentFeature = PrivacyFeature(rawValue: experimentData.parentID), + let subfeatureData = subfeatures(for: parentFeature)[subfeatureID] else { + return .disabled(.featureMissing) + } + return stateFor(parentFeature: parentFeature, + subfeatureData: subfeatureData, + subfeatureID: subfeatureID, + cohortID: experimentData.cohort, + assignCohortEnabled: false, + versionProvider: versionProvider, + randomizer: randomizer) } public func stateFor(_ subfeature: any PrivacySubfeature, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - let subfeatures = subfeatures(for: subfeature.parent) - guard let subfeatureData = subfeatures[subfeature.rawValue] else { return .disabled(.featureMissing) } + Self.experimentManagerQueue.sync { + guard let subfeatureData = subfeatures(for: subfeature.parent)[subfeature.rawValue] else { + return .disabled(.featureMissing) + } - return stateFor(parentFeature: subfeature.parent, subfeatureData: subfeatureData, subfeatureID: subfeature.rawValue, cohortID: cohortID, versionProvider: versionProvider, randomizer: randomizer) + return stateFor(parentFeature: subfeature.parent, + subfeatureData: subfeatureData, + subfeatureID: subfeature.rawValue, + cohortID: cohortID, + versionProvider: versionProvider, + randomizer: randomizer) + } } private func stateFor(parentFeature: PrivacyFeature, subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature, subfeatureID: SubfeatureID, cohortID: CohortID?, + assignCohortEnabled: Bool = true, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - Self.experimentManagerQueue.sync { - // Step 1: Check parent feature state - let parentState = stateFor(featureKey: parentFeature, versionProvider: versionProvider) - guard case .enabled = parentState else { return parentState } - - // Step 2: Check version - let satisfiesMinVersion = satisfiesMinVersion(subfeatureData.minSupportedVersion, versionProvider: versionProvider) - - // Step 3: Check sub-feature state - switch subfeatureData.state { - case PrivacyConfigurationData.State.enabled: - guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } - case PrivacyConfigurationData.State.internal: - guard internalUserDecider.isInternalUser else { return .disabled(.limitedToInternalUsers) } - guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } - default: return .disabled(.disabledInConfig) - } - - // Step 4: Handle Rollouts - if let rollout = subfeatureData.rollout, - !isRolloutEnabled(subfeatureID: subfeatureID, parentID: parentFeature.rawValue, rolloutSteps: rollout.steps, randomizer: randomizer) { - return .disabled(.stillInRollout) - } - - // Step 5: Check if a cohort was passed in the func - // If no corhort passed check for Target and Rollout - guard let passedCohort = cohortID else { - return checkTargets(subfeatureData) - } - - // Step 6: Verify there the cohort - // Check if cohort assigned and matches passed cohort - // If cohort not assigned - // Tries to assign if matching target - // Check if cohort assigned and matches passed cohort - let cohorts = subfeatureData.cohorts ?? [] - let targetsState = checkTargets(subfeatureData) - let assignedCohortResponse = experimentManager.cohort(for: ExperimentSubfeature(parentID: parentFeature.rawValue, subfeatureID: subfeatureID, cohorts: cohorts), assignIfEnabled: targetsState == .enabled) - let possibleDisabledReason: PrivacyConfigurationFeatureDisabledReason = assignedCohortResponse.didAttemptAssignment && targetsState != .enabled ? .targetDoesNotMatch : .experimentCohortDoesNotMatch - if let assignedCohort = assignedCohortResponse.cohortID { - return (assignedCohort == passedCohort) ? .enabled : .disabled(possibleDisabledReason) - } else { - return .disabled(possibleDisabledReason) - } + // Step 1: Check parent feature state + let parentState = stateFor(featureKey: parentFeature, versionProvider: versionProvider) + guard case .enabled = parentState else { return parentState } + + // Step 2: Check version + let satisfiesMinVersion = satisfiesMinVersion(subfeatureData.minSupportedVersion, versionProvider: versionProvider) + + // Step 3: Check sub-feature state + switch subfeatureData.state { + case PrivacyConfigurationData.State.enabled: + guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } + case PrivacyConfigurationData.State.internal: + guard internalUserDecider.isInternalUser else { return .disabled(.limitedToInternalUsers) } + guard satisfiesMinVersion else { return .disabled(.appVersionNotSupported) } + default: return .disabled(.disabledInConfig) + } + + // Step 4: Handle Rollouts + if let rollout = subfeatureData.rollout, + !isRolloutEnabled(subfeatureID: subfeatureID, parentID: parentFeature.rawValue, rolloutSteps: rollout.steps, randomizer: randomizer) { + return .disabled(.stillInRollout) + } + + // Step 5: Check if a cohort was passed in the func + // If no corhort passed check for Target and Rollout + guard let passedCohort = cohortID else { + return checkTargets(subfeatureData) + } + + // Step 5: Cohort handling + // Check if cohort assigned and matches passed cohort + // If cohort not assigned + // Tries to assign if matching target + // Check if cohort assigned and matches passed cohort + return checkCohortState(subfeatureData, + passedCohort: passedCohort, + assignCohortEnabled: assignCohortEnabled, + subfeatureID: subfeatureID, + parentFeature: parentFeature) + } + + // Check if cohort assigned and matches passed cohort + // If cohort not assigned + // Tries to assign if matching target + // Check if cohort assigned and matches passed cohort + private func checkCohortState(_ subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature, + passedCohort: CohortID?, + assignCohortEnabled: Bool, + subfeatureID: SubfeatureID, + parentFeature: PrivacyFeature) -> PrivacyConfigurationFeatureState { + let cohorts = subfeatureData.cohorts ?? [] + let targetsState = checkTargets(subfeatureData) + let assignIfEnabled = assignCohortEnabled && targetsState == .enabled + let assignedCohortResponse = experimentManager.cohort(for: ExperimentSubfeature(parentID: parentFeature.rawValue, subfeatureID: subfeatureID, cohorts: cohorts), assignIfEnabled: assignIfEnabled) + let possibleDisabledReason: PrivacyConfigurationFeatureDisabledReason = assignedCohortResponse.didAttemptAssignment && targetsState != .enabled ? .targetDoesNotMatch : .experimentCohortDoesNotMatch + if let assignedCohort = assignedCohortResponse.cohortID { + return (assignedCohort == passedCohort) ? .enabled : .disabled(possibleDisabledReason) + } else { + return .disabled(possibleDisabledReason) } } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift index 3ecc0eb9e..c2a8d63ed 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift @@ -943,4 +943,288 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).enabled") UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).lastRolloutCount") } + + func testAllActiveExperimentsEmptyIfNoAssignedExperiment() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "foo", data: featureJson) + let config = manager.privacyConfig + + let activeExperiments = config.getAllActiveExperiments() + XCTAssertTrue(activeExperiments.isEmpty) + XCTAssertNil(mockStore.experiments) + } + + func testAllActiveExperimentsReturnsOnlyActiveExperiments() { + var featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + }, + "inlineIconCredentials": { + "state": "enabled", + "minSupportedVersion": 1, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 0 + }, + { + "name": "green", + "weight": 1 + } + ] + }, + "accessCredentialManagement": { + "state": "enabled", + "minSupportedVersion": 3, + "targets": [ + { + "localeCountry": "CA" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "green", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + + manager.reload(etag: "foo", data: featureJson) + var config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.inlineIconCredentials, cohortID: "green")) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.accessCredentialManagement, cohortID: "control")) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + XCTAssertEqual(experimentManager.cohort(for: AutofillSubfeature.credentialsSaving.rawValue), "control") + XCTAssertEqual(experimentManager.cohort(for: AutofillSubfeature.inlineIconCredentials.rawValue), "green") + XCTAssertNil(experimentManager.cohort(for: AutofillSubfeature.accessCredentialManagement.rawValue)) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) + + var activeExperiments = config.getAllActiveExperiments() + XCTAssertEqual(activeExperiments.count, 2) + XCTAssertEqual(activeExperiments[AutofillSubfeature.credentialsSaving.rawValue]?.cohort, "control") + XCTAssertEqual(activeExperiments[AutofillSubfeature.inlineIconCredentials.rawValue]?.cohort, "green") + XCTAssertNil(activeExperiments[AutofillSubfeature.accessCredentialManagement.rawValue]) + + // When an assigned cohort is removed it's not part of active experiments + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "blue", + "weight": 1 + } + ] + }, + "inlineIconCredentials": { + "state": "enabled", + "minSupportedVersion": 1, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 0 + }, + { + "name": "green", + "weight": 1 + } + ] + }, + "accessCredentialManagement": { + "state": "enabled", + "minSupportedVersion": 3, + "targets": [ + { + "localeCountry": "CA" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "green", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + + manager.reload(etag: "foo", data: featureJson) + config = manager.privacyConfig + + activeExperiments = config.getAllActiveExperiments() + XCTAssertEqual(activeExperiments.count, 1) + XCTAssertNil(activeExperiments[AutofillSubfeature.credentialsSaving.rawValue]) + XCTAssertEqual(activeExperiments[AutofillSubfeature.inlineIconCredentials.rawValue]?.cohort, "green") + XCTAssertNil(activeExperiments[AutofillSubfeature.accessCredentialManagement.rawValue]) + + // When feature disabled an assigned cohort it's not part of active experiments + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "blue", + "weight": 1 + } + ] + }, + "inlineIconCredentials": { + "state": "disabled", + "minSupportedVersion": 1, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 0 + }, + { + "name": "green", + "weight": 1 + } + ] + }, + "accessCredentialManagement": { + "state": "enabled", + "minSupportedVersion": 3, + "targets": [ + { + "localeCountry": "CA" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "green", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + + manager.reload(etag: "foo", data: featureJson) + config = manager.privacyConfig + + activeExperiments = config.getAllActiveExperiments() + XCTAssertTrue(activeExperiments.isEmpty) + } + } From 2595fcf64be99675ecbe7dc5bf51d7c15005d343 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 15 Nov 2024 12:52:53 +0100 Subject: [PATCH 17/49] sort linting issues --- .../PrivacyConfig/AppPrivacyConfiguration.swift | 1 - .../PrivacyConfig/PrivacyConfigurationManager.swift | 2 +- .../ContentBlocker/WebViewTestHelper.swift | 5 ++--- Tests/BrowserServicesKitTests/GPC/GPCTests.swift | 2 +- .../AddPrivacyConfigurationExperimentTests.swift | 6 +----- .../PrivacyConfig/PrivacyConfigurationReferenceTests.swift | 2 +- 6 files changed, 6 insertions(+), 12 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 4fc56d76c..960256120 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -191,7 +191,6 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool { - switch stateFor(subfeature, cohortID: cohortID, versionProvider: versionProvider, randomizer: randomizer) { case .enabled: return true diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index 6fa567e3a..fff2f1482 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -142,7 +142,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { identifier: embeddedConfigData.etag, localProtection: localProtection, internalUserDecider: internalUserDecider, - locale: locale, + locale: locale, experimentManager: experimentCohortManager, installDate: installDate) } diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift index bced88b71..f803d11e9 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift @@ -197,7 +197,8 @@ final class WebKitTestHelper { return AppPrivacyConfiguration(data: privacyData, identifier: "", localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), experimentManager: MockExperimentCohortsManager()) + internalUserDecider: DefaultInternalUserDecider(), + experimentManager: MockExperimentCohortsManager()) } static func prepareContentBlockingRules(trackerData: TrackerData, @@ -232,6 +233,4 @@ class MockExperimentCohortsManager: ExperimentCohortsManaging { func cohort(for experiment: BrowserServicesKit.ExperimentSubfeature, assignIfEnabled: Bool) -> (cohortID: BrowserServicesKit.CohortID?, didAttemptAssignment: Bool) { return (nil, true) } - - } diff --git a/Tests/BrowserServicesKitTests/GPC/GPCTests.swift b/Tests/BrowserServicesKitTests/GPC/GPCTests.swift index 8ee3ec2e2..9ffca4ad1 100644 --- a/Tests/BrowserServicesKitTests/GPC/GPCTests.swift +++ b/Tests/BrowserServicesKitTests/GPC/GPCTests.swift @@ -42,7 +42,7 @@ final class GPCTests: XCTestCase { appConfig = AppPrivacyConfiguration(data: privacyData, identifier: "", localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), + internalUserDecider: DefaultInternalUserDecider(), experimentManager: MockExperimentCohortsManager()) } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift index c2a8d63ed..8a4591fe7 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift @@ -1,6 +1,5 @@ // // AddPrivacyConfigurationExperimentTests.swift -// DuckDuckGo // // Copyright © 2024 DuckDuckGo. All rights reserved. // @@ -31,7 +30,6 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { let subfeatureName = "credentialsSaving" - override func setUp() { locale = Locale(identifier: "fr_US") mockEmbeddedData = MockEmbeddedDataProvider(data: featureJson, etag: "test") @@ -43,7 +41,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale, + locale: locale, experimentCohortManager: experimentManager) } @@ -55,7 +53,6 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { manager = nil } - func testCohortOnlyAssignedWhenCallingStateForSubfeature() { featureJson = """ @@ -938,7 +935,6 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "blue") } - func clearRolloutData(feature: String, subFeature: String) { UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).enabled") UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).lastRolloutCount") diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift index 81c01d9bb..b4d0d1e45 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift @@ -42,7 +42,7 @@ final class PrivacyConfigurationReferenceTests: XCTestCase { let privacyConfiguration = AppPrivacyConfiguration(data: privacyConfigurationData, identifier: UUID().uuidString, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), + internalUserDecider: DefaultInternalUserDecider(), experimentManager: MockExperimentCohortsManager()) for test in testConfig.tests { if test.exceptPlatforms.contains(.macosBrowser) || test.exceptPlatforms.contains(.iosBrowser) { From 86d6f7bd6dc15a831346c071b0fcee7c7220de06 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 15 Nov 2024 12:59:52 +0100 Subject: [PATCH 18/49] fix linting --- .../PrivacyConfig/AppPrivacyConfiguration.swift | 9 ++++----- .../ContentBlocker/WebViewTestHelper.swift | 2 +- 2 files changed, 5 insertions(+), 6 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 960256120..4e3a09c63 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -199,7 +199,6 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } } - public func getAllActiveExperiments(versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Experiments { Self.experimentManagerQueue.sync { @@ -234,7 +233,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { return .disabled(.featureMissing) } - return stateFor(parentFeature: subfeature.parent, + return stateFor(parentFeature: subfeature.parent, subfeatureData: subfeatureData, subfeatureID: subfeature.rawValue, cohortID: cohortID, @@ -296,9 +295,9 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { // Tries to assign if matching target // Check if cohort assigned and matches passed cohort private func checkCohortState(_ subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature, - passedCohort: CohortID?, - assignCohortEnabled: Bool, - subfeatureID: SubfeatureID, + passedCohort: CohortID?, + assignCohortEnabled: Bool, + subfeatureID: SubfeatureID, parentFeature: PrivacyFeature) -> PrivacyConfigurationFeatureState { let cohorts = subfeatureData.cohorts ?? [] let targetsState = checkTargets(subfeatureData) diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift index f803d11e9..bf7451670 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift @@ -197,7 +197,7 @@ final class WebKitTestHelper { return AppPrivacyConfiguration(data: privacyData, identifier: "", localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), + internalUserDecider: DefaultInternalUserDecider(), experimentManager: MockExperimentCohortsManager()) } From 95c44700e0d558baad080e855ee204c2a97645fc Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 15 Nov 2024 13:07:18 +0100 Subject: [PATCH 19/49] don't break the API --- .../PrivacyConfig/AppPrivacyConfiguration.swift | 2 +- .../PrivacyConfig/ExperimentCohortsManager.swift | 2 +- .../ContentBlocker/WebViewTestHelper.swift | 3 +-- Tests/BrowserServicesKitTests/GPC/GPCTests.swift | 3 +-- .../PrivacyConfig/PrivacyConfigurationReferenceTests.swift | 3 +-- 5 files changed, 5 insertions(+), 8 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 4e3a09c63..46e827a22 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -44,7 +44,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { internalUserDecider: InternalUserDecider, userDefaults: UserDefaults = UserDefaults(), locale: Locale = Locale.current, - experimentManager: ExperimentCohortsManaging, + experimentManager: ExperimentCohortsManaging = ExperimentCohortsManager(), installDate: Date? = nil) { self.data = data self.identifier = identifier diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift index 2173160f8..e52e7b569 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift @@ -71,7 +71,7 @@ public class ExperimentCohortsManager: ExperimentCohortsManaging { } } - public init(store: ExperimentsDataStoring, randomizer: @escaping (Range) -> Double = Double.random(in:)) { + public init(store: ExperimentsDataStoring = ExperimentsDataStore(), randomizer: @escaping (Range) -> Double = Double.random(in:)) { self.store = store self.randomizer = randomizer } diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift index bf7451670..db78351d9 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift @@ -197,8 +197,7 @@ final class WebKitTestHelper { return AppPrivacyConfiguration(data: privacyData, identifier: "", localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), - experimentManager: MockExperimentCohortsManager()) + internalUserDecider: DefaultInternalUserDecider()) } static func prepareContentBlockingRules(trackerData: TrackerData, diff --git a/Tests/BrowserServicesKitTests/GPC/GPCTests.swift b/Tests/BrowserServicesKitTests/GPC/GPCTests.swift index 9ffca4ad1..1eee4af76 100644 --- a/Tests/BrowserServicesKitTests/GPC/GPCTests.swift +++ b/Tests/BrowserServicesKitTests/GPC/GPCTests.swift @@ -42,8 +42,7 @@ final class GPCTests: XCTestCase { appConfig = AppPrivacyConfiguration(data: privacyData, identifier: "", localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), - experimentManager: MockExperimentCohortsManager()) + internalUserDecider: DefaultInternalUserDecider()) } func testWhenGPCEnableDomainIsHttpThenISGPCEnabledTrue() { diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift index b4d0d1e45..2eed811a6 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/PrivacyConfigurationReferenceTests.swift @@ -42,8 +42,7 @@ final class PrivacyConfigurationReferenceTests: XCTestCase { let privacyConfiguration = AppPrivacyConfiguration(data: privacyConfigurationData, identifier: UUID().uuidString, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - experimentManager: MockExperimentCohortsManager()) + internalUserDecider: DefaultInternalUserDecider()) for test in testConfig.tests { if test.exceptPlatforms.contains(.macosBrowser) || test.exceptPlatforms.contains(.iosBrowser) { os_log("Skipping test %@", test.name) From 727e813d83a799614ad70f4210eaaeda582f3cf6 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Thu, 21 Nov 2024 14:23:58 +0100 Subject: [PATCH 20/49] implement PixelExperimentKit --- .../BrowserServicesKit-Package.xcscheme | 24 + Package.swift | 20 + .../AppPrivacyConfiguration.swift | 9 +- .../PrivacyConfig/PrivacyConfiguration.swift | 2 + .../PrivacyConfigurationManager.swift | 18 +- .../PixelExperimentKit.swift | 243 ++++++++ Sources/PixelKit/PixelKit.swift | 15 +- Sources/PixelKit/PixelKitEventV2.swift | 22 + .../XCTestCase+PixelKit.swift | 2 + .../PrivacyConfigurationManagerMock.swift | 1 + .../Autofill/AutofillTestHelper.swift | 3 +- .../UserContentControllerTests.swift | 1 + .../DefaultFeatureFlaggerTests.swift | 8 +- .../FingerprintingReferenceTests.swift | 3 +- .../GPC/GPCReferenceTests.swift | 3 +- .../LinkProtection/AmpMatchingTests.swift | 3 +- .../LinkProtection/URLParameterTests.swift | 3 +- .../AdClickAttributionFeatureTests.swift | 3 +- ...dPrivacyConfigurationExperimentTests.swift | 48 +- .../AppPrivacyConfigurationTests.swift | 93 ++- .../ReferrerTrimmingTests.swift | 3 +- .../Resources/privacy-reference-tests | 2 +- .../HTTPSUpgradeReferenceTests.swift | 3 +- ...SubscriptionFeatureAvailabilityTests.swift | 3 +- Tests/DDGSyncTests/Mocks/Mocks.swift | 1 + .../DuckPlayerContingencyHandlerTests.swift | 18 +- .../PixelExperimentKitTests.swift | 551 ++++++++++++++++++ 27 files changed, 1051 insertions(+), 54 deletions(-) create mode 100644 Sources/PixelExperimentKit/PixelExperimentKit.swift create mode 100644 Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift diff --git a/.swiftpm/xcode/xcshareddata/xcschemes/BrowserServicesKit-Package.xcscheme b/.swiftpm/xcode/xcshareddata/xcschemes/BrowserServicesKit-Package.xcscheme index 56a2ef845..f6fd11f65 100644 --- a/.swiftpm/xcode/xcshareddata/xcschemes/BrowserServicesKit-Package.xcscheme +++ b/.swiftpm/xcode/xcshareddata/xcschemes/BrowserServicesKit-Package.xcscheme @@ -539,6 +539,20 @@ ReferencedContainer = "container:"> + + + + + + + + Bool func stateFor(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> PrivacyConfigurationFeatureState diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index fff2f1482..0bb13355b 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -20,6 +20,10 @@ import Foundation import Combine import Common +public protocol PrivacyConfigurationDelegate: AnyObject { + func didAssignCohort(_ cohort: CohortID, to experiment: SubfeatureID) +} + public protocol EmbeddedDataProvider { var embeddedDataEtag: String { get } @@ -60,6 +64,8 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { public let internalUserDecider: InternalUserDecider + private let reportExperimentCohortAssignment: (_ cohortID: CohortID, _ SubfeatureID: SubfeatureID) -> Void + private let updatesSubject = PassthroughSubject() public var updatesPublisher: AnyPublisher { updatesSubject.eraseToAnyPublisher() @@ -114,7 +120,8 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { internalUserDecider: InternalUserDecider, locale: Locale = Locale.current, experimentCohortManager: ExperimentCohortsManaging = ExperimentCohortsManager(store: ExperimentsDataStore()), - installDate: Date? = nil + installDate: Date? = nil, + reportExperimentCohortAssignment: @escaping (_ cohortID: CohortID, _ SubfeatureID: SubfeatureID) -> Void ) { self.embeddedDataProvider = embeddedDataProvider self.localProtection = localProtection @@ -123,6 +130,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { self.experimentCohortManager = experimentCohortManager self.locale = locale self.installDate = installDate + self.reportExperimentCohortAssignment = reportExperimentCohortAssignment reload(etag: fetchedETag, data: fetchedData) } @@ -135,6 +143,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { internalUserDecider: internalUserDecider, locale: locale, experimentManager: experimentCohortManager, + delegate: self, installDate: installDate) } @@ -144,6 +153,7 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { internalUserDecider: internalUserDecider, locale: locale, experimentManager: experimentCohortManager, + delegate: self, installDate: installDate) } @@ -181,3 +191,9 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { return result } } + +extension PrivacyConfigurationManager: PrivacyConfigurationDelegate { + public func didAssignCohort(_ cohort: CohortID, to experiment: SubfeatureID) { + reportExperimentCohortAssignment(cohort, experiment) + } +} diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift new file mode 100644 index 000000000..2dfc658c0 --- /dev/null +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -0,0 +1,243 @@ +// +// PixelExperimentKit.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import PixelKit +import BrowserServicesKit +import Foundation + +struct ExperimentEvent: PixelKitEvent { + var name: String + var parameters: [String : String]? +} + +public protocol ExperimentActionPixelStore { + func removeObject(forKey defaultName: String) + func integer(forKey defaultName: String) -> Int + func set(_ value: Int, forKey defaultName: String) +} + +extension PixelKit { + + struct Constants { + static let enrollmentEventPrefix = "experiment_enroll" + static let metricsEventPrefix = "experiment_metrics" + static let metricKey = "metric" + static let conversionWindowDaysKey = "conversionWindowDays" + static let valueKey = "value" + static let enrollmentDateKey = "enrollmentDate" + static let searchMetricValue = "search" + static let appUseMetricValue = "app_use" + } + + // Static property to hold shared dependencies + struct ExperimentConfig { + static var privacyConfigManager: PrivacyConfigurationManager? + static var store: ExperimentActionPixelStore = UserDefaults.standard + static var fireFunction: (PixelKitEvent, PixelKit.Frequency, Bool) -> Void = { event, frequency, includeAppVersion in + fire(event, frequency: frequency, includeAppVersionParameter: includeAppVersion) + } + } + + // Setup method to initialize dependencies + public static func configureExperimentKit( + privacyConfigManager: PrivacyConfigurationManager, + store: ExperimentActionPixelStore = UserDefaults.standard, + fire: @escaping (PixelKitEvent, PixelKit.Frequency, Bool) -> Void = { event, frequency, includeAppVersion in + fire(event, frequency: frequency, includeAppVersionParameter: includeAppVersion) + } + ) { + ExperimentConfig.privacyConfigManager = privacyConfigManager + ExperimentConfig.store = store + ExperimentConfig.fireFunction = fire + } + + /// Fires a pixel indicating the user's enrollment in an experiment. + /// - Parameters: + /// - subfeatureID: Identifier for the subfeature associated with the experiment. + /// - experiment: Data about the experiment like cohort and enrollment date + public static func fireExperimentEnrollmentPixel(subfeatureID: SubfeatureID, experiment: ExperimentData) { + let eventName = "\(Self.Constants.enrollmentEventPrefix)_\(subfeatureID)_\(experiment.cohort)" + let event = ExperimentEvent(name: eventName, parameters: [Self.Constants.enrollmentDateKey: experiment.enrollmentDate.toYYYYMMDDInET()]) + ExperimentConfig.fireFunction(event, .uniqueIncludingParameters, false) + } + + /// Fires a pixel for a specific action in an experiment, based on conversion window and value thresholds (if value is a number). + /// - Parameters: + /// - subfeatureID: Identifier for the subfeature associated with the experiment. + /// - metric: The name of the metric being tracked (e.g., "searches"). + /// - conversionWindowDays: The range of days after enrollment during which the action is valid. + /// - value: A specific value associated to the action. It could be the target number of actions required to fire the pixel. + /// + /// This function: + /// 1. Validates if the experiment is active. + /// 2. Ensures the user is within the specified conversion window. + /// 3. Tracks actions performed and sends the pixel once the target value is reached (if applicable). + public static func fireExperimentPixel(for subfeatureID: SubfeatureID, metric: String, conversionWindowDays: ClosedRange, value: String) { + // Check is active experiment for user + guard let privacyConfigManager = ExperimentConfig.privacyConfigManager else { + assertionFailure("PrivacyConfigurationManager is not configured") + return + } + guard let experimentData = privacyConfigManager.privacyConfig.getAllActiveExperiments()[subfeatureID] else { return } + + Self.fireExperimentPixelForActiveExperiment(subfeatureID, experimentData: experimentData, metric: metric, conversionWindowDays: conversionWindowDays, value: value) + } + + /// Fires search-related experiment pixels for all active experiments. + /// + /// This function iterates through all active experiments and triggers + /// pixel firing based on predefined search-related value and conversion window mappings. + /// - The value and conversion windows define when and how many search actions + /// must occur before the pixel is fired. + public static func fireSearchExperimentPixels() { + let valueConversionDictionary: [Int: [ClosedRange]] = [ + 1: [0...0, 1...1, 2...2, 3...3, 4...4, 5...5, 6...6, 7...7, 5...7], + 4: [5...7, 8...15], + 6: [5...7, 8...15], + 11: [5...7, 8...15], + 21: [5...7, 8...15], + 30: [5...7, 8...15] + ] + guard let privacyConfigManager = ExperimentConfig.privacyConfigManager else { + assertionFailure("PrivacyConfigurationManager is not configured") + return + } + privacyConfigManager.privacyConfig.getAllActiveExperiments().forEach { experiment in + fireExperimentPixelsfor( + experiment.key, + experimentData: experiment.value, + metric: Self.Constants.searchMetricValue, + valueConversionDictionary: valueConversionDictionary + ) + } + } + + /// Fires app retention-related experiment pixels for all active experiments. + /// + /// This function iterates through all active experiments and triggers + /// pixel firing based on predefined app retention value and conversion window mappings. + /// - The value and conversion windows define when and how many app usage actions + /// must occur before the pixel is fired. + public static func fireAppRetentionExperimentPixels() { + let valueConversionDictionary: [Int: [ClosedRange]] = [ + 1: [1...1, 2...2, 3...3, 4...4, 5...5, 6...6, 7...7, 5...7], + 4: [5...7, 8...15], + 6: [5...7, 8...15], + 11: [5...7, 8...15], + 21: [5...7, 8...15], + 30: [5...7, 8...15] + ] + guard let privacyConfigManager = ExperimentConfig.privacyConfigManager else { + assertionFailure("PrivacyConfigurationManager is not configured") + return + } + privacyConfigManager.privacyConfig.getAllActiveExperiments().forEach { experiment in + fireExperimentPixelsfor( + experiment.key, + experimentData: experiment.value, + metric: Self.Constants.appUseMetricValue, + valueConversionDictionary: valueConversionDictionary + ) + } + } + + private static func fireExperimentPixelsfor( + _ experiment: SubfeatureID, + experimentData: ExperimentData, + metric: String, + valueConversionDictionary: [Int: [ClosedRange]] + ) { + valueConversionDictionary.forEach { value, ranges in + ranges.forEach { range in + fireExperimentPixelForActiveExperiment( + experiment, + experimentData: experimentData, + metric: metric, + conversionWindowDays: range, + value: "\(value)" + ) + } + } + } + + private static func fireExperimentPixelForActiveExperiment(_ subfeatureID: SubfeatureID, experimentData: ExperimentData,metric: String, conversionWindowDays: ClosedRange, value: String) { + // Set parameters, event name, store key + let eventName = "\(Self.Constants.metricsEventPrefix)_\(subfeatureID)_\(experimentData.cohort)" + let parameters: [String: String] = [ + Self.Constants.metricKey: metric, + Self.Constants.conversionWindowDaysKey: "\(conversionWindowDays.lowerBound.description)-\(conversionWindowDays.upperBound.description)", + Self.Constants.valueKey: value, + Self.Constants.enrollmentDateKey: experimentData.enrollmentDate.toYYYYMMDDInET() + ] + let event = ExperimentEvent(name: eventName, parameters: parameters) + let eventStoreKey = eventName + "_" + parameters.escapedString() + + // Check if user is in conversion window + // if not don't send pixel and remove related action from the store + guard isUserInConversionWindow(conversionWindowDays, enrollmentDate: experimentData.enrollmentDate) else { + ExperimentConfig.store.removeObject(forKey: eventStoreKey) + return + } + + // Check if value is a number + // if so check if the action for the given experiment and parameter has been performed a number of time >= than the required + // if so send the pixel + // if not increase the count of the action + // if value is not a number send the pixel + if let numberOfAction = Int(value), numberOfAction > 1 { + let actualActionNumber = ExperimentConfig.store.integer(forKey: eventStoreKey) + if actualActionNumber >= numberOfAction { + ExperimentConfig.fireFunction(event, .uniqueIncludingParameters, false) + } else { + ExperimentConfig.store.set(actualActionNumber + 1, forKey: eventStoreKey) + } + } else { + ExperimentConfig.fireFunction(event, .uniqueIncludingParameters, false) + } + } + + private static func isUserInConversionWindow(_ conversionWindowDays: ClosedRange, enrollmentDate: Date) -> Bool { + guard let startOfWindowDate = enrollmentDate.addDays(conversionWindowDays.lowerBound) else { return false } + guard let endOfWindowDate = enrollmentDate.addDays(conversionWindowDays.upperBound) else { return false } + return Date() >= startOfWindowDate && Date() <= endOfWindowDate + } +} + +extension Dictionary where Key: Comparable { + func escapedString(pairSeparator: String = ":", entrySeparator: String = ",") -> String { + return self.sorted { $0.key < $1.key } + .map { "\("\($0.key)".replacingOccurrences(of: entrySeparator, with: "\\\(entrySeparator)"))\(pairSeparator)\("\($0.value)".replacingOccurrences(of: entrySeparator, with: "\\\(entrySeparator)"))" } + .joined(separator: entrySeparator) + } +} + +extension UserDefaults: ExperimentActionPixelStore {} + +extension Date { + public func toYYYYMMDDInET() -> String { + let formatter = DateFormatter() + formatter.dateFormat = "yyyy-MM-dd" + formatter.timeZone = TimeZone(identifier: "America/New_York") + return formatter.string(from: self) + } + + func addDays(_ days: Int) -> Date? { + return Calendar.current.date(byAdding: .day, value: days, to: self) + } +} diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index 4f01a5647..a3b027e5e 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -31,10 +31,13 @@ public final class PixelKit { /// [Legacy] Used in Pixel.fire(...) as .unique but without the `_u` requirement in the name case legacyInitial - /// Sent only once ever. The timestamp for this pixel is stored. + /// Sent only once ever (based on pixel name only.) The timestamp for this pixel is stored. /// Note: This is the only pixel that MUST end with `_u`, Name for pixels of this type must end with if it doesn't an assertion is fired. case unique + /// Sent only once ever (based on pixel name AND parameters). The timestamp for this pixel is stored. + case uniqueIncludingParameters + /// [Legacy] Used in Pixel.fire(...) as .daily but without the `_d` automatically added to the name case legacyDaily @@ -67,6 +70,8 @@ public final class PixelKit { "Legacy Daily and Count" case .dailyAndCount: "Daily and Count" + case .uniqueIncludingParameters: + "Unique Including Parameters" } } } @@ -222,6 +227,14 @@ public final class PixelKit { } else { printDebugInfo(pixelName: pixelName, frequency: frequency, parameters: newParams, skipped: true) } + case .uniqueIncludingParameters: + let pixelNameAndParams = pixelName + newParams.description + if !pixelHasBeenFiredEver(pixelNameAndParams) { + fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) + updatePixelLastFireDate(pixelName: pixelNameAndParams) + } else { + printDebugInfo(pixelName: pixelName, frequency: frequency, parameters: newParams, skipped: true) + } case .legacyDaily: reportErrorIf(pixel: pixelName, endsWith: "_u") reportErrorIf(pixel: pixelName, endsWith: "_d") diff --git a/Sources/PixelKit/PixelKitEventV2.swift b/Sources/PixelKit/PixelKitEventV2.swift index dc641454c..b0070b972 100644 --- a/Sources/PixelKit/PixelKitEventV2.swift +++ b/Sources/PixelKit/PixelKitEventV2.swift @@ -43,6 +43,28 @@ public protocol PixelFiring { func fire(_ event: PixelKitEventV2, frequency: PixelKit.Frequency) + + static func fire(_ event: PixelKitEvent, + frequency: PixelKit.Frequency, + withHeaders headers: [String: String], + withAdditionalParameters parameters: [String: String]?, + withError error: Error?, + allowedQueryReservedCharacters: CharacterSet?, + includeAppVersionParameter: Bool, + onComplete: (Bool, Error?) -> Void) +} + +extension PixelFiring { + public static func fire(_ event: PixelKitEvent, + frequency: PixelKit.Frequency = .standard, + withHeaders headers: [String: String] = [:], + withAdditionalParameters parameters: [String: String]? = nil, + withError error: Error? = nil, + allowedQueryReservedCharacters: CharacterSet? = nil, + includeAppVersionParameter: Bool = true, + onComplete: (Bool, Error?) -> Void = { _, _ in }) { + return fire(event, frequency: frequency, withHeaders: headers, withAdditionalParameters: parameters, withError: error, allowedQueryReservedCharacters: allowedQueryReservedCharacters, includeAppVersionParameter: includeAppVersionParameter, onComplete: onComplete) + } } extension PixelKit: PixelFiring { diff --git a/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift b/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift index 371bd3c42..bdb5bb2cc 100644 --- a/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift +++ b/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift @@ -161,6 +161,8 @@ public extension XCTestCase { case .dailyAndCount: expectedPixelNames.append(originalName.appending("_daily")) expectedPixelNames.append(originalName.appending("_count")) + case .uniqueIncludingParameters: + expectedPixelNames.append(originalName) } return expectedPixelNames } diff --git a/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift b/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift index 76ecc8b87..3bc2e3e66 100644 --- a/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift +++ b/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift @@ -21,6 +21,7 @@ import BrowserServicesKit import Combine class PrivacyConfigurationMock: PrivacyConfiguration { + var delegate: (any BrowserServicesKit.PrivacyConfigurationDelegate)? var identifier: String = "id" var version: String? = "123456789" diff --git a/Tests/BrowserServicesKitTests/Autofill/AutofillTestHelper.swift b/Tests/BrowserServicesKitTests/Autofill/AutofillTestHelper.swift index 67e825e89..fde5bd69a 100644 --- a/Tests/BrowserServicesKitTests/Autofill/AutofillTestHelper.swift +++ b/Tests/BrowserServicesKitTests/Autofill/AutofillTestHelper.swift @@ -31,7 +31,8 @@ struct AutofillTestHelper { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) return manager } } diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift b/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift index 78d527308..de620e223 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift @@ -311,6 +311,7 @@ class PrivacyConfigurationManagerMock: PrivacyConfigurationManaging { } class PrivacyConfigurationMock: PrivacyConfiguration { + var delegate: (any PrivacyConfigurationDelegate)? func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool { true } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift index 3eb2db17e..d5b1411a8 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift @@ -177,14 +177,15 @@ final class DefaultFeatureFlaggerTests: XCTestCase { } // MARK: - Helpers - + private func createFeatureFlagger(withMockedConfigData data: Data = DefaultFeatureFlaggerTests.embeddedConfig()) -> DefaultFeatureFlagger { let mockEmbeddedData = MockEmbeddedDataProvider(data: data, etag: "embeddedConfigETag") let manager = PrivacyConfigurationManager(fetchedETag: nil, fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) let internalUserDecider = DefaultInternalUserDecider(store: internalUserDeciderStore) return DefaultFeatureFlagger(internalUserDecider: internalUserDecider, privacyConfigManager: manager) } @@ -195,7 +196,8 @@ final class DefaultFeatureFlaggerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) let internalUserDecider = DefaultInternalUserDecider(store: internalUserDeciderStore) overrides = CapturingFeatureFlagOverriding() diff --git a/Tests/BrowserServicesKitTests/Fingerprinting/FingerprintingReferenceTests.swift b/Tests/BrowserServicesKitTests/Fingerprinting/FingerprintingReferenceTests.swift index 68ce6e903..407cf616e 100644 --- a/Tests/BrowserServicesKitTests/Fingerprinting/FingerprintingReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/Fingerprinting/FingerprintingReferenceTests.swift @@ -62,7 +62,8 @@ final class FingerprintingReferenceTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) }() override func tearDown() { diff --git a/Tests/BrowserServicesKitTests/GPC/GPCReferenceTests.swift b/Tests/BrowserServicesKitTests/GPC/GPCReferenceTests.swift index afc346069..16df75491 100644 --- a/Tests/BrowserServicesKitTests/GPC/GPCReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/GPC/GPCReferenceTests.swift @@ -48,7 +48,8 @@ final class GPCReferenceTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) } func testGPCHeader() throws { diff --git a/Tests/BrowserServicesKitTests/LinkProtection/AmpMatchingTests.swift b/Tests/BrowserServicesKitTests/LinkProtection/AmpMatchingTests.swift index e4a2bb3d9..0504a4285 100644 --- a/Tests/BrowserServicesKitTests/LinkProtection/AmpMatchingTests.swift +++ b/Tests/BrowserServicesKitTests/LinkProtection/AmpMatchingTests.swift @@ -72,7 +72,8 @@ final class AmpMatchingTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) } private var contentBlockingManager: ContentBlockerRulesManager { diff --git a/Tests/BrowserServicesKitTests/LinkProtection/URLParameterTests.swift b/Tests/BrowserServicesKitTests/LinkProtection/URLParameterTests.swift index e6e2a3e73..3a019a0e7 100644 --- a/Tests/BrowserServicesKitTests/LinkProtection/URLParameterTests.swift +++ b/Tests/BrowserServicesKitTests/LinkProtection/URLParameterTests.swift @@ -59,7 +59,8 @@ final class URLParameterTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) } private lazy var urlParamTestSuite: URLParamRefTests = { diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AdClickAttributionFeatureTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AdClickAttributionFeatureTests.swift index 72b44867e..2f4afcafc 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AdClickAttributionFeatureTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AdClickAttributionFeatureTests.swift @@ -74,7 +74,8 @@ class AdClickAttributionFeatureTests: XCTestCase { fetchedData: nil, embeddedDataProvider: dataProvider, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) let feature = AdClickAttributionFeature(with: config) diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift index 8a4591fe7..d29e65fab 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift @@ -27,6 +27,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { var experimentManager: ExperimentCohortsManager! var manager: PrivacyConfigurationManager! var locale: Locale! + var assignedCohort: CohortID? + var assignedExperiment: SubfeatureID? let subfeatureName = "credentialsSaving" @@ -42,7 +44,11 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), locale: locale, - experimentCohortManager: experimentManager) + experimentCohortManager: experimentManager, + reportExperimentCohortAssignment: { [weak self] (cohort: CohortID, subfeatureID: SubfeatureID) in + self?.assignedCohort = cohort + self?.assignedExperiment = subfeatureID + }) } override func tearDown() { @@ -1223,4 +1229,44 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { XCTAssertTrue(activeExperiments.isEmpty) } + func testOnAssignmentReportAssignmentFunctionCalled() { + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "targets": [ + { + "localeCountry": "US" + } + ], + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + manager.reload(etag: "foo", data: featureJson) + let config = manager.privacyConfig + + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertEqual(self.assignedCohort, "control") + XCTAssertEqual(self.assignedExperiment, "credentialsSaving") + } } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift index e3be10a1f..0340dfb04 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift @@ -87,7 +87,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_,_ in }) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertEqual(manager.reload(etag: nil, data: nil), PrivacyConfigurationManager.ReloadResult.embedded) @@ -112,7 +113,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: downloadedConfig, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertEqual(manager.fetchedConfigData?.etag, downloadedConfigETag) @@ -135,7 +137,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -160,7 +163,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: corruptedConfig, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -188,7 +192,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -219,7 +224,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: corruptedConfig, embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -241,7 +247,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: corruptedConfig, embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -289,7 +296,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -346,7 +354,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -419,7 +428,8 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, internalUserDecider: DefaultInternalUserDecider(), - installDate: installDate).privacyConfig + installDate: installDate, + reportExperimentCohortAssignment: {_, _ in}).privacyConfig } func testInstalledDaysCheckReturnsCorrectly() { @@ -502,7 +512,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore)) + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig mockInternalUserStore.isInternalUser = true @@ -546,7 +557,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -564,7 +576,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore)) + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig mockInternalUserStore.isInternalUser = true @@ -581,7 +594,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -620,7 +634,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -655,7 +670,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -709,7 +725,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -784,7 +801,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -819,7 +837,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -846,7 +865,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -861,7 +881,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -875,7 +896,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -890,7 +912,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig @@ -940,7 +963,8 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale) + locale: locale, + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -957,7 +981,8 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale) + locale: locale, + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -974,7 +999,8 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale) + locale: locale, + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -1014,7 +1040,8 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale) + locale: locale, + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -1061,7 +1088,8 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale) + locale: locale, + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) @@ -1080,7 +1108,8 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale) + locale: locale, + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) @@ -1168,7 +1197,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore)) + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig XCTAssertFalse(config.trackerAllowlist.entries.isEmpty) @@ -1182,7 +1212,8 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore)) + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), + reportExperimentCohortAssignment: {_, _ in}) let config = manager.privacyConfig XCTAssert(config.trackerAllowlist.entries.isEmpty) diff --git a/Tests/BrowserServicesKitTests/ReferrerTrimming/ReferrerTrimmingTests.swift b/Tests/BrowserServicesKitTests/ReferrerTrimming/ReferrerTrimmingTests.swift index 02fffdf3e..4f6b43b92 100644 --- a/Tests/BrowserServicesKitTests/ReferrerTrimming/ReferrerTrimmingTests.swift +++ b/Tests/BrowserServicesKitTests/ReferrerTrimming/ReferrerTrimmingTests.swift @@ -63,7 +63,8 @@ class ReferrerTrimmingTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) } private var contentBlockingManager: ContentBlockerRulesManager { diff --git a/Tests/BrowserServicesKitTests/Resources/privacy-reference-tests b/Tests/BrowserServicesKitTests/Resources/privacy-reference-tests index 6133e7d9d..a603ff9af 160000 --- a/Tests/BrowserServicesKitTests/Resources/privacy-reference-tests +++ b/Tests/BrowserServicesKitTests/Resources/privacy-reference-tests @@ -1 +1 @@ -Subproject commit 6133e7d9d9cd5f1b925cab1971b4d785dc639df7 +Subproject commit a603ff9af22ca3ff7ce2e7ffbfe18c447d9f23e8 diff --git a/Tests/BrowserServicesKitTests/SmarterEncryption/HTTPSUpgradeReferenceTests.swift b/Tests/BrowserServicesKitTests/SmarterEncryption/HTTPSUpgradeReferenceTests.swift index 3fb3bfe77..d1b7faac5 100644 --- a/Tests/BrowserServicesKitTests/SmarterEncryption/HTTPSUpgradeReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/SmarterEncryption/HTTPSUpgradeReferenceTests.swift @@ -79,7 +79,8 @@ final class HTTPSUpgradeReferenceTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) } private lazy var httpsUpgradesTestSuite: HTTPSUpgradesRefTests = { diff --git a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift index 0334f4f7a..6f619cf1a 100644 --- a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift +++ b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift @@ -212,7 +212,8 @@ final class SubscriptionFeatureAvailabilityTests: XCTestCase { } class MockPrivacyConfiguration: PrivacyConfiguration { - + var delegate: (any PrivacyConfigurationDelegate)? + func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool { true } func stateFor(featureKey: BrowserServicesKit.PrivacyFeature, versionProvider: BrowserServicesKit.AppVersionProvider) -> BrowserServicesKit.PrivacyConfigurationFeatureState { diff --git a/Tests/DDGSyncTests/Mocks/Mocks.swift b/Tests/DDGSyncTests/Mocks/Mocks.swift index b65b8abdb..b95cb9455 100644 --- a/Tests/DDGSyncTests/Mocks/Mocks.swift +++ b/Tests/DDGSyncTests/Mocks/Mocks.swift @@ -162,6 +162,7 @@ class MockPrivacyConfigurationManager: PrivacyConfigurationManaging { } class MockPrivacyConfiguration: PrivacyConfiguration { + var delegate: (any PrivacyConfigurationDelegate)? func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool { true } diff --git a/Tests/DuckPlayerTests/DuckPlayerContingencyHandlerTests.swift b/Tests/DuckPlayerTests/DuckPlayerContingencyHandlerTests.swift index 15414161a..2ac1e79b3 100644 --- a/Tests/DuckPlayerTests/DuckPlayerContingencyHandlerTests.swift +++ b/Tests/DuckPlayerTests/DuckPlayerContingencyHandlerTests.swift @@ -30,7 +30,8 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: {_, _ in }) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertTrue(handler.shouldDisplayContingencyMessage) @@ -44,7 +45,8 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertFalse(handler.shouldDisplayContingencyMessage) @@ -58,7 +60,8 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertFalse(handler.shouldDisplayContingencyMessage) @@ -72,7 +75,8 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertFalse(handler.shouldDisplayContingencyMessage) @@ -86,7 +90,8 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertEqual(handler.learnMoreURL, URL(string: MockConfig.learnMoreURL)) @@ -99,7 +104,8 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider()) + internalUserDecider: DefaultInternalUserDecider(), + reportExperimentCohortAssignment: { _, _ in }) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertNil(handler.learnMoreURL) diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift new file mode 100644 index 000000000..ca37ec54f --- /dev/null +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -0,0 +1,551 @@ +// +// PixelExperimentKitTests.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import XCTest +@testable import PixelExperimentKit +@testable import BrowserServicesKit +import PixelKit + +final class PixelExperimentKitTests: XCTestCase { + var featureJson: Data = "{}".data(using: .utf8)! + var mockPixelStore: MockExperimentActionPixelStore! + var mockExperimentStore: MockExperimentDataStore! + var mockEmbeddedData: MockEmbeddedDataProvider! + var privacyConfigurationManager: PrivacyConfigurationManager! + var firedEvent: PixelKitEvent? + var firedFrequency: PixelKit.Frequency? + var firedIncludeAppVersion: Bool? + + override func setUp() { + super.setUp() + mockEmbeddedData = MockEmbeddedDataProvider(data: featureJson, etag: "test") + mockPixelStore = MockExperimentActionPixelStore() + mockExperimentStore = MockExperimentDataStore() + privacyConfigurationManager = PrivacyConfigurationManager(fetchedETag: nil, + fetchedData: nil, + embeddedDataProvider: mockEmbeddedData, + localProtection: MockDomainsProtectionStore(), + internalUserDecider: DefaultInternalUserDecider(store: MockInternalUserStoring()), + experimentCohortManager: ExperimentCohortsManager(store: mockExperimentStore), + reportExperimentCohortAssignment: { _, _ in }) + PixelKit.configureExperimentKit(privacyConfigManager: privacyConfigurationManager, store: mockPixelStore, fire: { event, frequency, includeAppVersion in + self.firedEvent = event + self.firedFrequency = frequency + self.firedIncludeAppVersion = includeAppVersion + }) + } + + override func tearDown() { + mockEmbeddedData = nil + mockPixelStore = nil + mockExperimentStore = nil + privacyConfigurationManager = nil + firedEvent = nil + firedFrequency = nil + firedIncludeAppVersion = nil + } + + func testFireExperimentEnrollmentPixelSendsExpectedData() { + // GIVEN + let subfeatureID = "testSubfeature" + let cohort = "A" + let enrollmentDate = Date(timeIntervalSince1970: 0) + let experimentData = ExperimentData(parentID: "parent", cohort: cohort, enrollmentDate: enrollmentDate) + let expectedEventName = "experiment_enroll_\(subfeatureID)_\(cohort)" + let expectedParameters = ["enrollmentDate": enrollmentDate.toYYYYMMDDInET()] + + // WHEN + PixelKit.fireExperimentEnrollmentPixel(subfeatureID: subfeatureID, experiment: experimentData) + + // THEN + XCTAssertEqual(firedEvent?.name, expectedEventName) + XCTAssertEqual(firedEvent?.parameters, expectedParameters) + XCTAssertEqual(firedFrequency, .uniqueIncludingParameters) + XCTAssertFalse(firedIncludeAppVersion ?? true) + } + + func testFireExperimentPixel_WithValidExperimentAndConversionWindowAndValueNotNumber() { + // GIVEN + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + privacyConfigurationManager.reload(etag: "", data: featureJson) + + let subfeatureID = "credentialsSaving" + let cohort = "control" + let enrollmentDate = Date().addingTimeInterval(-3 * 24 * 60 * 60) // 5 days ago + print(enrollmentDate) + let conversionWindow = 3...3 + let value = "true" + let expectedEventName = "experiment_metrics_\(subfeatureID)_\(cohort)" + let expectedParameters = [ + "metric": "someMetric", + "conversionWindowDays": "3-3", + "value": value, + "enrollmentDate": enrollmentDate.toYYYYMMDDInET() + ] + let experimentData = ExperimentData(parentID: "autofill", cohort: cohort, enrollmentDate: enrollmentDate) + mockExperimentStore.experiments = [subfeatureID: experimentData] + + // WHEN + PixelKit.fireExperimentPixel(for: subfeatureID, metric: "someMetric", conversionWindowDays: conversionWindow, value: value) + + // THEN + XCTAssertEqual(firedEvent?.name, expectedEventName) + XCTAssertEqual(firedEvent?.parameters, expectedParameters) + XCTAssertEqual(firedFrequency, .uniqueIncludingParameters) + XCTAssertFalse(firedIncludeAppVersion ?? true) + } + + func testFireExperimentPixel_WithValidExperimentAndConversionWindowAndValue1() { + // GIVEN + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + privacyConfigurationManager.reload(etag: "", data: featureJson) + + let subfeatureID = "credentialsSaving" + let cohort = "control" + let enrollmentDate = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago + print(enrollmentDate) + let conversionWindow = 3...7 + let value = "1" + let expectedEventName = "experiment_metrics_\(subfeatureID)_\(cohort)" + let expectedParameters = [ + "metric": "someMetric", + "conversionWindowDays": "3-7", + "value": value, + "enrollmentDate": enrollmentDate.toYYYYMMDDInET() + ] + let experimentData = ExperimentData(parentID: "autofill", cohort: cohort, enrollmentDate: enrollmentDate) + mockExperimentStore.experiments = [subfeatureID: experimentData] + + // WHEN + PixelKit.fireExperimentPixel(for: subfeatureID, metric: "someMetric", conversionWindowDays: conversionWindow, value: value) + + // THEN + XCTAssertEqual(firedEvent?.name, expectedEventName) + XCTAssertEqual(firedEvent?.parameters, expectedParameters) + XCTAssertEqual(firedFrequency, .uniqueIncludingParameters) + XCTAssertFalse(firedIncludeAppVersion ?? true) + } + + func testFireExperimentPixel_WithValidExperimentAndConversionWindowAndValueN() { + // GIVEN + featureJson = + """ + { + "features": { + "autofill": { + "state": "enabled", + "exceptions": [], + "features": { + "credentialsSaving": { + "state": "enabled", + "minSupportedVersion": 2, + "cohorts": [ + { + "name": "control", + "weight": 1 + }, + { + "name": "blue", + "weight": 0 + } + ] + } + } + } + } + } + """.data(using: .utf8)! + privacyConfigurationManager.reload(etag: "", data: featureJson) + + let subfeatureID = "credentialsSaving" + let cohort = "control" + let enrollmentDate = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago + print(enrollmentDate) + let conversionWindow = 3...7 + let randomNumber = Int.random(in: 1...100) + let value = "\(randomNumber)" + let expectedEventName = "experiment_metrics_\(subfeatureID)_\(cohort)" + let expectedParameters = [ + "metric": "someMetric", + "conversionWindowDays": "3-7", + "value": value, + "enrollmentDate": enrollmentDate.toYYYYMMDDInET() + ] + let experimentData = ExperimentData(parentID: "autofill", cohort: cohort, enrollmentDate: enrollmentDate) + mockExperimentStore.experiments = [subfeatureID: experimentData] + + // WHEN calling fire before expected number of calls + for n in 0.. Int { + return store[defaultName] ?? 0 + } + + func set(_ value: Int, forKey defaultName: String) { + store[defaultName] = value + } +} + +class MockExperimentDataStore: ExperimentsDataStoring { + var experiments: Experiments? +} + +class MockEmbeddedDataProvider: EmbeddedDataProvider { + var embeddedDataEtag: String + + var embeddedData: Data + + init(data: Data, etag: String) { + embeddedData = data + embeddedDataEtag = etag + } +} + +final class MockDomainsProtectionStore: DomainsProtectionStore { + var unprotectedDomains = Set() + + func disableProtection(forDomain domain: String) { + unprotectedDomains.insert(domain) + } + + func enableProtection(forDomain domain: String) { + unprotectedDomains.remove(domain) + } +} + +final class MockInternalUserStoring: InternalUserStoring { + var isInternalUser: Bool = false +} From 69dc2fd9e05b497f32f34f97f401b260d0934c33 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Wed, 27 Nov 2024 23:32:39 +0100 Subject: [PATCH 21/49] implement framework --- .../ExperimentCohortsManager.swift | 59 ++++--- .../ExperimentsDataStore.swift | 0 .../FeatureFlagger/FeatureFlagger.swift | 163 +++++++++++++++++- .../AppPrivacyConfiguration.swift | 115 ++++-------- .../PrivacyConfig/PrivacyConfiguration.swift | 24 +-- .../PrivacyConfigurationManager.swift | 2 - .../PrivacyConfigurationManagerMock.swift | 18 +- .../UserContentControllerTests.swift | 16 +- .../ContentBlocker/WebViewTestHelper.swift | 8 +- .../DefaultFeatureFlaggerTests.swift | 156 ++++++++++++++++- .../ExperimentCohortsManagerTests.swift | 44 +++-- .../ExperimentsDataStoreTests.swift | 16 +- .../FeatureFlagLocalOverridesTests.swift | 2 +- .../FeatureFlaggerExperimentsTests.swift} | 131 +++++--------- .../FeatureFlagging/TestFeatureFlag.swift | 4 +- .../AppPrivacyConfigurationTests.swift | 14 +- ...SubscriptionFeatureAvailabilityTests.swift | 16 +- Tests/DDGSyncTests/Mocks/Mocks.swift | 16 +- 18 files changed, 537 insertions(+), 267 deletions(-) rename Sources/BrowserServicesKit/{PrivacyConfig => FeatureFlagger}/ExperimentCohortsManager.swift (62%) rename Sources/BrowserServicesKit/{PrivacyConfig => FeatureFlagger}/ExperimentsDataStore.swift (100%) rename Tests/BrowserServicesKitTests/{PrivacyConfig => FeatureFlagging}/ExperimentCohortsManagerTests.swift (72%) rename Tests/BrowserServicesKitTests/{PrivacyConfig => FeatureFlagging}/ExperimentsDataStoreTests.swift (90%) rename Tests/BrowserServicesKitTests/{PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift => FeatureFlagging/FeatureFlaggerExperimentsTests.swift} (85%) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift similarity index 62% rename from Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift rename to Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift index e52e7b569..7231b98bf 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift @@ -30,31 +30,44 @@ public typealias ParentFeatureID = String public struct ExperimentData: Codable, Equatable { public let parentID: String - public let cohort: String + public let cohortID: String public let enrollmentDate: Date } public typealias Experiments = [String: ExperimentData] public protocol ExperimentCohortsManaging { - /// Retrieves all the experiments a user is enrolled into + /// Retrieves all the experiments a user is enrolled in var experiments: Experiments? { get } - /// Retrieves the assigned cohort for a given experiment subfeature, or attempts to assign a new cohort if none is currently assigned - /// and `assignIfEnabled` is set to true. If a cohort is already assigned but does not match any valid cohorts for the experiment, - /// the cohort will be removed. + /// Resolves the cohort for a given experiment subfeature. + /// + /// This method determines whether the user is currently assigned to a valid cohort + /// for the specified experiment. If the assigned cohort is valid (i.e., it matches + /// one of the experiment's defined cohorts), the method returns the assigned cohort. + /// Otherwise, the invalid cohort is removed, and a new cohort is assigned if + /// `isAssignCohortEnabled` is `true`. /// /// - Parameters: - /// - experiment: The `ExperimentSubfeature` for which to retrieve, assign, or remove a cohort. This subfeature includes - /// relevant identifiers and potential cohorts that may be assigned. - /// - assignIfEnabled: A Boolean value that determines whether a new cohort should be assigned if none is currently assigned. - /// If `true`, the function will attempt to assign a cohort from the available options; otherwise, it will only check for existing assignments. + /// - experiment: The `ExperimentSubfeature` representing the experiment and its associated cohorts. + /// - isAssignCohortEnabled: A Boolean value indicating whether cohort assignment is allowed + /// if the user is not already assigned to a valid cohort. + /// + /// - Returns: The valid `CohortID` assigned to the user for the experiment, or `nil` + /// if no valid cohort exists and `isAssignCohortEnabled` is `false`. + /// + /// - Behavior: + /// 1. Retrieves the currently assigned cohort for the experiment using the `subfeatureID`. + /// 2. Validates if the assigned cohort exists within the experiment's cohort list: + /// - If valid, the assigned cohort is returned. + /// - If invalid, the cohort is removed from storage. + /// 3. If cohort assignment is enabled (`isAssignCohortEnabled` is `true`), a new cohort + /// is assigned based on the experiment's cohort weights and saved in storage. + /// - Cohort assignment is probabilistic, determined by the cohort weights. /// - /// - Returns: A tuple containing: - /// - `cohortID`: The identifier of the assigned cohort if one exists, or `nil` if no cohort was assigned, if assignment failed, or if the cohort was removed. - /// - `didAttemptAssignment`: A Boolean indicating whether an assignment attempt was made. This will be `true` if `assignIfEnabled` - /// is `true` and no cohort was previously assigned, and `false` otherwise. - func cohort(for experiment: ExperimentSubfeature, assignIfEnabled: Bool) -> (cohortID: CohortID?, didAttemptAssignment: Bool) + func resolveCohort(for experiment: ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? + + } public class ExperimentCohortsManager: ExperimentCohortsManaging { @@ -76,18 +89,21 @@ public class ExperimentCohortsManager: ExperimentCohortsManaging { self.randomizer = randomizer } - public func cohort(for experiment: ExperimentSubfeature, assignIfEnabled: Bool) -> (cohortID: CohortID?, didAttemptAssignment: Bool) { + public func resolveCohort(for experiment: ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? { queue.sync { let assignedCohort = cohort(for: experiment.subfeatureID) if experiment.cohorts.contains(where: { $0.name == assignedCohort }) { - return (assignedCohort, false) + return (assignedCohort) } else { removeCohort(from: experiment.subfeatureID) } - - return assignIfEnabled ? (assignCohort(to: experiment), true) : (nil, true) + return isAssignCohortEnabled ? assignCohort(to: experiment) : nil } } +} + +// MARK: Helper functions +extension ExperimentCohortsManager { private func assignCohort(to subfeature: ExperimentSubfeature) -> CohortID? { let cohorts = subfeature.cohorts @@ -109,10 +125,10 @@ public class ExperimentCohortsManager: ExperimentCohortsManaging { func cohort(for subfeatureID: SubfeatureID) -> CohortID? { guard let experiments = store.experiments else { return nil } - return experiments[subfeatureID]?.cohort + return experiments[subfeatureID]?.cohortID } - private func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? { + private func enrollmentDate(for subfeatureID: SubfeatureID) -> Date? { guard let experiments = store.experiments else { return nil } return experiments[subfeatureID]?.enrollmentDate } @@ -125,9 +141,8 @@ public class ExperimentCohortsManager: ExperimentCohortsManaging { private func saveCohort(_ cohort: CohortID, in experimentID: SubfeatureID, parentID: ParentFeatureID) { var experiments = store.experiments ?? Experiments() - let experimentData = ExperimentData(parentID: parentID, cohort: cohort, enrollmentDate: Date()) + let experimentData = ExperimentData(parentID: parentID, cohortID: cohort, enrollmentDate: Date()) experiments[experimentID] = experimentData store.experiments = experiments } - } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentsDataStore.swift similarity index 100% rename from Sources/BrowserServicesKit/PrivacyConfig/ExperimentsDataStore.swift rename to Sources/BrowserServicesKit/FeatureFlagger/ExperimentsDataStore.swift diff --git a/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift b/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift index d6f5fdee4..191ec0063 100644 --- a/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift @@ -18,6 +18,8 @@ import Foundation +public protocol CohortEnum: RawRepresentable, CaseIterable where RawValue == String {} + /// This protocol defines a common interface for feature flags managed by FeatureFlagger. /// /// It should be implemented by the feature flag type in client apps. @@ -53,7 +55,43 @@ public protocol FeatureFlagDescribing: CaseIterable { /// case .sync: /// return .disabled /// case .cookieConsent: - /// return .internalOnly + /// return .internalOnly() + /// case .credentialsAutofill: + /// return .remoteDevelopment(.subfeature(AutofillSubfeature.credentialsAutofill)) + /// case .duckPlayer: + /// return .remoteReleasable(.feature(.duckPlayer)) + /// } + /// } + /// ``` + var source: FeatureFlagSource { get } +} + +/// This protocol defines a common interface for experiment feature flags managed by FeatureFlagger. +/// +/// It should be implemented by the feature flag type in client apps. +/// +public protocol FeatureFlagExperimentDescribing { + + /// Returns a string representation of the flag + var rawValue: String { get } + + /// Defines the source of the experiment feature flag, which corresponds to + /// where the final flag value should come from. + /// + /// Example client implementation: + /// + /// ``` + /// public enum FeatureFlag: FeatureFlagDescribing { + /// case sync + /// case autofill + /// case cookieConsent + /// case duckPlayer + /// + /// var source: FeatureFlagSource { + /// case .sync: + /// return .disabled + /// case .cookieConsent: + /// return .internalOnly(cohort) /// case .credentialsAutofill: /// return .remoteDevelopment(.subfeature(AutofillSubfeature.credentialsAutofill)) /// case .duckPlayer: @@ -62,6 +100,9 @@ public protocol FeatureFlagDescribing: CaseIterable { /// } /// ``` var source: FeatureFlagSource { get } + + + associatedtype Cohort: CohortEnum } public enum FeatureFlagSource { @@ -69,7 +110,7 @@ public enum FeatureFlagSource { case disabled /// Enabled for internal users only. Cannot be toggled remotely - case internalOnly + case internalOnly((any CohortEnum)? = nil) /// Toggled remotely using PrivacyConfiguration but only for internal users. Otherwise, disabled. case remoteDevelopment(PrivacyConfigFeatureLevel) @@ -107,6 +148,62 @@ public protocol FeatureFlagger: AnyObject { /// when the non-overridden feature flag value is required. /// func isFeatureOn(for featureFlag: Flag, allowOverride: Bool) -> Bool + + /// Resolves the cohort for a subfeature if the subfeature is enabled. + /// + /// This method checks the state of the subfeature in the `PrivacyConfiguration`. If the subfeature + /// is enabled or disabled due to a target mismatch, it resolves the cohort using the `ExperimentManager`. + /// + /// - Parameter subfeature: A subfeature conforming to `PrivacySubfeature`. + /// + /// - Returns: The `CohortID` associated with the subfeature if enabled, or `nil` otherwise. + /// + /// - Behavior: + /// - If the subfeature state is `.enabled`: + /// - Resolves and assigns a cohort using `resolveCohort(isAssignCohortEnabled: true)`. + /// - If the subfeature state is `.disabled(.targetDoesNotMatch)`: + /// - Resolves the cohort without assigning a new one (`isAssignCohortEnabled: false`). + /// - For other states: Returns `nil`. + /// + func getCohortIfEnabled(_ subfeature: any PrivacySubfeature) -> CohortID? + + /// Retrieves the cohort for a feature flag if the feature is enabled. + /// + /// This method determines the source of the feature flag and evaluates its eligibility based on + /// the user's internal status and the privacy configuration. It supports different sources, such as + /// disabled features, internal-only features, and remotely toggled features. + /// + /// - Parameter featureFlag: A feature flag conforming to `FeatureFlagDescribing`. + /// + /// - Returns: The `CohortID` associated with the feature flag, or `nil` if the feature is disabled or + /// does not meet the eligibility criteria. + /// + /// - Behavior: + /// - For `.disabled`: Returns `nil`. + /// - For `.internalOnly`: Returns the cohort if the user is an internal user. + /// - For `.remoteDevelopment` and `.remoteReleasable`: + /// - If the feature is a subfeature, resolves its cohort using `getCohortIfEnabled(_ subfeature:)`. + /// - Returns `nil` if the user is not eligible. + /// + func getCohortIfEnabled(for featureFlag: Flag) -> (any CohortEnum)? + + /// Retrieves all active experiments currently assigned to the user. + /// + /// This method iterates over the experiments stored in the `ExperimentManager` and checks their state + /// against the current `PrivacyConfiguration`. If an experiment's state is enabled or disabled due to + /// a target mismatch, and its assigned cohort matches the resolved cohort, it is considered active. + /// + /// - Returns: A dictionary of active experiments where the key is the experiment's subfeature ID, + /// and the value is the associated `ExperimentData`. + /// + /// - Behavior: + /// 1. Fetches all enrolled experiments from the `ExperimentManager`. + /// 2. For each experiment: + /// - Retrieves its state from the `PrivacyConfiguration`. + /// - Validates its assigned cohort using `resolveCohort` in the `ExperimentManager`. + /// 3. If the experiment passes validation, it is added to the result dictionary. + /// + func getAllActiveExperiments() -> Experiments } public extension FeatureFlagger { @@ -126,14 +223,17 @@ public class DefaultFeatureFlagger: FeatureFlagger { public let internalUserDecider: InternalUserDecider public let privacyConfigManager: PrivacyConfigurationManaging + private let experimentManager: ExperimentCohortsManaging? public let localOverrides: FeatureFlagLocalOverriding? public init( internalUserDecider: InternalUserDecider, - privacyConfigManager: PrivacyConfigurationManaging + privacyConfigManager: PrivacyConfigurationManaging, + experimentManager: ExperimentCohortsManaging? ) { self.internalUserDecider = internalUserDecider self.privacyConfigManager = privacyConfigManager + self.experimentManager = experimentManager self.localOverrides = nil } @@ -141,11 +241,13 @@ public class DefaultFeatureFlagger: FeatureFlagger { internalUserDecider: InternalUserDecider, privacyConfigManager: PrivacyConfigurationManaging, localOverrides: FeatureFlagLocalOverriding, + experimentManager: ExperimentCohortsManaging?, for: Flag.Type ) { self.internalUserDecider = internalUserDecider self.privacyConfigManager = privacyConfigManager self.localOverrides = localOverrides + self.experimentManager = experimentManager localOverrides.featureFlagger = self // Clear all overrides if not an internal user @@ -173,6 +275,61 @@ public class DefaultFeatureFlagger: FeatureFlagger { } } + public func getAllActiveExperiments() -> Experiments { + var activeExperiments = [String: ExperimentData]() + guard let enrolledExperiments = experimentManager?.experiments else { return activeExperiments } + let config = privacyConfigManager.privacyConfig + + for (subfeatureID, experimentData) in enrolledExperiments { + let state = config.stateFor(subfeatureID: subfeatureID, parentFeatureID: experimentData.parentID) + guard state == .enabled || state == .disabled(.targetDoesNotMatch) else { continue } + let cohorts = config.cohorts(subfeatureID: subfeatureID, parentFeatureID: experimentData.parentID) ?? [] + let experimentSubfeature = ExperimentSubfeature(parentID: experimentData.parentID, subfeatureID: subfeatureID, cohorts: cohorts) + + if experimentManager?.resolveCohort(for: experimentSubfeature, isAssignCohortEnabled: false) == experimentData.cohortID { + activeExperiments[subfeatureID] = experimentData + } + } + + return activeExperiments + } + + public func getCohortIfEnabled(for featureFlag: Flag) -> (any CohortEnum)? { + switch featureFlag.source { + case .disabled: + return nil + case .internalOnly(let cohort): + return cohort + case .remoteDevelopment(_) where !internalUserDecider.isInternalUser: + return nil + case .remoteReleasable(let featureType), + .remoteDevelopment(let featureType) where internalUserDecider.isInternalUser: + if case .subfeature(let subfeature) = featureType { + if let resolvedCohortID = getCohortIfEnabled(subfeature) { + return Flag.Cohort.allCases.first { return $0.rawValue == resolvedCohortID } + } + } + return nil + default: + return nil + } + } + + public func getCohortIfEnabled(_ subfeature: any PrivacySubfeature) -> CohortID? { + let config = privacyConfigManager.privacyConfig + let featureState = config.stateFor(subfeature) + let cohorts = config.cohorts(for: subfeature) + let experiment = ExperimentSubfeature(parentID: subfeature.parent.rawValue, subfeatureID: subfeature.rawValue, cohorts: cohorts ?? []) + switch featureState { + case .enabled: + return experimentManager?.resolveCohort(for: experiment, isAssignCohortEnabled: true) + case .disabled(.targetDoesNotMatch): + return experimentManager?.resolveCohort(for: experiment, isAssignCohortEnabled: false) + default: + return nil + } + } + private func isEnabled(_ featureType: PrivacyConfigFeatureLevel) -> Bool { switch featureType { case .feature(let feature): diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 46e827a22..2904dcf95 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -34,7 +34,6 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { private let internalUserDecider: InternalUserDecider private let userDefaults: UserDefaults private let locale: Locale - private let experimentManager: ExperimentCohortsManaging private let installDate: Date? static let experimentManagerQueue = DispatchQueue(label: "com.experimentManager.queue") @@ -44,7 +43,6 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { internalUserDecider: InternalUserDecider, userDefaults: UserDefaults = UserDefaults(), locale: Locale = Locale.current, - experimentManager: ExperimentCohortsManaging = ExperimentCohortsManager(), installDate: Date? = nil) { self.data = data self.identifier = identifier @@ -52,7 +50,6 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { self.internalUserDecider = internalUserDecider self.userDefaults = userDefaults self.locale = locale - self.experimentManager = experimentManager self.installDate = installDate } @@ -188,10 +185,9 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } public func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, - cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool { - switch stateFor(subfeature, cohortID: cohortID, versionProvider: versionProvider, randomizer: randomizer) { + switch stateFor(subfeature, versionProvider: versionProvider, randomizer: randomizer) { case .enabled: return true case .disabled: @@ -199,54 +195,19 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } } - public func getAllActiveExperiments(versionProvider: AppVersionProvider, - randomizer: (Range) -> Double) -> Experiments { - Self.experimentManagerQueue.sync { - guard let assignedExperiments = experimentManager.experiments else { return [:] } - return assignedExperiments.filter { key, value in - stateFor(subfeatureID: key, experimentData: value, versionProvider: versionProvider, randomizer: randomizer) == .enabled - } - } - } - - private func stateFor(subfeatureID: SubfeatureID, experimentData: ExperimentData, versionProvider: AppVersionProvider, - randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - guard let parentFeature = PrivacyFeature(rawValue: experimentData.parentID), - let subfeatureData = subfeatures(for: parentFeature)[subfeatureID] else { - return .disabled(.featureMissing) - } - return stateFor(parentFeature: parentFeature, - subfeatureData: subfeatureData, - subfeatureID: subfeatureID, - cohortID: experimentData.cohort, - assignCohortEnabled: false, - versionProvider: versionProvider, - randomizer: randomizer) - } - public func stateFor(_ subfeature: any PrivacySubfeature, - cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - Self.experimentManagerQueue.sync { - guard let subfeatureData = subfeatures(for: subfeature.parent)[subfeature.rawValue] else { - return .disabled(.featureMissing) - } - - return stateFor(parentFeature: subfeature.parent, - subfeatureData: subfeatureData, - subfeatureID: subfeature.rawValue, - cohortID: cohortID, - versionProvider: versionProvider, - randomizer: randomizer) + guard let subfeatureData = subfeatures(for: subfeature.parent)[subfeature.rawValue] else { + return .disabled(.featureMissing) } + + return stateFor(subfeatureID: subfeature.rawValue, subfeatureData: subfeatureData, parentFeature: subfeature.parent, versionProvider: versionProvider, randomizer: randomizer) } - private func stateFor(parentFeature: PrivacyFeature, + private func stateFor(subfeatureID: SubfeatureID, subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature, - subfeatureID: SubfeatureID, - cohortID: CohortID?, - assignCohortEnabled: Bool = true, + parentFeature: PrivacyFeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { // Step 1: Check parent feature state @@ -272,43 +233,8 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { return .disabled(.stillInRollout) } - // Step 5: Check if a cohort was passed in the func - // If no corhort passed check for Target and Rollout - guard let passedCohort = cohortID else { - return checkTargets(subfeatureData) - } - - // Step 5: Cohort handling - // Check if cohort assigned and matches passed cohort - // If cohort not assigned - // Tries to assign if matching target - // Check if cohort assigned and matches passed cohort - return checkCohortState(subfeatureData, - passedCohort: passedCohort, - assignCohortEnabled: assignCohortEnabled, - subfeatureID: subfeatureID, - parentFeature: parentFeature) - } - - // Check if cohort assigned and matches passed cohort - // If cohort not assigned - // Tries to assign if matching target - // Check if cohort assigned and matches passed cohort - private func checkCohortState(_ subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature, - passedCohort: CohortID?, - assignCohortEnabled: Bool, - subfeatureID: SubfeatureID, - parentFeature: PrivacyFeature) -> PrivacyConfigurationFeatureState { - let cohorts = subfeatureData.cohorts ?? [] - let targetsState = checkTargets(subfeatureData) - let assignIfEnabled = assignCohortEnabled && targetsState == .enabled - let assignedCohortResponse = experimentManager.cohort(for: ExperimentSubfeature(parentID: parentFeature.rawValue, subfeatureID: subfeatureID, cohorts: cohorts), assignIfEnabled: assignIfEnabled) - let possibleDisabledReason: PrivacyConfigurationFeatureDisabledReason = assignedCohortResponse.didAttemptAssignment && targetsState != .enabled ? .targetDoesNotMatch : .experimentCohortDoesNotMatch - if let assignedCohort = assignedCohortResponse.cohortID { - return (assignedCohort == passedCohort) ? .enabled : .disabled(possibleDisabledReason) - } else { - return .disabled(possibleDisabledReason) - } + // Step 5: Check Targets + return checkTargets(subfeatureData) } private func checkTargets(_ subfeatureData: PrivacyConfigurationData.PrivacyFeature.Feature?) -> PrivacyConfigurationFeatureState { @@ -320,7 +246,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { } private func matchTargets(targets: [PrivacyConfigurationData.PrivacyFeature.Feature.Target]) -> Bool { - return targets.contains { target in + targets.contains { target in (target.localeCountry == nil || target.localeCountry == locale.regionCode) && (target.localeLanguage == nil || target.localeLanguage == locale.languageCode) } @@ -354,7 +280,7 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { guard let domain = domain else { return true } return !isTempUnprotected(domain: domain) && !isUserUnprotected(domain: domain) && - !isInExceptionList(domain: domain, forFeature: .contentBlocking) + !isInExceptionList(domain: domain, forFeature: .contentBlocking) } public func isUserUnprotected(domain: String?) -> Bool { @@ -404,7 +330,26 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { public func userDisabledProtection(forDomain domain: String) { locallyUnprotected.disableProtection(forDomain: domain.punycodeEncodedHostname.lowercased()) } + +} + +extension AppPrivacyConfiguration { + public func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID, versionProvider: AppVersionProvider, + randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + guard let parentFeature = PrivacyFeature(rawValue: parentFeatureID) else { return .disabled(.featureMissing) } + guard let subfeatureData = subfeatures(for: parentFeature)[subfeatureID] else { return .disabled(.featureMissing) } + return stateFor(subfeatureID: subfeatureID, subfeatureData: subfeatureData, parentFeature: parentFeature, versionProvider: versionProvider, randomizer: randomizer) + } + + public func cohorts(for subfeature: any PrivacySubfeature) -> [PrivacyConfigurationData.Cohort]? { + subfeatures(for: subfeature.parent)[subfeature.rawValue]?.cohorts + } + + public func cohorts(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID) -> [PrivacyConfigurationData.Cohort]? { + guard let parentFeature = PrivacyFeature(rawValue: parentFeatureID) else { return nil } + return subfeatures(for: parentFeature)[subfeatureID]?.cohorts + } } extension Array where Element == String { diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift index 9131566e0..d7a6c9a9e 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift @@ -58,8 +58,8 @@ public protocol PrivacyConfiguration { func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool func stateFor(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> PrivacyConfigurationFeatureState - func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool - func stateFor(_ subfeature: any PrivacySubfeature, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState + func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool + func stateFor(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState /// Domains for which given PrivacyFeature is disabled. /// @@ -105,9 +105,10 @@ public protocol PrivacyConfiguration { /// Adds given domain to locally unprotected list. func userDisabledProtection(forDomain: String) - /// Gives the list of all the active experiments an user is enrolled in - func getAllActiveExperiments(versionProvider: AppVersionProvider, - randomizer: (Range) -> Double) -> Experiments + func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID, versionProvider: AppVersionProvider, + randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState + func cohorts(for subfeature: any PrivacySubfeature) -> [PrivacyConfigurationData.Cohort]? + func cohorts(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID) -> [PrivacyConfigurationData.Cohort]? } public extension PrivacyConfiguration { @@ -119,15 +120,16 @@ public extension PrivacyConfiguration { return stateFor(featureKey: featureKey, versionProvider: AppVersionProvider()) } - func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, cohortID: CohortID? = nil, randomizer: (Range) -> Double = Double.random(in:)) -> Bool { - return isSubfeatureEnabled(subfeature, cohortID: cohortID, versionProvider: AppVersionProvider(), randomizer: randomizer) + func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, randomizer: (Range) -> Double = Double.random(in:)) -> Bool { + return isSubfeatureEnabled(subfeature, versionProvider: AppVersionProvider(), randomizer: randomizer) } - func stateFor(_ subfeature: any PrivacySubfeature, cohortID: CohortID? = nil, randomizer: (Range) -> Double = Double.random(in:)) -> PrivacyConfigurationFeatureState { - return stateFor(subfeature, cohortID: cohortID, versionProvider: AppVersionProvider(), randomizer: randomizer) + func stateFor(_ subfeature: any PrivacySubfeature, randomizer: (Range) -> Double = Double.random(in:)) -> PrivacyConfigurationFeatureState { + return stateFor(subfeature, versionProvider: AppVersionProvider(), randomizer: randomizer) } - func getAllActiveExperiments(versionProvider: AppVersionProvider = AppVersionProvider(), randomizer: (Range) -> Double = Double.random(in:)) -> Experiments { - return getAllActiveExperiments(versionProvider: versionProvider, randomizer: randomizer) + func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID ,randomizer: (Range) -> Double = Double.random(in:)) -> PrivacyConfigurationFeatureState { + return stateFor(subfeatureID: subfeatureID, parentFeatureID: parentFeatureID, versionProvider: AppVersionProvider(), randomizer: randomizer) } + } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index fff2f1482..711ead877 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -134,7 +134,6 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { localProtection: localProtection, internalUserDecider: internalUserDecider, locale: locale, - experimentManager: experimentCohortManager, installDate: installDate) } @@ -143,7 +142,6 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { localProtection: localProtection, internalUserDecider: internalUserDecider, locale: locale, - experimentManager: experimentCohortManager, installDate: installDate) } diff --git a/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift b/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift index 76ecc8b87..29adc74fb 100644 --- a/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift +++ b/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift @@ -55,12 +55,12 @@ class PrivacyConfigurationMock: PrivacyConfiguration { } var enabledSubfeaturesForVersions: [String: Set] = [:] - func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool { + func isSubfeatureEnabled(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> Bool { return enabledSubfeaturesForVersions[subfeature.rawValue]?.contains(versionProvider.appVersion() ?? "") ?? false } - func stateFor(_ subfeature: any PrivacySubfeature, cohortID: CohortID?, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { - if isSubfeatureEnabled(subfeature, cohortID: cohortID, versionProvider: versionProvider, randomizer: randomizer) { + func stateFor(_ subfeature: any PrivacySubfeature, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState { + if isSubfeatureEnabled(subfeature, versionProvider: versionProvider, randomizer: randomizer) { return .enabled } return .disabled(.disabledInConfig) // this is not used in platform tests, so mocking this poorly for now @@ -98,6 +98,18 @@ class PrivacyConfigurationMock: PrivacyConfiguration { return userUnprotected.contains(domain ?? "") } + func stateFor(subfeatureID: BrowserServicesKit.SubfeatureID, parentFeatureID: BrowserServicesKit.ParentFeatureID, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { + return .enabled + } + + func cohorts(for subfeature: any BrowserServicesKit.PrivacySubfeature) -> [BrowserServicesKit.PrivacyConfigurationData.Cohort]? { + return nil + } + + func cohorts(subfeatureID: BrowserServicesKit.SubfeatureID, parentFeatureID: BrowserServicesKit.ParentFeatureID) -> [BrowserServicesKit.PrivacyConfigurationData.Cohort]? { + return nil + } + } class PrivacyConfigurationManagerMock: PrivacyConfigurationManaging { diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift b/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift index 78d527308..767250149 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift @@ -318,14 +318,26 @@ class PrivacyConfigurationMock: PrivacyConfiguration { return .enabled } - func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { + func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { true } - func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { + func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { return .enabled } + func stateFor(subfeatureID: BrowserServicesKit.SubfeatureID, parentFeatureID: BrowserServicesKit.ParentFeatureID, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { + return .enabled + } + + func cohorts(for subfeature: any BrowserServicesKit.PrivacySubfeature) -> [BrowserServicesKit.PrivacyConfigurationData.Cohort]? { + return nil + } + + func cohorts(subfeatureID: BrowserServicesKit.SubfeatureID, parentFeatureID: BrowserServicesKit.ParentFeatureID) -> [BrowserServicesKit.PrivacyConfigurationData.Cohort]? { + return nil + } + var identifier: String = "abcd" var version: String? = "123456789" var userUnprotectedDomains: [String] = [] diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift index db78351d9..346542682 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift @@ -227,9 +227,9 @@ final class WebKitTestHelper { } class MockExperimentCohortsManager: ExperimentCohortsManaging { - var experiments: BrowserServicesKit.Experiments? - - func cohort(for experiment: BrowserServicesKit.ExperimentSubfeature, assignIfEnabled: Bool) -> (cohortID: BrowserServicesKit.CohortID?, didAttemptAssignment: Bool) { - return (nil, true) + func resolveCohort(for experiment: BrowserServicesKit.ExperimentSubfeature, isAssignCohortEnabled: Bool) -> BrowserServicesKit.CohortID? { + return nil } + + var experiments: BrowserServicesKit.Experiments? } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift index 3eb2db17e..2387a9753 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift @@ -52,15 +52,18 @@ final class CapturingFeatureFlagOverriding: FeatureFlagLocalOverriding { final class DefaultFeatureFlaggerTests: XCTestCase { var internalUserDeciderStore: MockInternalUserStoring! + var experimentManager: MockExperimentManager! var overrides: CapturingFeatureFlagOverriding! override func setUp() { super.setUp() internalUserDeciderStore = MockInternalUserStoring() + experimentManager = MockExperimentManager() } override func tearDown() { internalUserDeciderStore = nil + experimentManager = nil super.tearDown() } @@ -72,9 +75,9 @@ final class DefaultFeatureFlaggerTests: XCTestCase { func testWhenInternalOnly_returnsIsInternalUserValue() { let featureFlagger = createFeatureFlagger() internalUserDeciderStore.isInternalUser = false - XCTAssertFalse(featureFlagger.isFeatureOn(for: FeatureFlagSource.internalOnly)) + XCTAssertFalse(featureFlagger.isFeatureOn(for: FeatureFlagSource.internalOnly())) internalUserDeciderStore.isInternalUser = true - XCTAssertTrue(featureFlagger.isFeatureOn(for: FeatureFlagSource.internalOnly)) + XCTAssertTrue(featureFlagger.isFeatureOn(for: FeatureFlagSource.internalOnly())) } func testWhenRemoteDevelopment_isNOTInternalUser_returnsFalse() { @@ -141,6 +144,129 @@ final class DefaultFeatureFlaggerTests: XCTestCase { assertFeatureFlagger(with: embeddedData, willReturn: false, for: sourceProvider) } + // MARK: - Experiments + + func testWhenGetCohortIfEnabled_andSourceDisabled_returnsNil() { + let featureFlagger = createFeatureFlagger() + let flag = FakeExperimentFlag(source: .disabled) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertNil(cohort) + } + + func estWhenGetCohortIfEnabled_andSourceInternal_returnsPassedCohort() { + let featureFlagger = createFeatureFlagger() + let flag = FakeExperimentFlag(source: .internalOnly(AutofillCohorts.blue)) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertEqual(cohort?.rawValue, AutofillCohorts.blue.rawValue) + } + + func testWhenGetCohortIfEnabled_andRemoteInternal_andInternalStateTrue_and_cohortAssigned_returnsAssignedCohort() { + internalUserDeciderStore.isInternalUser = true + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteDevelopment(.subfeature(AutofillSubfeature.credentialsAutofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertEqual(cohort?.rawValue, AutofillCohorts.control.rawValue) + } + + func testWhenGetCohortIfEnabled_andRemoteInternal_andInternalStateFalse_and_cohortAssigned_returnsNil() { + internalUserDeciderStore.isInternalUser = false + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteDevelopment(.subfeature(AutofillSubfeature.credentialsAutofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertNil(cohort) + } + + func testWhenGetCohortIfEnabled_andRemoteInternal_andInternalStateTrue_and_cohortAssigned_andFeaturePassed_returnsNil() { + internalUserDeciderStore.isInternalUser = true + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteDevelopment(.feature(.autofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertNil(cohort) + } + + func testWhenGetCohortIfEnabled_andRemoteInternal_andInternalStateTrue_and_cohortNotAssigned_returnsNil() { + internalUserDeciderStore.isInternalUser = true + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = nil + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteDevelopment(.subfeature(AutofillSubfeature.credentialsAutofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertNil(cohort) + } + + func testWhenGetCohortIfEnabled_andRemoteInternal_andInternalStateTrue_and_cohortAssignedButNorMatchingEnum_returnsNil() { + internalUserDeciderStore.isInternalUser = true + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = "some" + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteDevelopment(.subfeature(AutofillSubfeature.credentialsAutofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertNil(cohort) + } + + func testWhenGetCohortIfEnabled_andRemoteReleasable_and_cohortAssigned_returnsAssignedCohort() { + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteReleasable(.subfeature(AutofillSubfeature.credentialsAutofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertEqual(cohort?.rawValue, AutofillCohorts.control.rawValue) + } + + func testWhenGetCohortIfEnabled_andRemoteReleasable_and_cohortAssigned_andFeaturePassed_returnsNil() { + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteReleasable(.feature(.autofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertNil(cohort) + } + + func testWhenGetCohortIfEnabled_andRemoteReleasable_and_cohortNotAssigned_andFeaturePassed_returnsNil() { + internalUserDeciderStore.isInternalUser = true + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = nil + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteReleasable(.subfeature(AutofillSubfeature.credentialsAutofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertNil(cohort) + } + + func testWhenGetCohortIfEnabled_andRemoteReleasable_and_cohortAssignedButNotMatchingEnum_returnsNil() { + internalUserDeciderStore.isInternalUser = true + let subfeature = AutofillSubfeature.credentialsAutofill + experimentManager.cohortToReturn = "some" + let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) + + let flag = FakeExperimentFlag(source: .remoteReleasable(.subfeature(AutofillSubfeature.credentialsAutofill))) + let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) + let cohort = featureFlagger.getCohortIfEnabled(for: flag) + XCTAssertNil(cohort) + } + + // MARK: - Overrides func testWhenFeatureFlaggerIsInitializedWithLocalOverridesAndUserIsNotInternalThenAllFlagsAreCleared() throws { @@ -186,7 +312,7 @@ final class DefaultFeatureFlaggerTests: XCTestCase { localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider()) let internalUserDecider = DefaultInternalUserDecider(store: internalUserDeciderStore) - return DefaultFeatureFlagger(internalUserDecider: internalUserDecider, privacyConfigManager: manager) + return DefaultFeatureFlagger(internalUserDecider: internalUserDecider, privacyConfigManager: manager, experimentManager: experimentManager) } private func createFeatureFlaggerWithLocalOverrides(withMockedConfigData data: Data = DefaultFeatureFlaggerTests.embeddedConfig()) -> DefaultFeatureFlagger { @@ -203,6 +329,7 @@ final class DefaultFeatureFlaggerTests: XCTestCase { internalUserDecider: internalUserDecider, privacyConfigManager: manager, localOverrides: overrides, + experimentManager: nil, for: TestFeatureFlag.self ) } @@ -243,3 +370,26 @@ extension FeatureFlagSource: FeatureFlagDescribing { public var rawValue: String { "rawValue" } public var source: FeatureFlagSource { self } } + +class MockExperimentManager: ExperimentCohortsManaging { + var cohortToReturn: CohortID? + var experiments: BrowserServicesKit.Experiments? = nil + + func resolveCohort(for experiment: BrowserServicesKit.ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? { + return cohortToReturn + } +} + + +struct FakeExperimentFlag: FeatureFlagExperimentDescribing { + typealias Cohort = AutofillCohorts + + var rawValue: String = "fake-experiment" + + var source: FeatureFlagSource +} + +enum AutofillCohorts: String, CohortEnum { + case control + case blue +} diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift similarity index 72% rename from Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift rename to Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift index 0a5114d50..80ffa7594 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift @@ -55,16 +55,16 @@ final class ExperimentCohortsManagerTests: XCTestCase { ) let expectedDate1 = Date() - experimentData1 = ExperimentData(parentID: "TestParent", cohort: cohort1.name, enrollmentDate: expectedDate1) + experimentData1 = ExperimentData(parentID: "TestParent", cohortID: cohort1.name, enrollmentDate: expectedDate1) let expectedDate2 = Date().addingTimeInterval(60) - experimentData2 = ExperimentData(parentID: "TestParent", cohort: cohort2.name, enrollmentDate: expectedDate2) + experimentData2 = ExperimentData(parentID: "TestParent", cohortID: cohort2.name, enrollmentDate: expectedDate2) let expectedDate3 = Date() - experimentData3 = ExperimentData(parentID: "TestParent", cohort: cohort3.name, enrollmentDate: expectedDate3) + experimentData3 = ExperimentData(parentID: "TestParent", cohortID: cohort3.name, enrollmentDate: expectedDate3) let expectedDate4 = Date().addingTimeInterval(60) - experimentData4 = ExperimentData(parentID: "TestParent", cohort: cohort4.name, enrollmentDate: expectedDate4) + experimentData4 = ExperimentData(parentID: "TestParent", cohortID: cohort4.name, enrollmentDate: expectedDate4) } override func tearDown() { @@ -94,12 +94,12 @@ final class ExperimentCohortsManagerTests: XCTestCase { mockStore.experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] // WHEN - let result1 = experimentCohortsManager.cohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort1, cohort2]), assignIfEnabled: false).cohortID - let result2 = experimentCohortsManager.cohort(for: ExperimentSubfeature(parentID: experimentData2.parentID, subfeatureID: subfeatureName2, cohorts: [cohort2, cohort3]), assignIfEnabled: false).cohortID + let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort1, cohort2]), isAssignCohortEnabled: false) + let result2 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData2.parentID, subfeatureID: subfeatureName2, cohorts: [cohort2, cohort3]), isAssignCohortEnabled: false) // THEN - XCTAssertEqual(result1, experimentData1.cohort) - XCTAssertEqual(result2, experimentData2.cohort) + XCTAssertEqual(result1, experimentData1.cohortID) + XCTAssertEqual(result2, experimentData2.cohortID) } func testCohortAssignIfEnabledWhenNoCohortExists() { @@ -109,12 +109,11 @@ final class ExperimentCohortsManagerTests: XCTestCase { let experiment = ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: cohorts) // WHEN - let result = experimentCohortsManager.cohort(for: experiment, assignIfEnabled: true) + let result = experimentCohortsManager.resolveCohort(for: experiment, isAssignCohortEnabled: true) // THEN - XCTAssertNotNil(result.cohortID) - XCTAssertTrue(result.didAttemptAssignment) - XCTAssertEqual(result.cohortID, experimentData1.cohort) + XCTAssertNotNil(result) + XCTAssertEqual(result, experimentData1.cohortID) } func testCohortDoesNotAssignIfAssignIfEnabledIsFalse() { @@ -124,11 +123,10 @@ final class ExperimentCohortsManagerTests: XCTestCase { let experiment = ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: cohorts) // WHEN - let result = experimentCohortsManager.cohort(for: experiment, assignIfEnabled: false) + let result = experimentCohortsManager.resolveCohort(for: experiment, isAssignCohortEnabled: false) // THEN - XCTAssertNil(result.cohortID) - XCTAssertTrue(result.didAttemptAssignment) + XCTAssertNil(result) } func testCohortDoesNotAssignIfAssignIfEnabledIsTrueButNoCohortsAvailable() { @@ -137,11 +135,10 @@ final class ExperimentCohortsManagerTests: XCTestCase { let experiment = ExperimentSubfeature(parentID: "TestParent", subfeatureID: "NonExistentSubfeature", cohorts: []) // WHEN - let result = experimentCohortsManager.cohort(for: experiment, assignIfEnabled: true) + let result = experimentCohortsManager.resolveCohort(for: experiment, isAssignCohortEnabled: true) // THEN - XCTAssertNil(result.cohortID) - XCTAssertTrue(result.didAttemptAssignment) + XCTAssertNil(result) } func testCohortReassignsCohortIfAssignedCohortDoesNotExistAndAssignIfEnabledIsTrue() { @@ -149,10 +146,10 @@ final class ExperimentCohortsManagerTests: XCTestCase { mockStore.experiments = [subfeatureName1: experimentData1] // WHEN - let result1 = experimentCohortsManager.cohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), assignIfEnabled: true).cohortID + let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), isAssignCohortEnabled: true) // THEN - XCTAssertEqual(result1, experimentData3.cohort) + XCTAssertEqual(result1, experimentData3.cohortID) } func testCohortDoesNotReassignsCohortIfAssignedCohortDoesNotExistAndAssignIfEnabledIsTrue() { @@ -160,7 +157,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { mockStore.experiments = [subfeatureName1: experimentData1] // WHEN - let result1 = experimentCohortsManager.cohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), assignIfEnabled: false).cohortID + let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), isAssignCohortEnabled: false) // THEN XCTAssertNil(result1) @@ -180,11 +177,10 @@ final class ExperimentCohortsManagerTests: XCTestCase { ) // WHEN - let result = experimentCohortsManager.cohort(for: experiment, assignIfEnabled: true) + let result = experimentCohortsManager.resolveCohort(for: experiment, isAssignCohortEnabled: true) // THEN - XCTAssertEqual(result.cohortID, experimentData3.cohort) - XCTAssertTrue(result.didAttemptAssignment) + XCTAssertEqual(result, experimentData3.cohortID) } } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentsDataStoreTests.swift similarity index 90% rename from Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift rename to Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentsDataStoreTests.swift index 8816daec2..77da51663 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/ExperimentsDataStoreTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentsDataStoreTests.swift @@ -47,8 +47,8 @@ final class ExperimentsDataStoreTests: XCTestCase { func testExperimentsGetReturnsDecodedExperiments() { // GIVEN - let experimentData1 = ExperimentData(parentID: "parent", cohort: "TestCohort1", enrollmentDate: Date()) - let experimentData2 = ExperimentData(parentID: "parent", cohort: "TestCohort2", enrollmentDate: Date()) + let experimentData1 = ExperimentData(parentID: "parent", cohortID: "TestCohort1", enrollmentDate: Date()) + let experimentData2 = ExperimentData(parentID: "parent", cohortID: "TestCohort2", enrollmentDate: Date()) let experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] let encoder = JSONEncoder() @@ -62,17 +62,17 @@ final class ExperimentsDataStoreTests: XCTestCase { // THEN let timeDifference1 = abs(experimentData1.enrollmentDate.timeIntervalSince(result?[subfeatureName1]?.enrollmentDate ?? Date())) let timeDifference2 = abs(experimentData2.enrollmentDate.timeIntervalSince(result?[subfeatureName2]?.enrollmentDate ?? Date())) - XCTAssertEqual(result?[subfeatureName1]?.cohort, experimentData1.cohort) + XCTAssertEqual(result?[subfeatureName1]?.cohortID, experimentData1.cohortID) XCTAssertLessThanOrEqual(timeDifference1, 1.0) - XCTAssertEqual(result?[subfeatureName2]?.cohort, experimentData2.cohort) + XCTAssertEqual(result?[subfeatureName2]?.cohortID, experimentData2.cohortID) XCTAssertLessThanOrEqual(timeDifference2, 1.0) } func testExperimentsSetEncodesAndStoresData() throws { // GIVEN - let experimentData1 = ExperimentData(parentID: "parent", cohort: "TestCohort1", enrollmentDate: Date()) - let experimentData2 = ExperimentData(parentID: "parent2", cohort: "TestCohort2", enrollmentDate: Date()) + let experimentData1 = ExperimentData(parentID: "parent", cohortID: "TestCohort1", enrollmentDate: Date()) + let experimentData2 = ExperimentData(parentID: "parent2", cohortID: "TestCohort2", enrollmentDate: Date()) let experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] // WHEN @@ -85,10 +85,10 @@ final class ExperimentsDataStoreTests: XCTestCase { let decodedExperiments = try? decoder.decode(Experiments.self, from: storedData) let timeDifference1 = abs(experimentData1.enrollmentDate.timeIntervalSince(decodedExperiments?[subfeatureName1]?.enrollmentDate ?? Date())) let timeDifference2 = abs(experimentData2.enrollmentDate.timeIntervalSince(decodedExperiments?[subfeatureName2]?.enrollmentDate ?? Date())) - XCTAssertEqual(decodedExperiments?[subfeatureName1]?.cohort, experimentData1.cohort) + XCTAssertEqual(decodedExperiments?[subfeatureName1]?.cohortID, experimentData1.cohortID) XCTAssertEqual(decodedExperiments?[subfeatureName1]?.parentID, experimentData1.parentID) XCTAssertLessThanOrEqual(timeDifference1, 1.0) - XCTAssertEqual(decodedExperiments?[subfeatureName2]?.cohort, experimentData2.cohort) + XCTAssertEqual(decodedExperiments?[subfeatureName2]?.cohortID, experimentData2.cohortID) XCTAssertEqual(decodedExperiments?[subfeatureName2]?.parentID, experimentData2.parentID) XCTAssertLessThanOrEqual(timeDifference2, 1.0) } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlagLocalOverridesTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlagLocalOverridesTests.swift index 5e2407ca1..5223ff059 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlagLocalOverridesTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlagLocalOverridesTests.swift @@ -46,7 +46,7 @@ final class FeatureFlagLocalOverridesTests: XCTestCase { let internalUserDecider = DefaultInternalUserDecider(store: internalUserDeciderStore) let privacyConfig = MockPrivacyConfiguration() let privacyConfigManager = MockPrivacyConfigurationManager(privacyConfig: privacyConfig, internalUserDecider: internalUserDecider) - featureFlagger = DefaultFeatureFlagger(internalUserDecider: internalUserDecider, privacyConfigManager: privacyConfigManager) + featureFlagger = DefaultFeatureFlagger(internalUserDecider: internalUserDecider, privacyConfigManager: privacyConfigManager, experimentManager: nil) keyValueStore = MockKeyValueStore() actionHandler = CapturingFeatureFlagLocalOverridesHandler() diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift similarity index 85% rename from Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift rename to Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift index 8a4591fe7..c2950d55a 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AddPrivacyConfigurationExperimentTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift @@ -1,5 +1,5 @@ // -// AddPrivacyConfigurationExperimentTests.swift +// FeatureFlaggerExperimentsTests.swift // // Copyright © 2024 DuckDuckGo. All rights reserved. // @@ -19,7 +19,7 @@ import XCTest @testable import BrowserServicesKit -final class AddPrivacyConfigurationExperimentTests: XCTestCase { +final class FeatureFlaggerExperimentsTests: XCTestCase { var featureJson: Data = "{}".data(using: .utf8)! var mockEmbeddedData: MockEmbeddedDataProvider! @@ -27,6 +27,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { var experimentManager: ExperimentCohortsManager! var manager: PrivacyConfigurationManager! var locale: Locale! + var featureFlagger: FeatureFlagger! let subfeatureName = "credentialsSaving" @@ -43,6 +44,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), locale: locale, experimentCohortManager: experimentManager) + featureFlagger = DefaultFeatureFlagger(internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), privacyConfigManager: manager, experimentManager: experimentManager) } override func tearDown() { @@ -84,19 +86,18 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { manager.reload(etag: "", data: featureJson) let config = manager.privacyConfig - // we haven't called isEnabled yet, so cohorts should not be yet assigned + // we haven't called getCohortIfEnabled yet, so cohorts should not be yet assigned XCTAssertNil(mockStore.experiments) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) - // we call isEnabled() without cohort, cohort should not be assigned either + // we call isSubfeatureEnabled() hould not be assigned either XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) XCTAssertNil(mockStore.experiments) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) - // we call isEnabled(cohort), then we should assign cohort - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) - XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, cohortID: "blue"), .disabled(.experimentCohortDoesNotMatch)) + // we call getCohortIfEnabled(cohort), then we should assign cohort + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") } @@ -132,8 +133,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { manager.reload(etag: "", data: featureJson) var config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -164,8 +164,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { manager.reload(etag: "", data: featureJson) config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -190,8 +189,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { manager.reload(etag: "", data: featureJson) config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -226,10 +224,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "2", data: featureJson) - var config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -262,19 +258,15 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "2", data: featureJson) - config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "red")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "red") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "red") } func testDisablingFeatureDisablesCohort() { // Initially subfeature for both cohorts is disabled - var config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertNil(mockStore.experiments) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) @@ -307,10 +299,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -343,10 +333,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -379,10 +367,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") } @@ -423,32 +409,24 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { """.data(using: .utf8)! } manager.reload(etag: "", data: featureJson(country: "FR", language: "fr")) - var config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertNil(mockStore.experiments) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) manager.reload(etag: "", data: featureJson(country: "US", language: "en")) - config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertNil(mockStore.experiments) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) manager.reload(etag: "", data: featureJson(country: "US", language: "fr")) - config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") // once cohort is assigned, changing targets shall not affect feature state manager.reload(etag: "", data: featureJson(country: "IT", language: "it")) - config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -475,18 +453,14 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson2) - config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) // re-populate experiment to re-assign new cohort, should not be assigned as it has wrong targets manager.reload(etag: "", data: featureJson(country: "IT", language: "it")) - config = manager.privacyConfig - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) } @@ -525,10 +499,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - var config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -566,10 +538,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -607,10 +577,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -652,10 +620,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -701,7 +667,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { XCTAssertTrue(mockStore.experiments?.isEmpty ?? true) XCTAssertNil(experimentManager.cohort(for: subfeatureName), "control") - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") let currentTime = Date().timeIntervalSince1970 let enrollmentTime = mockStore.experiments?[subfeatureName]?.enrollmentDate.timeIntervalSince1970 @@ -757,8 +723,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { clearRolloutData(feature: "autofill", subFeature: "credentialsSaving") XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -805,8 +770,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -849,8 +813,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { var config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -891,8 +854,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -929,13 +891,12 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "blue")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "blue") XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "blue") } - func clearRolloutData(feature: String, subFeature: String) { + private func clearRolloutData(feature: String, subFeature: String) { UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).enabled") UserDefaults().set(nil, forKey: "config.\(feature).\(subFeature).lastRolloutCount") } @@ -976,7 +937,7 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { manager.reload(etag: "foo", data: featureJson) let config = manager.privacyConfig - let activeExperiments = config.getAllActiveExperiments() + let activeExperiments = featureFlagger.getAllActiveExperiments() XCTAssertTrue(activeExperiments.isEmpty) XCTAssertNil(mockStore.experiments) } @@ -1054,21 +1015,21 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "foo", data: featureJson) - var config = manager.privacyConfig - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.inlineIconCredentials, cohortID: "green")) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.accessCredentialManagement, cohortID: "control")) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.inlineIconCredentials), "green") + XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.accessCredentialManagement)) + XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: AutofillSubfeature.credentialsSaving.rawValue), "control") XCTAssertEqual(experimentManager.cohort(for: AutofillSubfeature.inlineIconCredentials.rawValue), "green") XCTAssertNil(experimentManager.cohort(for: AutofillSubfeature.accessCredentialManagement.rawValue)) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) - var activeExperiments = config.getAllActiveExperiments() + var activeExperiments = featureFlagger.getAllActiveExperiments() XCTAssertEqual(activeExperiments.count, 2) - XCTAssertEqual(activeExperiments[AutofillSubfeature.credentialsSaving.rawValue]?.cohort, "control") - XCTAssertEqual(activeExperiments[AutofillSubfeature.inlineIconCredentials.rawValue]?.cohort, "green") + XCTAssertEqual(activeExperiments[AutofillSubfeature.credentialsSaving.rawValue]?.cohortID, "control") + XCTAssertEqual(activeExperiments[AutofillSubfeature.inlineIconCredentials.rawValue]?.cohortID, "green") XCTAssertNil(activeExperiments[AutofillSubfeature.accessCredentialManagement.rawValue]) // When an assigned cohort is removed it's not part of active experiments @@ -1140,12 +1101,11 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "foo", data: featureJson) - config = manager.privacyConfig - activeExperiments = config.getAllActiveExperiments() + activeExperiments = featureFlagger.getAllActiveExperiments() XCTAssertEqual(activeExperiments.count, 1) XCTAssertNil(activeExperiments[AutofillSubfeature.credentialsSaving.rawValue]) - XCTAssertEqual(activeExperiments[AutofillSubfeature.inlineIconCredentials.rawValue]?.cohort, "green") + XCTAssertEqual(activeExperiments[AutofillSubfeature.inlineIconCredentials.rawValue]?.cohortID, "green") XCTAssertNil(activeExperiments[AutofillSubfeature.accessCredentialManagement.rawValue]) // When feature disabled an assigned cohort it's not part of active experiments @@ -1217,9 +1177,8 @@ final class AddPrivacyConfigurationExperimentTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "foo", data: featureJson) - config = manager.privacyConfig - activeExperiments = config.getAllActiveExperiments() + activeExperiments = featureFlagger.getAllActiveExperiments() XCTAssertTrue(activeExperiments.isEmpty) } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/TestFeatureFlag.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/TestFeatureFlag.swift index 8c1a43ee7..8d8ec9929 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/TestFeatureFlag.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/TestFeatureFlag.swift @@ -35,11 +35,11 @@ enum TestFeatureFlag: String, FeatureFlagDescribing { var source: FeatureFlagSource { switch self { case .nonOverridableFlag: - return .internalOnly + return .internalOnly() case .overridableFlagDisabledByDefault: return .disabled case .overridableFlagEnabledByDefault: - return .internalOnly + return .internalOnly() } } } diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift index e3be10a1f..9388b14e9 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift @@ -586,13 +586,13 @@ class AppPrivacyConfigurationTests: XCTestCase { let config = manager.privacyConfig let oldVersionProvider = MockAppVersionProvider(appVersion: "1.35.0") - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: oldVersionProvider, randomizer: Double.random(in:))) - XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: oldVersionProvider, randomizer: Double.random(in:)), .disabled(.appVersionNotSupported)) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: oldVersionProvider, randomizer: Double.random(in:))) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, versionProvider: oldVersionProvider, randomizer: Double.random(in:)), .disabled(.appVersionNotSupported)) let currentVersionProvider = MockAppVersionProvider(appVersion: "1.36.0") - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: currentVersionProvider, randomizer: Double.random(in:))) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: currentVersionProvider, randomizer: Double.random(in:))) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) let futureVersionProvider = MockAppVersionProvider(appVersion: "2.16.0") - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: futureVersionProvider, randomizer: Double.random(in:))) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: futureVersionProvider, randomizer: Double.random(in:))) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) } @@ -661,12 +661,12 @@ class AppPrivacyConfigurationTests: XCTestCase { let oldVersionProvider = MockAppVersionProvider(appVersion: "1.35.0") XCTAssertFalse(config.isEnabled(featureKey: .autofill, versionProvider: oldVersionProvider)) - XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: oldVersionProvider, randomizer: Double.random(in:))) - XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: oldVersionProvider, randomizer: Double.random(in:)), .disabled(.appVersionNotSupported)) + XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: oldVersionProvider, randomizer: Double.random(in:))) + XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving, versionProvider: oldVersionProvider, randomizer: Double.random(in:)), .disabled(.appVersionNotSupported)) let currentVersionProvider = MockAppVersionProvider(appVersion: "1.36.0") XCTAssertTrue(config.isEnabled(featureKey: .autofill, versionProvider: currentVersionProvider)) - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: nil, versionProvider: currentVersionProvider, randomizer: Double.random(in:))) + XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, versionProvider: currentVersionProvider, randomizer: Double.random(in:))) XCTAssertEqual(config.stateFor(AutofillSubfeature.credentialsSaving), .enabled) } diff --git a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift index 0334f4f7a..388ad69b9 100644 --- a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift +++ b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift @@ -221,17 +221,29 @@ class MockPrivacyConfiguration: PrivacyConfiguration { var isSubfeatureEnabledCheck: ((any PrivacySubfeature) -> Bool)? - func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { + func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { isSubfeatureEnabledCheck?(subfeature) ?? false } - func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { + func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { if isSubfeatureEnabledCheck?(subfeature) == true { return .enabled } return .disabled(.disabledInConfig) } + func stateFor(subfeatureID: BrowserServicesKit.SubfeatureID, parentFeatureID: BrowserServicesKit.ParentFeatureID, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { + return .enabled + } + + func cohorts(for subfeature: any BrowserServicesKit.PrivacySubfeature) -> [BrowserServicesKit.PrivacyConfigurationData.Cohort]? { + return nil + } + + func cohorts(subfeatureID: BrowserServicesKit.SubfeatureID, parentFeatureID: BrowserServicesKit.ParentFeatureID) -> [BrowserServicesKit.PrivacyConfigurationData.Cohort]? { + return nil + } + var identifier: String = "abcd" var version: String? = "123456789" var userUnprotectedDomains: [String] = [] diff --git a/Tests/DDGSyncTests/Mocks/Mocks.swift b/Tests/DDGSyncTests/Mocks/Mocks.swift index b65b8abdb..4745f155c 100644 --- a/Tests/DDGSyncTests/Mocks/Mocks.swift +++ b/Tests/DDGSyncTests/Mocks/Mocks.swift @@ -169,14 +169,26 @@ class MockPrivacyConfiguration: PrivacyConfiguration { return .enabled } - func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { + func isSubfeatureEnabled(_ subfeature: any BrowserServicesKit.PrivacySubfeature, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> Bool { true } - func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, cohortID: BrowserServicesKit.CohortID?, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { + func stateFor(_ subfeature: any BrowserServicesKit.PrivacySubfeature, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { return .enabled } + func stateFor(subfeatureID: BrowserServicesKit.SubfeatureID, parentFeatureID: BrowserServicesKit.ParentFeatureID, versionProvider: BrowserServicesKit.AppVersionProvider, randomizer: (Range) -> Double) -> BrowserServicesKit.PrivacyConfigurationFeatureState { + return .enabled + } + + func cohorts(for subfeature: any BrowserServicesKit.PrivacySubfeature) -> [BrowserServicesKit.PrivacyConfigurationData.Cohort]? { + return nil + } + + func cohorts(subfeatureID: BrowserServicesKit.SubfeatureID, parentFeatureID: BrowserServicesKit.ParentFeatureID) -> [BrowserServicesKit.PrivacyConfigurationData.Cohort]? { + return nil + } + var identifier: String = "abcd" var version: String? = "123456789" var userUnprotectedDomains: [String] = [] From cfe3b130fd018f9ad3be901f89eb07f6b9a88a4c Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Wed, 27 Nov 2024 23:48:28 +0100 Subject: [PATCH 22/49] clean up --- .../ExperimentCohortsManager.swift | 2 -- .../FeatureFlagger/FeatureFlagger.swift | 29 +++++++++++++++---- .../AppPrivacyConfiguration.swift | 1 - .../Features/PrivacyFeature.swift | 2 +- .../PrivacyConfig/PrivacyConfiguration.swift | 3 ++ .../PrivacyConfigurationManager.swift | 3 -- .../ContentBlocker/WebViewTestHelper.swift | 6 ++-- .../DefaultFeatureFlaggerTests.swift | 4 +-- .../FeatureFlaggerExperimentsTests.swift | 4 +-- 9 files changed, 33 insertions(+), 21 deletions(-) diff --git a/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift index 7231b98bf..86a196334 100644 --- a/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift @@ -66,8 +66,6 @@ public protocol ExperimentCohortsManaging { /// - Cohort assignment is probabilistic, determined by the cohort weights. /// func resolveCohort(for experiment: ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? - - } public class ExperimentCohortsManager: ExperimentCohortsManaging { diff --git a/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift b/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift index 191ec0063..446193b8f 100644 --- a/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift @@ -101,7 +101,27 @@ public protocol FeatureFlagExperimentDescribing { /// ``` var source: FeatureFlagSource { get } - + /// Represents the possible groups or variants within an experiment. + /// + /// The `Cohort` type is used to define user groups or test variations for feature + /// experimentation. Each cohort typically corresponds to a specific behavior or configuration + /// applied to a subset of users. For example, in an A/B test, you might define cohorts such as + /// `control` and `treatment`. + /// + /// Each cohort must conform to the `CohortEnum` protocol, which ensures that the cohort type + /// is an `enum` with `String` raw values and provides access to all possible cases + /// through `CaseIterable`. + /// + /// Example: + /// ``` + /// public enum AutofillCohorts: String, CohortEnum { + /// case control + /// case treatment + /// } + /// ``` + /// + /// The `Cohort` type allows dynamic resolution of cohorts by their raw `String` value, + /// making it easy to map user configurations to specific cohort groups. associatedtype Cohort: CohortEnum } @@ -290,17 +310,16 @@ public class DefaultFeatureFlagger: FeatureFlagger { activeExperiments[subfeatureID] = experimentData } } - return activeExperiments } - + public func getCohortIfEnabled(for featureFlag: Flag) -> (any CohortEnum)? { switch featureFlag.source { case .disabled: return nil case .internalOnly(let cohort): return cohort - case .remoteDevelopment(_) where !internalUserDecider.isInternalUser: + case .remoteDevelopment where !internalUserDecider.isInternalUser: return nil case .remoteReleasable(let featureType), .remoteDevelopment(let featureType) where internalUserDecider.isInternalUser: @@ -321,7 +340,7 @@ public class DefaultFeatureFlagger: FeatureFlagger { let cohorts = config.cohorts(for: subfeature) let experiment = ExperimentSubfeature(parentID: subfeature.parent.rawValue, subfeatureID: subfeature.rawValue, cohorts: cohorts ?? []) switch featureState { - case .enabled: + case .enabled: return experimentManager?.resolveCohort(for: experiment, isAssignCohortEnabled: true) case .disabled(.targetDoesNotMatch): return experimentManager?.resolveCohort(for: experiment, isAssignCohortEnabled: false) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 2904dcf95..2390aff4d 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -330,7 +330,6 @@ public struct AppPrivacyConfiguration: PrivacyConfiguration { public func userDisabledProtection(forDomain domain: String) { locallyUnprotected.disableProtection(forDomain: domain.punycodeEncodedHostname.lowercased()) } - } extension AppPrivacyConfiguration { diff --git a/Sources/BrowserServicesKit/PrivacyConfig/Features/PrivacyFeature.swift b/Sources/BrowserServicesKit/PrivacyConfig/Features/PrivacyFeature.swift index 4eda0d30e..fc79ba107 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/Features/PrivacyFeature.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/Features/PrivacyFeature.swift @@ -19,7 +19,7 @@ import Foundation /// Features whose `rawValue` should be the key to access their corresponding `PrivacyConfigurationData.PrivacyFeature` object -public enum PrivacyFeature: String, CaseIterable { +public enum PrivacyFeature: String { case contentBlocking case duckPlayer case fingerprintingTemporaryStorage diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift index d7a6c9a9e..bd7c39239 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift @@ -105,9 +105,12 @@ public protocol PrivacyConfiguration { /// Adds given domain to locally unprotected list. func userDisabledProtection(forDomain: String) + // APIs used for Exmpriments func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState + func cohorts(for subfeature: any PrivacySubfeature) -> [PrivacyConfigurationData.Cohort]? + func cohorts(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID) -> [PrivacyConfigurationData.Cohort]? } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index 711ead877..004729e70 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -56,7 +56,6 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { private let errorReporting: EventMapping? private let installDate: Date? private let locale: Locale - private let experimentCohortManager: ExperimentCohortsManaging public let internalUserDecider: InternalUserDecider @@ -113,14 +112,12 @@ public class PrivacyConfigurationManager: PrivacyConfigurationManaging { errorReporting: EventMapping? = nil, internalUserDecider: InternalUserDecider, locale: Locale = Locale.current, - experimentCohortManager: ExperimentCohortsManaging = ExperimentCohortsManager(store: ExperimentsDataStore()), installDate: Date? = nil ) { self.embeddedDataProvider = embeddedDataProvider self.localProtection = localProtection self.errorReporting = errorReporting self.internalUserDecider = internalUserDecider - self.experimentCohortManager = experimentCohortManager self.locale = locale self.installDate = installDate diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift index 346542682..d3a2348fc 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift @@ -227,9 +227,9 @@ final class WebKitTestHelper { } class MockExperimentCohortsManager: ExperimentCohortsManaging { - func resolveCohort(for experiment: BrowserServicesKit.ExperimentSubfeature, isAssignCohortEnabled: Bool) -> BrowserServicesKit.CohortID? { + func resolveCohort(for experiment: ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? { return nil } - - var experiments: BrowserServicesKit.Experiments? + + var experiments: Experiments? } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift index 2387a9753..68d738af2 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift @@ -266,7 +266,6 @@ final class DefaultFeatureFlaggerTests: XCTestCase { XCTAssertNil(cohort) } - // MARK: - Overrides func testWhenFeatureFlaggerIsInitializedWithLocalOverridesAndUserIsNotInternalThenAllFlagsAreCleared() throws { @@ -373,14 +372,13 @@ extension FeatureFlagSource: FeatureFlagDescribing { class MockExperimentManager: ExperimentCohortsManaging { var cohortToReturn: CohortID? - var experiments: BrowserServicesKit.Experiments? = nil + var experiments: BrowserServicesKit.Experiments? func resolveCohort(for experiment: BrowserServicesKit.ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? { return cohortToReturn } } - struct FakeExperimentFlag: FeatureFlagExperimentDescribing { typealias Cohort = AutofillCohorts diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift index c2950d55a..121aa2397 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift @@ -42,8 +42,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale, - experimentCohortManager: experimentManager) + locale: locale) featureFlagger = DefaultFeatureFlagger(internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), privacyConfigManager: manager, experimentManager: experimentManager) } @@ -935,7 +934,6 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { } """.data(using: .utf8)! manager.reload(etag: "foo", data: featureJson) - let config = manager.privacyConfig let activeExperiments = featureFlagger.getAllActiveExperiments() XCTAssertTrue(activeExperiments.isEmpty) From 26fa2ed4ae9e55e6336b4753409507c6f1f5038f Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Thu, 28 Nov 2024 01:23:58 +0100 Subject: [PATCH 23/49] fix linting --- .../PrivacyConfig/PrivacyConfiguration.swift | 5 ++--- 1 file changed, 2 insertions(+), 3 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift index bd7c39239..358ce21e9 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift @@ -106,8 +106,7 @@ public protocol PrivacyConfiguration { func userDisabledProtection(forDomain: String) // APIs used for Exmpriments - func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID, versionProvider: AppVersionProvider, - randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState + func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState func cohorts(for subfeature: any PrivacySubfeature) -> [PrivacyConfigurationData.Cohort]? @@ -131,7 +130,7 @@ public extension PrivacyConfiguration { return stateFor(subfeature, versionProvider: AppVersionProvider(), randomizer: randomizer) } - func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID ,randomizer: (Range) -> Double = Double.random(in:)) -> PrivacyConfigurationFeatureState { + func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID, randomizer: (Range) -> Double = Double.random(in:)) -> PrivacyConfigurationFeatureState { return stateFor(subfeatureID: subfeatureID, parentFeatureID: parentFeatureID, versionProvider: AppVersionProvider(), randomizer: randomizer) } From ee2f89103d690025971922de3367efb71bebc178 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Thu, 28 Nov 2024 13:09:57 +0100 Subject: [PATCH 24/49] have only one interface method --- .../FeatureFlagger/FeatureFlagger.swift | 32 ++---- .../DefaultFeatureFlaggerTests.swift | 24 ++-- .../FeatureFlaggerExperimentsTests.swift | 103 +++++++++++++----- 3 files changed, 93 insertions(+), 66 deletions(-) diff --git a/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift b/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift index 446193b8f..b609860ee 100644 --- a/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift @@ -18,7 +18,7 @@ import Foundation -public protocol CohortEnum: RawRepresentable, CaseIterable where RawValue == String {} +public protocol FlagCohort: RawRepresentable, CaseIterable where RawValue == CohortID {} /// This protocol defines a common interface for feature flags managed by FeatureFlagger. /// @@ -122,7 +122,7 @@ public protocol FeatureFlagExperimentDescribing { /// /// The `Cohort` type allows dynamic resolution of cohorts by their raw `String` value, /// making it easy to map user configurations to specific cohort groups. - associatedtype Cohort: CohortEnum + associatedtype CohortType: FlagCohort } public enum FeatureFlagSource { @@ -130,7 +130,7 @@ public enum FeatureFlagSource { case disabled /// Enabled for internal users only. Cannot be toggled remotely - case internalOnly((any CohortEnum)? = nil) + case internalOnly((any FlagCohort)? = nil) /// Toggled remotely using PrivacyConfiguration but only for internal users. Otherwise, disabled. case remoteDevelopment(PrivacyConfigFeatureLevel) @@ -169,24 +169,6 @@ public protocol FeatureFlagger: AnyObject { /// func isFeatureOn(for featureFlag: Flag, allowOverride: Bool) -> Bool - /// Resolves the cohort for a subfeature if the subfeature is enabled. - /// - /// This method checks the state of the subfeature in the `PrivacyConfiguration`. If the subfeature - /// is enabled or disabled due to a target mismatch, it resolves the cohort using the `ExperimentManager`. - /// - /// - Parameter subfeature: A subfeature conforming to `PrivacySubfeature`. - /// - /// - Returns: The `CohortID` associated with the subfeature if enabled, or `nil` otherwise. - /// - /// - Behavior: - /// - If the subfeature state is `.enabled`: - /// - Resolves and assigns a cohort using `resolveCohort(isAssignCohortEnabled: true)`. - /// - If the subfeature state is `.disabled(.targetDoesNotMatch)`: - /// - Resolves the cohort without assigning a new one (`isAssignCohortEnabled: false`). - /// - For other states: Returns `nil`. - /// - func getCohortIfEnabled(_ subfeature: any PrivacySubfeature) -> CohortID? - /// Retrieves the cohort for a feature flag if the feature is enabled. /// /// This method determines the source of the feature flag and evaluates its eligibility based on @@ -205,7 +187,7 @@ public protocol FeatureFlagger: AnyObject { /// - If the feature is a subfeature, resolves its cohort using `getCohortIfEnabled(_ subfeature:)`. /// - Returns `nil` if the user is not eligible. /// - func getCohortIfEnabled(for featureFlag: Flag) -> (any CohortEnum)? + func getCohortIfEnabled(for featureFlag: Flag) -> (any FlagCohort)? /// Retrieves all active experiments currently assigned to the user. /// @@ -313,7 +295,7 @@ public class DefaultFeatureFlagger: FeatureFlagger { return activeExperiments } - public func getCohortIfEnabled(for featureFlag: Flag) -> (any CohortEnum)? { + public func getCohortIfEnabled(for featureFlag: Flag) -> (any FlagCohort)? { switch featureFlag.source { case .disabled: return nil @@ -325,7 +307,7 @@ public class DefaultFeatureFlagger: FeatureFlagger { .remoteDevelopment(let featureType) where internalUserDecider.isInternalUser: if case .subfeature(let subfeature) = featureType { if let resolvedCohortID = getCohortIfEnabled(subfeature) { - return Flag.Cohort.allCases.first { return $0.rawValue == resolvedCohortID } + return Flag.CohortType.allCases.first { return $0.rawValue == resolvedCohortID } } } return nil @@ -334,7 +316,7 @@ public class DefaultFeatureFlagger: FeatureFlagger { } } - public func getCohortIfEnabled(_ subfeature: any PrivacySubfeature) -> CohortID? { + private func getCohortIfEnabled(_ subfeature: any PrivacySubfeature) -> CohortID? { let config = privacyConfigManager.privacyConfig let featureState = config.stateFor(subfeature) let cohorts = config.cohorts(for: subfeature) diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift index 68d738af2..36c7c744e 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift @@ -155,27 +155,27 @@ final class DefaultFeatureFlaggerTests: XCTestCase { func estWhenGetCohortIfEnabled_andSourceInternal_returnsPassedCohort() { let featureFlagger = createFeatureFlagger() - let flag = FakeExperimentFlag(source: .internalOnly(AutofillCohorts.blue)) + let flag = FakeExperimentFlag(source: .internalOnly(AutofillCohort.blue)) let cohort = featureFlagger.getCohortIfEnabled(for: flag) - XCTAssertEqual(cohort?.rawValue, AutofillCohorts.blue.rawValue) + XCTAssertEqual(cohort?.rawValue, AutofillCohort.blue.rawValue) } func testWhenGetCohortIfEnabled_andRemoteInternal_andInternalStateTrue_and_cohortAssigned_returnsAssignedCohort() { internalUserDeciderStore.isInternalUser = true let subfeature = AutofillSubfeature.credentialsAutofill - experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + experimentManager.cohortToReturn = AutofillCohort.control.rawValue let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) let flag = FakeExperimentFlag(source: .remoteDevelopment(.subfeature(AutofillSubfeature.credentialsAutofill))) let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) let cohort = featureFlagger.getCohortIfEnabled(for: flag) - XCTAssertEqual(cohort?.rawValue, AutofillCohorts.control.rawValue) + XCTAssertEqual(cohort?.rawValue, AutofillCohort.control.rawValue) } func testWhenGetCohortIfEnabled_andRemoteInternal_andInternalStateFalse_and_cohortAssigned_returnsNil() { internalUserDeciderStore.isInternalUser = false let subfeature = AutofillSubfeature.credentialsAutofill - experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + experimentManager.cohortToReturn = AutofillCohort.control.rawValue let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) let flag = FakeExperimentFlag(source: .remoteDevelopment(.subfeature(AutofillSubfeature.credentialsAutofill))) @@ -187,7 +187,7 @@ final class DefaultFeatureFlaggerTests: XCTestCase { func testWhenGetCohortIfEnabled_andRemoteInternal_andInternalStateTrue_and_cohortAssigned_andFeaturePassed_returnsNil() { internalUserDeciderStore.isInternalUser = true let subfeature = AutofillSubfeature.credentialsAutofill - experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + experimentManager.cohortToReturn = AutofillCohort.control.rawValue let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) let flag = FakeExperimentFlag(source: .remoteDevelopment(.feature(.autofill))) @@ -222,18 +222,18 @@ final class DefaultFeatureFlaggerTests: XCTestCase { func testWhenGetCohortIfEnabled_andRemoteReleasable_and_cohortAssigned_returnsAssignedCohort() { let subfeature = AutofillSubfeature.credentialsAutofill - experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + experimentManager.cohortToReturn = AutofillCohort.control.rawValue let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) let flag = FakeExperimentFlag(source: .remoteReleasable(.subfeature(AutofillSubfeature.credentialsAutofill))) let featureFlagger = createFeatureFlagger(withMockedConfigData: embeddedData) let cohort = featureFlagger.getCohortIfEnabled(for: flag) - XCTAssertEqual(cohort?.rawValue, AutofillCohorts.control.rawValue) + XCTAssertEqual(cohort?.rawValue, AutofillCohort.control.rawValue) } func testWhenGetCohortIfEnabled_andRemoteReleasable_and_cohortAssigned_andFeaturePassed_returnsNil() { let subfeature = AutofillSubfeature.credentialsAutofill - experimentManager.cohortToReturn = AutofillCohorts.control.rawValue + experimentManager.cohortToReturn = AutofillCohort.control.rawValue let embeddedData = Self.embeddedConfig(autofillSubfeatureForState: (subfeature: subfeature, state: "enabled")) let flag = FakeExperimentFlag(source: .remoteReleasable(.feature(.autofill))) @@ -379,15 +379,15 @@ class MockExperimentManager: ExperimentCohortsManaging { } } -struct FakeExperimentFlag: FeatureFlagExperimentDescribing { - typealias Cohort = AutofillCohorts +fileprivate struct FakeExperimentFlag: FeatureFlagExperimentDescribing { + typealias CohortType = AutofillCohort var rawValue: String = "fake-experiment" var source: FeatureFlagSource } -enum AutofillCohorts: String, CohortEnum { +fileprivate enum AutofillCohort: String, FlagCohort { case control case blue } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift index 121aa2397..b6931ef22 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift @@ -19,6 +19,51 @@ import XCTest @testable import BrowserServicesKit +struct CredentialsSavingFlag: FeatureFlagExperimentDescribing { + + typealias CohortType = Cohort + + var rawValue = "credentialSaving" + + var source: FeatureFlagSource = .remoteReleasable(.subfeature(AutofillSubfeature.credentialsSaving)) + + enum Cohort: String, FlagCohort { + case control + case blue + case red + } +} + +struct InlineIconCredentialsFlag: FeatureFlagExperimentDescribing { + + typealias CohortType = Cohort + + var rawValue = "inlineIconCredentials" + + var source: FeatureFlagSource = .remoteReleasable(.subfeature(AutofillSubfeature.inlineIconCredentials)) + + enum Cohort: String, FlagCohort { + case control + case blue + case green + } +} + +struct AccessCredentialManagementFlag: FeatureFlagExperimentDescribing { + + typealias CohortType = Cohort + + var rawValue = "accessCredentialManagement" + + var source: FeatureFlagSource = .remoteReleasable(.subfeature(AutofillSubfeature.accessCredentialManagement)) + + enum Cohort: String, FlagCohort { + case control + case blue + case green + } +} + final class FeatureFlaggerExperimentsTests: XCTestCase { var featureJson: Data = "{}".data(using: .utf8)! @@ -96,7 +141,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { XCTAssertNil(experimentManager.cohort(for: subfeatureName)) // we call getCohortIfEnabled(cohort), then we should assign cohort - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") } @@ -132,7 +177,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { manager.reload(etag: "", data: featureJson) var config = manager.privacyConfig - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -163,7 +208,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { manager.reload(etag: "", data: featureJson) config = manager.privacyConfig - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -188,7 +233,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { manager.reload(etag: "", data: featureJson) config = manager.privacyConfig - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())) XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -224,7 +269,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "2", data: featureJson) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -258,14 +303,14 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "2", data: featureJson) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "red") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.red.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "red") } func testDisablingFeatureDisablesCohort() { // Initially subfeature for both cohorts is disabled - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())) XCTAssertNil(mockStore.experiments) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) @@ -299,7 +344,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -333,7 +378,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -367,7 +412,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") } @@ -409,23 +454,23 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { } manager.reload(etag: "", data: featureJson(country: "FR", language: "fr")) - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())) XCTAssertNil(mockStore.experiments) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) manager.reload(etag: "", data: featureJson(country: "US", language: "en")) - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())) XCTAssertNil(mockStore.experiments) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) manager.reload(etag: "", data: featureJson(country: "US", language: "fr")) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") // once cohort is assigned, changing targets shall not affect feature state manager.reload(etag: "", data: featureJson(country: "IT", language: "it")) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -453,13 +498,13 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "", data: featureJson2) - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())) XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) // re-populate experiment to re-assign new cohort, should not be assigned as it has wrong targets manager.reload(etag: "", data: featureJson(country: "IT", language: "it")) - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving)) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())) XCTAssertTrue(mockStore.experiments?.isEmpty ?? false) XCTAssertNil(experimentManager.cohort(for: subfeatureName)) } @@ -499,7 +544,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -538,7 +583,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -577,7 +622,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -620,7 +665,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { """.data(using: .utf8)! manager.reload(etag: "", data: featureJson) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -666,7 +711,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { XCTAssertTrue(mockStore.experiments?.isEmpty ?? true) XCTAssertNil(experimentManager.cohort(for: subfeatureName), "control") - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) let currentTime = Date().timeIntervalSince1970 let enrollmentTime = mockStore.experiments?[subfeatureName]?.enrollmentDate.timeIntervalSince1970 @@ -722,7 +767,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { clearRolloutData(feature: "autofill", subFeature: "credentialsSaving") XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -769,7 +814,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -812,7 +857,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { var config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -853,7 +898,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.control.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "control") @@ -890,7 +935,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "blue") + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.CohortType.blue.rawValue) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: subfeatureName), "blue") } @@ -1014,9 +1059,9 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { manager.reload(etag: "foo", data: featureJson) - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.credentialsSaving), "control") - XCTAssertEqual(featureFlagger.getCohortIfEnabled(AutofillSubfeature.inlineIconCredentials), "green") - XCTAssertNil(featureFlagger.getCohortIfEnabled(AutofillSubfeature.accessCredentialManagement)) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: CredentialsSavingFlag())?.rawValue, CredentialsSavingFlag.Cohort.control.rawValue) + XCTAssertEqual(featureFlagger.getCohortIfEnabled(for: InlineIconCredentialsFlag())?.rawValue, InlineIconCredentialsFlag.Cohort.green.rawValue) + XCTAssertNil(featureFlagger.getCohortIfEnabled(for: AccessCredentialManagementFlag())) XCTAssertFalse(mockStore.experiments?.isEmpty ?? true) XCTAssertEqual(experimentManager.cohort(for: AutofillSubfeature.credentialsSaving.rawValue), "control") From b3224fe27f7a46e7c9defa14ad55264e811a4304 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Thu, 28 Nov 2024 13:17:50 +0100 Subject: [PATCH 25/49] fix lint --- .../FeatureFlagging/DefaultFeatureFlaggerTests.swift | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift index 36c7c744e..c8e17ac2d 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift @@ -379,7 +379,7 @@ class MockExperimentManager: ExperimentCohortsManaging { } } -fileprivate struct FakeExperimentFlag: FeatureFlagExperimentDescribing { +private struct FakeExperimentFlag: FeatureFlagExperimentDescribing { typealias CohortType = AutofillCohort var rawValue: String = "fake-experiment" @@ -387,7 +387,7 @@ fileprivate struct FakeExperimentFlag: FeatureFlagExperimentDescribing { var source: FeatureFlagSource } -fileprivate enum AutofillCohort: String, FlagCohort { +private enum AutofillCohort: String, FlagCohort { case control case blue } From b0574316b9b7d318d16ddbc659b1b74659d2efef Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Thu, 28 Nov 2024 22:51:42 +0100 Subject: [PATCH 26/49] address comments --- .../ExperimentCohortsManager.swift | 23 +++++++++---------- .../FeatureFlagger/FeatureFlagger.swift | 10 ++++---- .../PrivacyConfig/PrivacyConfiguration.swift | 2 +- .../ContentBlocker/WebViewTestHelper.swift | 2 +- .../DefaultFeatureFlaggerTests.swift | 2 +- .../ExperimentCohortsManagerTests.swift | 16 ++++++------- .../Resources/privacy-reference-tests | 2 +- 7 files changed, 27 insertions(+), 30 deletions(-) diff --git a/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift index 86a196334..8129cc0a6 100644 --- a/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift @@ -29,8 +29,8 @@ public typealias SubfeatureID = String public typealias ParentFeatureID = String public struct ExperimentData: Codable, Equatable { - public let parentID: String - public let cohortID: String + public let parentID: ParentFeatureID + public let cohortID: CohortID public let enrollmentDate: Date } @@ -46,26 +46,26 @@ public protocol ExperimentCohortsManaging { /// for the specified experiment. If the assigned cohort is valid (i.e., it matches /// one of the experiment's defined cohorts), the method returns the assigned cohort. /// Otherwise, the invalid cohort is removed, and a new cohort is assigned if - /// `isAssignCohortEnabled` is `true`. + /// `allowCohortReassignment` is `true`. /// /// - Parameters: /// - experiment: The `ExperimentSubfeature` representing the experiment and its associated cohorts. - /// - isAssignCohortEnabled: A Boolean value indicating whether cohort assignment is allowed + /// - allowCohortReassignment: A Boolean value indicating whether cohort assignment is allowed /// if the user is not already assigned to a valid cohort. /// /// - Returns: The valid `CohortID` assigned to the user for the experiment, or `nil` - /// if no valid cohort exists and `isAssignCohortEnabled` is `false`. + /// if no valid cohort exists and `allowCohortReassignment` is `false`. /// /// - Behavior: /// 1. Retrieves the currently assigned cohort for the experiment using the `subfeatureID`. /// 2. Validates if the assigned cohort exists within the experiment's cohort list: /// - If valid, the assigned cohort is returned. /// - If invalid, the cohort is removed from storage. - /// 3. If cohort assignment is enabled (`isAssignCohortEnabled` is `true`), a new cohort + /// 3. If cohort assignment is enabled (`allowCohortReassignment` is `true`), a new cohort /// is assigned based on the experiment's cohort weights and saved in storage. /// - Cohort assignment is probabilistic, determined by the cohort weights. /// - func resolveCohort(for experiment: ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? + func resolveCohort(for experiment: ExperimentSubfeature, allowCohortReassignment: Bool) -> CohortID? } public class ExperimentCohortsManager: ExperimentCohortsManaging { @@ -87,15 +87,14 @@ public class ExperimentCohortsManager: ExperimentCohortsManaging { self.randomizer = randomizer } - public func resolveCohort(for experiment: ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? { + public func resolveCohort(for experiment: ExperimentSubfeature, allowCohortReassignment: Bool) -> CohortID? { queue.sync { let assignedCohort = cohort(for: experiment.subfeatureID) if experiment.cohorts.contains(where: { $0.name == assignedCohort }) { - return (assignedCohort) - } else { - removeCohort(from: experiment.subfeatureID) + return assignedCohort } - return isAssignCohortEnabled ? assignCohort(to: experiment) : nil + removeCohort(from: experiment.subfeatureID) + return allowCohortReassignment ? assignCohort(to: experiment) : nil } } } diff --git a/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift b/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift index b609860ee..3ecb53d1e 100644 --- a/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/FeatureFlagger.swift @@ -278,8 +278,8 @@ public class DefaultFeatureFlagger: FeatureFlagger { } public func getAllActiveExperiments() -> Experiments { + guard let enrolledExperiments = experimentManager?.experiments else { return [:] } var activeExperiments = [String: ExperimentData]() - guard let enrolledExperiments = experimentManager?.experiments else { return activeExperiments } let config = privacyConfigManager.privacyConfig for (subfeatureID, experimentData) in enrolledExperiments { @@ -288,7 +288,7 @@ public class DefaultFeatureFlagger: FeatureFlagger { let cohorts = config.cohorts(subfeatureID: subfeatureID, parentFeatureID: experimentData.parentID) ?? [] let experimentSubfeature = ExperimentSubfeature(parentID: experimentData.parentID, subfeatureID: subfeatureID, cohorts: cohorts) - if experimentManager?.resolveCohort(for: experimentSubfeature, isAssignCohortEnabled: false) == experimentData.cohortID { + if experimentManager?.resolveCohort(for: experimentSubfeature, allowCohortReassignment: false) == experimentData.cohortID { activeExperiments[subfeatureID] = experimentData } } @@ -301,8 +301,6 @@ public class DefaultFeatureFlagger: FeatureFlagger { return nil case .internalOnly(let cohort): return cohort - case .remoteDevelopment where !internalUserDecider.isInternalUser: - return nil case .remoteReleasable(let featureType), .remoteDevelopment(let featureType) where internalUserDecider.isInternalUser: if case .subfeature(let subfeature) = featureType { @@ -323,9 +321,9 @@ public class DefaultFeatureFlagger: FeatureFlagger { let experiment = ExperimentSubfeature(parentID: subfeature.parent.rawValue, subfeatureID: subfeature.rawValue, cohorts: cohorts ?? []) switch featureState { case .enabled: - return experimentManager?.resolveCohort(for: experiment, isAssignCohortEnabled: true) + return experimentManager?.resolveCohort(for: experiment, allowCohortReassignment: true) case .disabled(.targetDoesNotMatch): - return experimentManager?.resolveCohort(for: experiment, isAssignCohortEnabled: false) + return experimentManager?.resolveCohort(for: experiment, allowCohortReassignment: false) default: return nil } diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift index 358ce21e9..8cb98866f 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift @@ -105,7 +105,7 @@ public protocol PrivacyConfiguration { /// Adds given domain to locally unprotected list. func userDisabledProtection(forDomain: String) - // APIs used for Exmpriments + // APIs used for Experiments func stateFor(subfeatureID: SubfeatureID, parentFeatureID: ParentFeatureID, versionProvider: AppVersionProvider, randomizer: (Range) -> Double) -> PrivacyConfigurationFeatureState func cohorts(for subfeature: any PrivacySubfeature) -> [PrivacyConfigurationData.Cohort]? diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift index d3a2348fc..bc979233e 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/WebViewTestHelper.swift @@ -227,7 +227,7 @@ final class WebKitTestHelper { } class MockExperimentCohortsManager: ExperimentCohortsManaging { - func resolveCohort(for experiment: ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? { + func resolveCohort(for experiment: ExperimentSubfeature, allowCohortReassignment: Bool) -> CohortID? { return nil } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift index c8e17ac2d..6f5fffbd4 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift @@ -374,7 +374,7 @@ class MockExperimentManager: ExperimentCohortsManaging { var cohortToReturn: CohortID? var experiments: BrowserServicesKit.Experiments? - func resolveCohort(for experiment: BrowserServicesKit.ExperimentSubfeature, isAssignCohortEnabled: Bool) -> CohortID? { + func resolveCohort(for experiment: BrowserServicesKit.ExperimentSubfeature, allowCohortReassignment: Bool) -> CohortID? { return cohortToReturn } } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift index 80ffa7594..48fc85355 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift @@ -94,8 +94,8 @@ final class ExperimentCohortsManagerTests: XCTestCase { mockStore.experiments = [subfeatureName1: experimentData1, subfeatureName2: experimentData2] // WHEN - let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort1, cohort2]), isAssignCohortEnabled: false) - let result2 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData2.parentID, subfeatureID: subfeatureName2, cohorts: [cohort2, cohort3]), isAssignCohortEnabled: false) + let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort1, cohort2]), allowCohortReassignment: false) + let result2 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData2.parentID, subfeatureID: subfeatureName2, cohorts: [cohort2, cohort3]), allowCohortReassignment: false) // THEN XCTAssertEqual(result1, experimentData1.cohortID) @@ -109,7 +109,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { let experiment = ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: cohorts) // WHEN - let result = experimentCohortsManager.resolveCohort(for: experiment, isAssignCohortEnabled: true) + let result = experimentCohortsManager.resolveCohort(for: experiment, allowCohortReassignment: true) // THEN XCTAssertNotNil(result) @@ -123,7 +123,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { let experiment = ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: cohorts) // WHEN - let result = experimentCohortsManager.resolveCohort(for: experiment, isAssignCohortEnabled: false) + let result = experimentCohortsManager.resolveCohort(for: experiment, allowCohortReassignment: false) // THEN XCTAssertNil(result) @@ -135,7 +135,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { let experiment = ExperimentSubfeature(parentID: "TestParent", subfeatureID: "NonExistentSubfeature", cohorts: []) // WHEN - let result = experimentCohortsManager.resolveCohort(for: experiment, isAssignCohortEnabled: true) + let result = experimentCohortsManager.resolveCohort(for: experiment, allowCohortReassignment: true) // THEN XCTAssertNil(result) @@ -146,7 +146,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { mockStore.experiments = [subfeatureName1: experimentData1] // WHEN - let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), isAssignCohortEnabled: true) + let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), allowCohortReassignment: true) // THEN XCTAssertEqual(result1, experimentData3.cohortID) @@ -157,7 +157,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { mockStore.experiments = [subfeatureName1: experimentData1] // WHEN - let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), isAssignCohortEnabled: false) + let result1 = experimentCohortsManager.resolveCohort(for: ExperimentSubfeature(parentID: experimentData1.parentID, subfeatureID: subfeatureName1, cohorts: [cohort2, cohort3]), allowCohortReassignment: false) // THEN XCTAssertNil(result1) @@ -177,7 +177,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { ) // WHEN - let result = experimentCohortsManager.resolveCohort(for: experiment, isAssignCohortEnabled: true) + let result = experimentCohortsManager.resolveCohort(for: experiment, allowCohortReassignment: true) // THEN XCTAssertEqual(result, experimentData3.cohortID) diff --git a/Tests/BrowserServicesKitTests/Resources/privacy-reference-tests b/Tests/BrowserServicesKitTests/Resources/privacy-reference-tests index 6133e7d9d..a603ff9af 160000 --- a/Tests/BrowserServicesKitTests/Resources/privacy-reference-tests +++ b/Tests/BrowserServicesKitTests/Resources/privacy-reference-tests @@ -1 +1 @@ -Subproject commit 6133e7d9d9cd5f1b925cab1971b4d785dc639df7 +Subproject commit a603ff9af22ca3ff7ce2e7ffbfe18c447d9f23e8 From 4756b60c5eb57880cebf5d70e8ce1e7994191e5b Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 29 Nov 2024 12:12:29 +0100 Subject: [PATCH 27/49] update after merge --- .../ExperimentCohortsManager.swift | 11 +- .../PixelExperimentKit.swift | 47 +- .../Autofill/AutofillTestHelper.swift | 3 +- .../DefaultFeatureFlaggerTests.swift | 6 +- .../FeatureFlaggerExperimentsTests.swift | 40 -- .../FingerprintingReferenceTests.swift | 3 +- .../GPC/GPCReferenceTests.swift | 3 +- .../LinkProtection/AmpMatchingTests.swift | 3 +- .../LinkProtection/URLParameterTests.swift | 3 +- .../AdClickAttributionFeatureTests.swift | 3 +- .../AppPrivacyConfigurationTests.swift | 93 ++-- .../ReferrerTrimmingTests.swift | 3 +- .../HTTPSUpgradeReferenceTests.swift | 3 +- .../DuckPlayerContingencyHandlerTests.swift | 18 +- .../PixelExperimentKitTests.swift | 434 +++++------------- 15 files changed, 196 insertions(+), 477 deletions(-) diff --git a/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift index 8129cc0a6..60a03f8c4 100644 --- a/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift @@ -18,24 +18,23 @@ import Foundation +public typealias CohortID = String +public typealias SubfeatureID = String +public typealias ParentFeatureID = String +public typealias Experiments = [String: ExperimentData] + public struct ExperimentSubfeature { let parentID: ParentFeatureID let subfeatureID: SubfeatureID let cohorts: [PrivacyConfigurationData.Cohort] } -public typealias CohortID = String -public typealias SubfeatureID = String -public typealias ParentFeatureID = String - public struct ExperimentData: Codable, Equatable { public let parentID: ParentFeatureID public let cohortID: CohortID public let enrollmentDate: Date } -public typealias Experiments = [String: ExperimentData] - public protocol ExperimentCohortsManaging { /// Retrieves all the experiments a user is enrolled in var experiments: Experiments? { get } diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 165f0474b..d0fe6367e 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -47,7 +47,7 @@ extension PixelKit { // Static property to hold shared dependencies struct ExperimentConfig { - static var privacyConfigManager: PrivacyConfigurationManager? + static var featureFlagger: FeatureFlagger? static var store: ExperimentActionPixelStore = UserDefaults.standard static var fireFunction: (PixelKitEvent, PixelKit.Frequency, Bool) -> Void = { event, frequency, includeAppVersion in fire(event, frequency: frequency, includeAppVersionParameter: includeAppVersion) @@ -56,13 +56,13 @@ extension PixelKit { // Setup method to initialize dependencies public static func configureExperimentKit( - privacyConfigManager: PrivacyConfigurationManager, + featureFlagger: FeatureFlagger, store: ExperimentActionPixelStore = UserDefaults.standard, fire: @escaping (PixelKitEvent, PixelKit.Frequency, Bool) -> Void = { event, frequency, includeAppVersion in fire(event, frequency: frequency, includeAppVersionParameter: includeAppVersion) } ) { - ExperimentConfig.privacyConfigManager = privacyConfigManager + ExperimentConfig.featureFlagger = featureFlagger ExperimentConfig.store = store ExperimentConfig.fireFunction = fire } @@ -90,11 +90,11 @@ extension PixelKit { /// 3. Tracks actions performed and sends the pixel once the target value is reached (if applicable). public static func fireExperimentPixel(for subfeatureID: SubfeatureID, metric: String, conversionWindowDays: ClosedRange, value: String) { // Check is active experiment for user - guard let privacyConfigManager = ExperimentConfig.privacyConfigManager else { - assertionFailure("PrivacyConfigurationManager is not configured") + guard let featureFlagger = ExperimentConfig.featureFlagger else { + assertionFailure("PixelKit is not configured for experiments") return } - guard let experimentData = privacyConfigManager.privacyConfig.getAllActiveExperiments()[subfeatureID] else { return } + guard let experimentData = featureFlagger.getAllActiveExperiments()[subfeatureID] else { return } Self.fireExperimentPixelForActiveExperiment(subfeatureID, experimentData: experimentData, metric: metric, conversionWindowDays: conversionWindowDays, value: value) } @@ -114,11 +114,11 @@ extension PixelKit { 21: [5...7, 8...15], 30: [5...7, 8...15] ] - guard let privacyConfigManager = ExperimentConfig.privacyConfigManager else { - assertionFailure("PrivacyConfigurationManager is not configured") + guard let featureFlagger = ExperimentConfig.featureFlagger else { + assertionFailure("PixelKit is not configured for experiments") return } - privacyConfigManager.privacyConfig.getAllActiveExperiments().forEach { experiment in + featureFlagger.getAllActiveExperiments().forEach { experiment in fireExperimentPixelsfor( experiment.key, experimentData: experiment.value, @@ -143,11 +143,11 @@ extension PixelKit { 21: [5...7, 8...15], 30: [5...7, 8...15] ] - guard let privacyConfigManager = ExperimentConfig.privacyConfigManager else { - assertionFailure("PrivacyConfigurationManager is not configured") + guard let featureFlagger = ExperimentConfig.featureFlagger else { + assertionFailure("PixelKit is not configured for experiments") return } - privacyConfigManager.privacyConfig.getAllActiveExperiments().forEach { experiment in + featureFlagger.getAllActiveExperiments().forEach { experiment in fireExperimentPixelsfor( experiment.key, experimentData: experiment.value, @@ -178,7 +178,7 @@ extension PixelKit { private static func fireExperimentPixelForActiveExperiment(_ subfeatureID: SubfeatureID, experimentData: ExperimentData,metric: String, conversionWindowDays: ClosedRange, value: String) { // Set parameters, event name, store key - let eventName = "\(Self.Constants.metricsEventPrefix)_\(subfeatureID)_\(experimentData.cohort)" + let eventName = "\(Self.Constants.metricsEventPrefix)_\(subfeatureID)_\(experimentData.cohortID)" let parameters: [String: String] = [ Self.Constants.metricKey: metric, Self.Constants.conversionWindowDaysKey: "\(conversionWindowDays.lowerBound.description)-\(conversionWindowDays.upperBound.description)", @@ -201,11 +201,10 @@ extension PixelKit { // if not increase the count of the action // if value is not a number send the pixel if let numberOfAction = Int(value), numberOfAction > 1 { - let actualActionNumber = ExperimentConfig.store.integer(forKey: eventStoreKey) + let actualActionNumber = ExperimentConfig.store.integer(forKey: eventStoreKey) + 1 + ExperimentConfig.store.set(actualActionNumber, forKey: eventStoreKey) if actualActionNumber >= numberOfAction { ExperimentConfig.fireFunction(event, .uniqueIncludingParameters, false) - } else { - ExperimentConfig.store.set(actualActionNumber + 1, forKey: eventStoreKey) } } else { ExperimentConfig.fireFunction(event, .uniqueIncludingParameters, false) @@ -213,9 +212,19 @@ extension PixelKit { } private static func isUserInConversionWindow(_ conversionWindowDays: ClosedRange, enrollmentDate: Date) -> Bool { - guard let startOfWindowDate = enrollmentDate.addDays(conversionWindowDays.lowerBound) else { return false } - guard let endOfWindowDate = enrollmentDate.addDays(conversionWindowDays.upperBound) else { return false } - return Date() >= startOfWindowDate && Date() <= endOfWindowDate + let calendar = Calendar.current + guard let startOfWindowDate = enrollmentDate.addDays(conversionWindowDays.lowerBound), + let endOfWindowDate = enrollmentDate.addDays(conversionWindowDays.upperBound) else { + return false + } + + // Normalize dates to the start of the day + let normalizedStart = calendar.startOfDay(for: startOfWindowDate) + let normalizedEnd = calendar.startOfDay(for: endOfWindowDate) + let currentDate = calendar.startOfDay(for: Date()) + + // Check if the current date falls within the normalized range + return currentDate >= normalizedStart && currentDate <= normalizedEnd } } diff --git a/Tests/BrowserServicesKitTests/Autofill/AutofillTestHelper.swift b/Tests/BrowserServicesKitTests/Autofill/AutofillTestHelper.swift index fde5bd69a..f0d7dd597 100644 --- a/Tests/BrowserServicesKitTests/Autofill/AutofillTestHelper.swift +++ b/Tests/BrowserServicesKitTests/Autofill/AutofillTestHelper.swift @@ -31,8 +31,7 @@ struct AutofillTestHelper { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) return manager } } diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift index d7c64ef12..e7fbbd457 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift @@ -309,8 +309,7 @@ final class DefaultFeatureFlaggerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let internalUserDecider = DefaultInternalUserDecider(store: internalUserDeciderStore) return DefaultFeatureFlagger(internalUserDecider: internalUserDecider, privacyConfigManager: manager, experimentManager: experimentManager) } @@ -321,8 +320,7 @@ final class DefaultFeatureFlaggerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let internalUserDecider = DefaultInternalUserDecider(store: internalUserDeciderStore) overrides = CapturingFeatureFlagOverriding() diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift index 4d2453a98..b6931ef22 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift @@ -1225,44 +1225,4 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { XCTAssertTrue(activeExperiments.isEmpty) } - func testOnAssignmentReportAssignmentFunctionCalled() { - featureJson = - """ - { - "features": { - "autofill": { - "state": "enabled", - "exceptions": [], - "features": { - "credentialsSaving": { - "state": "enabled", - "minSupportedVersion": 2, - "targets": [ - { - "localeCountry": "US" - } - ], - "cohorts": [ - { - "name": "control", - "weight": 1 - }, - { - "name": "blue", - "weight": 0 - } - ] - } - } - } - } - } - """.data(using: .utf8)! - manager.reload(etag: "foo", data: featureJson) - let config = manager.privacyConfig - - XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, cohortID: "control")) - XCTAssertEqual(self.assignedCohort, "control") - XCTAssertEqual(self.assignedExperiment, "credentialsSaving") - } } diff --git a/Tests/BrowserServicesKitTests/Fingerprinting/FingerprintingReferenceTests.swift b/Tests/BrowserServicesKitTests/Fingerprinting/FingerprintingReferenceTests.swift index c8184981c..e908e471c 100644 --- a/Tests/BrowserServicesKitTests/Fingerprinting/FingerprintingReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/Fingerprinting/FingerprintingReferenceTests.swift @@ -62,8 +62,7 @@ final class FingerprintingReferenceTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) }() override func tearDown() { diff --git a/Tests/BrowserServicesKitTests/GPC/GPCReferenceTests.swift b/Tests/BrowserServicesKitTests/GPC/GPCReferenceTests.swift index 345cc0851..b70afbca3 100644 --- a/Tests/BrowserServicesKitTests/GPC/GPCReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/GPC/GPCReferenceTests.swift @@ -48,8 +48,7 @@ final class GPCReferenceTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) } func testGPCHeader() throws { diff --git a/Tests/BrowserServicesKitTests/LinkProtection/AmpMatchingTests.swift b/Tests/BrowserServicesKitTests/LinkProtection/AmpMatchingTests.swift index 0504a4285..e4a2bb3d9 100644 --- a/Tests/BrowserServicesKitTests/LinkProtection/AmpMatchingTests.swift +++ b/Tests/BrowserServicesKitTests/LinkProtection/AmpMatchingTests.swift @@ -72,8 +72,7 @@ final class AmpMatchingTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) } private var contentBlockingManager: ContentBlockerRulesManager { diff --git a/Tests/BrowserServicesKitTests/LinkProtection/URLParameterTests.swift b/Tests/BrowserServicesKitTests/LinkProtection/URLParameterTests.swift index 3a019a0e7..e6e2a3e73 100644 --- a/Tests/BrowserServicesKitTests/LinkProtection/URLParameterTests.swift +++ b/Tests/BrowserServicesKitTests/LinkProtection/URLParameterTests.swift @@ -59,8 +59,7 @@ final class URLParameterTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) } private lazy var urlParamTestSuite: URLParamRefTests = { diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AdClickAttributionFeatureTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AdClickAttributionFeatureTests.swift index 2f4afcafc..72b44867e 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AdClickAttributionFeatureTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AdClickAttributionFeatureTests.swift @@ -74,8 +74,7 @@ class AdClickAttributionFeatureTests: XCTestCase { fetchedData: nil, embeddedDataProvider: dataProvider, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let feature = AdClickAttributionFeature(with: config) diff --git a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift index 6db0db592..9388b14e9 100644 --- a/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift +++ b/Tests/BrowserServicesKitTests/PrivacyConfig/AppPrivacyConfigurationTests.swift @@ -87,8 +87,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_,_ in }) + internalUserDecider: DefaultInternalUserDecider()) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertEqual(manager.reload(etag: nil, data: nil), PrivacyConfigurationManager.ReloadResult.embedded) @@ -113,8 +112,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: downloadedConfig, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertEqual(manager.fetchedConfigData?.etag, downloadedConfigETag) @@ -137,8 +135,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -163,8 +160,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: corruptedConfig, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -192,8 +188,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -224,8 +219,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: corruptedConfig, embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -247,8 +241,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: corruptedConfig, embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -296,8 +289,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) XCTAssertEqual(manager.embeddedConfigData.etag, mockEmbeddedData.embeddedDataEtag) XCTAssertNil(manager.fetchedConfigData) @@ -354,8 +346,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -428,8 +419,7 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: mockProtectionStore, internalUserDecider: DefaultInternalUserDecider(), - installDate: installDate, - reportExperimentCohortAssignment: {_, _ in}).privacyConfig + installDate: installDate).privacyConfig } func testInstalledDaysCheckReturnsCorrectly() { @@ -512,8 +502,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore)) let config = manager.privacyConfig mockInternalUserStore.isInternalUser = true @@ -557,8 +546,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -576,8 +564,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore)) let config = manager.privacyConfig mockInternalUserStore.isInternalUser = true @@ -594,8 +581,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -634,8 +620,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -670,8 +655,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -725,8 +709,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -801,8 +784,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -837,8 +819,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -865,8 +846,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -881,8 +861,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -896,8 +875,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -912,8 +890,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider()) let config = manager.privacyConfig @@ -963,8 +940,7 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale, - reportExperimentCohortAssignment: {_, _ in}) + locale: locale) let config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -981,8 +957,7 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale, - reportExperimentCohortAssignment: {_, _ in}) + locale: locale) let config = manager.privacyConfig XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -999,8 +974,7 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale, - reportExperimentCohortAssignment: {_, _ in}) + locale: locale) let config = manager.privacyConfig XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -1040,8 +1014,7 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale, - reportExperimentCohortAssignment: {_, _ in}) + locale: locale) let config = manager.privacyConfig XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving)) @@ -1088,8 +1061,7 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale, - reportExperimentCohortAssignment: {_, _ in}) + locale: locale) let config = manager.privacyConfig XCTAssertFalse(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) @@ -1108,8 +1080,7 @@ class AppPrivacyConfigurationTests: XCTestCase { embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - locale: locale, - reportExperimentCohortAssignment: {_, _ in}) + locale: locale) let config = manager.privacyConfig XCTAssertTrue(config.isSubfeatureEnabled(AutofillSubfeature.credentialsSaving, randomizer: mockRandom(in:))) @@ -1197,8 +1168,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore)) let config = manager.privacyConfig XCTAssertFalse(config.trackerAllowlist.entries.isEmpty) @@ -1212,8 +1182,7 @@ class AppPrivacyConfigurationTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore), - reportExperimentCohortAssignment: {_, _ in}) + internalUserDecider: DefaultInternalUserDecider(store: mockInternalUserStore)) let config = manager.privacyConfig XCTAssert(config.trackerAllowlist.entries.isEmpty) diff --git a/Tests/BrowserServicesKitTests/ReferrerTrimming/ReferrerTrimmingTests.swift b/Tests/BrowserServicesKitTests/ReferrerTrimming/ReferrerTrimmingTests.swift index 4f6b43b92..02fffdf3e 100644 --- a/Tests/BrowserServicesKitTests/ReferrerTrimming/ReferrerTrimmingTests.swift +++ b/Tests/BrowserServicesKitTests/ReferrerTrimming/ReferrerTrimmingTests.swift @@ -63,8 +63,7 @@ class ReferrerTrimmingTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) } private var contentBlockingManager: ContentBlockerRulesManager { diff --git a/Tests/BrowserServicesKitTests/SmarterEncryption/HTTPSUpgradeReferenceTests.swift b/Tests/BrowserServicesKitTests/SmarterEncryption/HTTPSUpgradeReferenceTests.swift index d1b7faac5..3fb3bfe77 100644 --- a/Tests/BrowserServicesKitTests/SmarterEncryption/HTTPSUpgradeReferenceTests.swift +++ b/Tests/BrowserServicesKitTests/SmarterEncryption/HTTPSUpgradeReferenceTests.swift @@ -79,8 +79,7 @@ final class HTTPSUpgradeReferenceTests: XCTestCase { fetchedData: nil, embeddedDataProvider: embeddedDataProvider, localProtection: localProtection, - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) } private lazy var httpsUpgradesTestSuite: HTTPSUpgradesRefTests = { diff --git a/Tests/DuckPlayerTests/DuckPlayerContingencyHandlerTests.swift b/Tests/DuckPlayerTests/DuckPlayerContingencyHandlerTests.swift index 2ac1e79b3..15414161a 100644 --- a/Tests/DuckPlayerTests/DuckPlayerContingencyHandlerTests.swift +++ b/Tests/DuckPlayerTests/DuckPlayerContingencyHandlerTests.swift @@ -30,8 +30,7 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: {_, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertTrue(handler.shouldDisplayContingencyMessage) @@ -45,8 +44,7 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertFalse(handler.shouldDisplayContingencyMessage) @@ -60,8 +58,7 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertFalse(handler.shouldDisplayContingencyMessage) @@ -75,8 +72,7 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertFalse(handler.shouldDisplayContingencyMessage) @@ -90,8 +86,7 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertEqual(handler.learnMoreURL, URL(string: MockConfig.learnMoreURL)) @@ -104,8 +99,7 @@ final class DuckPlayerContingencyHandlerTests: XCTestCase { fetchedData: nil, embeddedDataProvider: mockEmbeddedData, localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(), - reportExperimentCohortAssignment: { _, _ in }) + internalUserDecider: DefaultInternalUserDecider()) let handler = DefaultDuckPlayerContingencyHandler(privacyConfigurationManager: manager) XCTAssertNil(handler.learnMoreURL) diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index ca37ec54f..29e011120 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -21,44 +21,33 @@ import XCTest @testable import PixelExperimentKit @testable import BrowserServicesKit import PixelKit +import Combine final class PixelExperimentKitTests: XCTestCase { var featureJson: Data = "{}".data(using: .utf8)! var mockPixelStore: MockExperimentActionPixelStore! - var mockExperimentStore: MockExperimentDataStore! - var mockEmbeddedData: MockEmbeddedDataProvider! - var privacyConfigurationManager: PrivacyConfigurationManager! - var firedEvent: PixelKitEvent? - var firedFrequency: PixelKit.Frequency? - var firedIncludeAppVersion: Bool? + var mockFeatureFlagger: MockFeatureFlagger! + var firedEvent = [PixelKitEvent]() + var firedFrequency = [PixelKit.Frequency]() + var firedIncludeAppVersion = [Bool]() override func setUp() { super.setUp() - mockEmbeddedData = MockEmbeddedDataProvider(data: featureJson, etag: "test") mockPixelStore = MockExperimentActionPixelStore() - mockExperimentStore = MockExperimentDataStore() - privacyConfigurationManager = PrivacyConfigurationManager(fetchedETag: nil, - fetchedData: nil, - embeddedDataProvider: mockEmbeddedData, - localProtection: MockDomainsProtectionStore(), - internalUserDecider: DefaultInternalUserDecider(store: MockInternalUserStoring()), - experimentCohortManager: ExperimentCohortsManager(store: mockExperimentStore), - reportExperimentCohortAssignment: { _, _ in }) - PixelKit.configureExperimentKit(privacyConfigManager: privacyConfigurationManager, store: mockPixelStore, fire: { event, frequency, includeAppVersion in - self.firedEvent = event - self.firedFrequency = frequency - self.firedIncludeAppVersion = includeAppVersion + mockFeatureFlagger = MockFeatureFlagger() + PixelKit.configureExperimentKit(featureFlagger: mockFeatureFlagger, store: mockPixelStore, fire: { event, frequency, includeAppVersion in + self.firedEvent.append(event) + self.firedFrequency.append(frequency) + self.firedIncludeAppVersion.append(includeAppVersion) }) } override func tearDown() { - mockEmbeddedData = nil mockPixelStore = nil - mockExperimentStore = nil - privacyConfigurationManager = nil - firedEvent = nil - firedFrequency = nil - firedIncludeAppVersion = nil + mockFeatureFlagger = nil + firedEvent = [] + firedFrequency = [] + firedIncludeAppVersion = [] } func testFireExperimentEnrollmentPixelSendsExpectedData() { @@ -66,7 +55,7 @@ final class PixelExperimentKitTests: XCTestCase { let subfeatureID = "testSubfeature" let cohort = "A" let enrollmentDate = Date(timeIntervalSince1970: 0) - let experimentData = ExperimentData(parentID: "parent", cohort: cohort, enrollmentDate: enrollmentDate) + let experimentData = ExperimentData(parentID: "parent", cohortID: cohort, enrollmentDate: enrollmentDate) let expectedEventName = "experiment_enroll_\(subfeatureID)_\(cohort)" let expectedParameters = ["enrollmentDate": enrollmentDate.toYYYYMMDDInET()] @@ -74,47 +63,18 @@ final class PixelExperimentKitTests: XCTestCase { PixelKit.fireExperimentEnrollmentPixel(subfeatureID: subfeatureID, experiment: experimentData) // THEN - XCTAssertEqual(firedEvent?.name, expectedEventName) - XCTAssertEqual(firedEvent?.parameters, expectedParameters) - XCTAssertEqual(firedFrequency, .uniqueIncludingParameters) - XCTAssertFalse(firedIncludeAppVersion ?? true) + XCTAssertEqual(firedEvent[0].name, expectedEventName) + XCTAssertEqual(firedEvent[0].parameters, expectedParameters) + XCTAssertEqual(firedFrequency[0], .uniqueIncludingParameters) + XCTAssertFalse(firedIncludeAppVersion[0]) } func testFireExperimentPixel_WithValidExperimentAndConversionWindowAndValueNotNumber() { // GIVEN - featureJson = - """ - { - "features": { - "autofill": { - "state": "enabled", - "exceptions": [], - "features": { - "credentialsSaving": { - "state": "enabled", - "minSupportedVersion": 2, - "cohorts": [ - { - "name": "control", - "weight": 1 - }, - { - "name": "blue", - "weight": 0 - } - ] - } - } - } - } - } - """.data(using: .utf8)! - privacyConfigurationManager.reload(etag: "", data: featureJson) let subfeatureID = "credentialsSaving" let cohort = "control" let enrollmentDate = Date().addingTimeInterval(-3 * 24 * 60 * 60) // 5 days ago - print(enrollmentDate) let conversionWindow = 3...3 let value = "true" let expectedEventName = "experiment_metrics_\(subfeatureID)_\(cohort)" @@ -124,54 +84,24 @@ final class PixelExperimentKitTests: XCTestCase { "value": value, "enrollmentDate": enrollmentDate.toYYYYMMDDInET() ] - let experimentData = ExperimentData(parentID: "autofill", cohort: cohort, enrollmentDate: enrollmentDate) - mockExperimentStore.experiments = [subfeatureID: experimentData] + let experimentData = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate) + mockFeatureFlagger.experiments = [subfeatureID: experimentData] // WHEN PixelKit.fireExperimentPixel(for: subfeatureID, metric: "someMetric", conversionWindowDays: conversionWindow, value: value) // THEN - XCTAssertEqual(firedEvent?.name, expectedEventName) - XCTAssertEqual(firedEvent?.parameters, expectedParameters) - XCTAssertEqual(firedFrequency, .uniqueIncludingParameters) - XCTAssertFalse(firedIncludeAppVersion ?? true) + XCTAssertEqual(firedEvent[0].name, expectedEventName) + XCTAssertEqual(firedEvent[0].parameters, expectedParameters) + XCTAssertEqual(firedFrequency[0], .uniqueIncludingParameters) + XCTAssertFalse(firedIncludeAppVersion[0]) } func testFireExperimentPixel_WithValidExperimentAndConversionWindowAndValue1() { // GIVEN - featureJson = - """ - { - "features": { - "autofill": { - "state": "enabled", - "exceptions": [], - "features": { - "credentialsSaving": { - "state": "enabled", - "minSupportedVersion": 2, - "cohorts": [ - { - "name": "control", - "weight": 1 - }, - { - "name": "blue", - "weight": 0 - } - ] - } - } - } - } - } - """.data(using: .utf8)! - privacyConfigurationManager.reload(etag: "", data: featureJson) - let subfeatureID = "credentialsSaving" let cohort = "control" let enrollmentDate = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago - print(enrollmentDate) let conversionWindow = 3...7 let value = "1" let expectedEventName = "experiment_metrics_\(subfeatureID)_\(cohort)" @@ -181,54 +111,24 @@ final class PixelExperimentKitTests: XCTestCase { "value": value, "enrollmentDate": enrollmentDate.toYYYYMMDDInET() ] - let experimentData = ExperimentData(parentID: "autofill", cohort: cohort, enrollmentDate: enrollmentDate) - mockExperimentStore.experiments = [subfeatureID: experimentData] + let experimentData = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate) + mockFeatureFlagger.experiments = [subfeatureID: experimentData] // WHEN PixelKit.fireExperimentPixel(for: subfeatureID, metric: "someMetric", conversionWindowDays: conversionWindow, value: value) // THEN - XCTAssertEqual(firedEvent?.name, expectedEventName) - XCTAssertEqual(firedEvent?.parameters, expectedParameters) - XCTAssertEqual(firedFrequency, .uniqueIncludingParameters) - XCTAssertFalse(firedIncludeAppVersion ?? true) + XCTAssertEqual(firedEvent[0].name, expectedEventName) + XCTAssertEqual(firedEvent[0].parameters, expectedParameters) + XCTAssertEqual(firedFrequency[0], .uniqueIncludingParameters) + XCTAssertFalse(firedIncludeAppVersion[0]) } func testFireExperimentPixel_WithValidExperimentAndConversionWindowAndValueN() { // GIVEN - featureJson = - """ - { - "features": { - "autofill": { - "state": "enabled", - "exceptions": [], - "features": { - "credentialsSaving": { - "state": "enabled", - "minSupportedVersion": 2, - "cohorts": [ - { - "name": "control", - "weight": 1 - }, - { - "name": "blue", - "weight": 0 - } - ] - } - } - } - } - } - """.data(using: .utf8)! - privacyConfigurationManager.reload(etag: "", data: featureJson) - let subfeatureID = "credentialsSaving" let cohort = "control" let enrollmentDate = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago - print(enrollmentDate) let conversionWindow = 3...7 let randomNumber = Int.random(in: 1...100) let value = "\(randomNumber)" @@ -239,169 +139,79 @@ final class PixelExperimentKitTests: XCTestCase { "value": value, "enrollmentDate": enrollmentDate.toYYYYMMDDInET() ] - let experimentData = ExperimentData(parentID: "autofill", cohort: cohort, enrollmentDate: enrollmentDate) - mockExperimentStore.experiments = [subfeatureID: experimentData] + let experimentData = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate) + mockFeatureFlagger.experiments = [subfeatureID: experimentData] // WHEN calling fire before expected number of calls - for n in 0..(for featureFlag: Flag) -> (any FlagCohort)? where Flag : FeatureFlagExperimentDescribing { + return nil + } + + func getAllActiveExperiments() -> Experiments { + return experiments + } - init(data: Data, etag: String) { - embeddedData = data - embeddedDataEtag = etag + func isFeatureOn(for featureFlag: Flag, allowOverride: Bool) -> Bool where Flag : FeatureFlagDescribing { + return false } } -final class MockDomainsProtectionStore: DomainsProtectionStore { - var unprotectedDomains = Set() +final class MockInternalUserDecider: InternalUserDecider { + var isInternalUser: Bool = false - func disableProtection(forDomain domain: String) { - unprotectedDomains.insert(domain) + var isInternalUserPublisher: AnyPublisher { + Just(false).eraseToAnyPublisher() } - func enableProtection(forDomain domain: String) { - unprotectedDomains.remove(domain) + func markUserAsInternalIfNeeded(forUrl url: URL?, response: HTTPURLResponse?) -> Bool { + return false } } - -final class MockInternalUserStoring: InternalUserStoring { - var isInternalUser: Bool = false -} From d202980fa0ddf983b2d28f47c744d5e141657394 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 29 Nov 2024 15:29:31 +0100 Subject: [PATCH 28/49] cleanup --- .../AppPrivacyConfiguration.swift | 2 +- .../PrivacyConfig/PrivacyConfiguration.swift | 2 - .../PrivacyConfigurationManager.swift | 4 -- .../PrivacyConfigurationManagerMock.swift | 1 - .../UserContentControllerTests.swift | 1 - ...SubscriptionFeatureAvailabilityTests.swift | 1 - Tests/DDGSyncTests/Mocks/Mocks.swift | 1 - .../PixelExperimentKitTests.swift | 62 +++++++++++++++++++ 8 files changed, 63 insertions(+), 11 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index c6aa63a8e..4bab7e698 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -19,7 +19,7 @@ import Foundation import Common -public class AppPrivacyConfiguration: PrivacyConfiguration { +public struct AppPrivacyConfiguration: PrivacyConfiguration { private enum Constants { static let enabledKey = "enabled" diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift index 803e11a0f..8cb98866f 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfiguration.swift @@ -55,8 +55,6 @@ public protocol PrivacyConfiguration { /// Trackers that has been allow listed because of site breakage var trackerAllowlist: PrivacyConfigurationData.TrackerAllowlist { get } - var delegate: PrivacyConfigurationDelegate? { get set } - func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool func stateFor(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> PrivacyConfigurationFeatureState diff --git a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift index 5e50290d8..004729e70 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/PrivacyConfigurationManager.swift @@ -20,10 +20,6 @@ import Foundation import Combine import Common -public protocol PrivacyConfigurationDelegate: AnyObject { - func didAssignCohort(_ cohort: CohortID, to experiment: SubfeatureID) -} - public protocol EmbeddedDataProvider { var embeddedDataEtag: String { get } diff --git a/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift b/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift index 65d5e146f..29adc74fb 100644 --- a/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift +++ b/Tests/BrokenSitePromptTests/PrivacyConfigurationManagerMock.swift @@ -21,7 +21,6 @@ import BrowserServicesKit import Combine class PrivacyConfigurationMock: PrivacyConfiguration { - var delegate: (any BrowserServicesKit.PrivacyConfigurationDelegate)? var identifier: String = "id" var version: String? = "123456789" diff --git a/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift b/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift index 673e9307f..767250149 100644 --- a/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift +++ b/Tests/BrowserServicesKitTests/ContentBlocker/UserContentControllerTests.swift @@ -311,7 +311,6 @@ class PrivacyConfigurationManagerMock: PrivacyConfigurationManaging { } class PrivacyConfigurationMock: PrivacyConfiguration { - var delegate: (any PrivacyConfigurationDelegate)? func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool { true } diff --git a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift index bfdea0e7c..0c2410156 100644 --- a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift +++ b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift @@ -212,7 +212,6 @@ final class SubscriptionFeatureAvailabilityTests: XCTestCase { } class MockPrivacyConfiguration: PrivacyConfiguration { - var delegate: (any PrivacyConfigurationDelegate)? func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool { true } diff --git a/Tests/DDGSyncTests/Mocks/Mocks.swift b/Tests/DDGSyncTests/Mocks/Mocks.swift index 4f47f2442..4745f155c 100644 --- a/Tests/DDGSyncTests/Mocks/Mocks.swift +++ b/Tests/DDGSyncTests/Mocks/Mocks.swift @@ -162,7 +162,6 @@ class MockPrivacyConfigurationManager: PrivacyConfigurationManaging { } class MockPrivacyConfiguration: PrivacyConfiguration { - var delegate: (any PrivacyConfigurationDelegate)? func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool { true } diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index 29e011120..e95706a19 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -294,7 +294,69 @@ final class PixelExperimentKitTests: XCTestCase { $0.name == "experiment_metrics_\(subfeatureID2)_\(cohort2)" } ) + } + + func testFireAppRetentionExperimentPixels_WithMultipleExperiments() { + // GIVEN + let subfeatureID1 = "credentialsSaving" + let cohort1 = "control" + let enrollmentDate1 = Date().addingTimeInterval(-6 * 24 * 60 * 60) // 6 days ago + let experimentData1 = ExperimentData(parentID: "autofill", cohortID: cohort1, enrollmentDate: enrollmentDate1) + + let subfeatureID2 = "inlineIconCredentials" + let cohort2 = "test" + let enrollmentDate2 = Date().addingTimeInterval(-10 * 24 * 60 * 60) // 10 days ago + let experimentData2 = ExperimentData(parentID: "autofill", cohortID: cohort2, enrollmentDate: enrollmentDate2) + + mockFeatureFlagger.experiments = [ + subfeatureID1: experimentData1, + subfeatureID2: experimentData2 + ] + + // WHEN + PixelKit.fireAppRetentionExperimentPixels() + + // THEN + // Verify pixel for the first experiment + XCTAssertTrue( + firedEvent.contains { + $0.name == "experiment_metrics_\(subfeatureID1)_\(cohort1)" + } + ) + XCTAssertTrue( + firedEvent.contains { + $0.parameters?[PixelKit.Constants.metricKey] == PixelKit.Constants.appUseMetricValue + } + ) + XCTAssertTrue( + firedEvent.contains { + $0.parameters?[PixelKit.Constants.conversionWindowDaysKey] == "5-7" + } + ) + XCTAssertTrue( + firedEvent.contains { + $0.parameters?[PixelKit.Constants.conversionWindowDaysKey] == "6-6" + } + ) + + // Verify no pixel fired for the second experiment (outside conversion window) + XCTAssertNotNil(mockPixelStore.store) + XCTAssertFalse( + firedEvent.contains { + $0.name == "experiment_metrics_\(subfeatureID2)_\(cohort2)" + } + ) + + // Verify no pixel fired that after 4 reps second experiment pixel is sent(outside conversion window) + PixelKit.fireAppRetentionExperimentPixels() + PixelKit.fireAppRetentionExperimentPixels() + PixelKit.fireAppRetentionExperimentPixels() + XCTAssertTrue( + firedEvent.contains { + $0.name == "experiment_metrics_\(subfeatureID2)_\(cohort2)" + } + ) } } From bfb7a5e5429608876379a67066ba9ef0c5a48884 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 29 Nov 2024 15:34:38 +0100 Subject: [PATCH 29/49] fix lint errors --- .../PrivacyConfig/AppPrivacyConfiguration.swift | 2 +- .../FeatureFlagging/DefaultFeatureFlaggerTests.swift | 2 +- .../Subscription/SubscriptionFeatureAvailabilityTests.swift | 2 +- Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift | 6 +++--- 4 files changed, 6 insertions(+), 6 deletions(-) diff --git a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift index 4bab7e698..2390aff4d 100644 --- a/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift +++ b/Sources/BrowserServicesKit/PrivacyConfig/AppPrivacyConfiguration.swift @@ -19,7 +19,7 @@ import Foundation import Common -public struct AppPrivacyConfiguration: PrivacyConfiguration { +public struct AppPrivacyConfiguration: PrivacyConfiguration { private enum Constants { static let enabledKey = "enabled" diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift index e7fbbd457..5687c5267 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/DefaultFeatureFlaggerTests.swift @@ -302,7 +302,7 @@ final class DefaultFeatureFlaggerTests: XCTestCase { } // MARK: - Helpers - + private func createFeatureFlagger(withMockedConfigData data: Data = DefaultFeatureFlaggerTests.embeddedConfig()) -> DefaultFeatureFlagger { let mockEmbeddedData = MockEmbeddedDataProvider(data: data, etag: "embeddedConfigETag") let manager = PrivacyConfigurationManager(fetchedETag: nil, diff --git a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift index 0c2410156..388ad69b9 100644 --- a/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift +++ b/Tests/BrowserServicesKitTests/Subscription/SubscriptionFeatureAvailabilityTests.swift @@ -212,7 +212,7 @@ final class SubscriptionFeatureAvailabilityTests: XCTestCase { } class MockPrivacyConfiguration: PrivacyConfiguration { - + func isEnabled(featureKey: PrivacyFeature, versionProvider: AppVersionProvider) -> Bool { true } func stateFor(featureKey: BrowserServicesKit.PrivacyFeature, versionProvider: BrowserServicesKit.AppVersionProvider) -> BrowserServicesKit.PrivacyConfigurationFeatureState { diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index e95706a19..40992caf8 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -221,7 +221,7 @@ final class PixelExperimentKitTests: XCTestCase { ] let eventStoreKey = expectedEventName + "_" + expectedParameters.escapedString() print(eventStoreKey) - mockPixelStore.store = [eventStoreKey : 2] + mockPixelStore.store = [eventStoreKey: 2] // WHEN PixelKit.fireExperimentPixel(for: subfeatureID, metric: "someMetric", conversionWindowDays: conversionWindow, value: value) @@ -385,7 +385,7 @@ class MockFeatureFlagger: FeatureFlagger { var localOverrides: (any BrowserServicesKit.FeatureFlagLocalOverriding)? - func getCohortIfEnabled(for featureFlag: Flag) -> (any FlagCohort)? where Flag : FeatureFlagExperimentDescribing { + func getCohortIfEnabled(for featureFlag: Flag) -> (any FlagCohort)? where Flag: FeatureFlagExperimentDescribing { return nil } @@ -393,7 +393,7 @@ class MockFeatureFlagger: FeatureFlagger { return experiments } - func isFeatureOn(for featureFlag: Flag, allowOverride: Bool) -> Bool where Flag : FeatureFlagDescribing { + func isFeatureOn(for featureFlag: Flag, allowOverride: Bool) -> Bool where Flag: FeatureFlagDescribing { return false } } From a98e9c4be66809c51593c3469f04ef2eaa4a8977 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 29 Nov 2024 16:00:27 +0100 Subject: [PATCH 30/49] add pixel unique parameter test --- .../PixelExperimentKit.swift | 16 ++---- .../Extensions/Disctionary+PixelKit.swift | 26 ++++++++++ Sources/PixelKit/PixelKit.swift | 10 ++-- .../XCTestCase+PixelKit.swift | 2 +- .../PixelExperimentKitTests.swift | 10 ++-- Tests/PixelKitTests/PixelKitTests.swift | 50 +++++++++++++++++++ 6 files changed, 91 insertions(+), 23 deletions(-) create mode 100644 Sources/PixelKit/Extensions/Disctionary+PixelKit.swift diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index d0fe6367e..ce3fbe809 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -74,7 +74,7 @@ extension PixelKit { public static func fireExperimentEnrollmentPixel(subfeatureID: SubfeatureID, experiment: ExperimentData) { let eventName = "\(Self.Constants.enrollmentEventPrefix)_\(subfeatureID)_\(experiment.cohortID)" let event = ExperimentEvent(name: eventName, parameters: [Self.Constants.enrollmentDateKey: experiment.enrollmentDate.toYYYYMMDDInET()]) - ExperimentConfig.fireFunction(event, .uniqueIncludingParameters, false) + ExperimentConfig.fireFunction(event, .uniqueByNameAndParameters, false) } /// Fires a pixel for a specific action in an experiment, based on conversion window and value thresholds (if value is a number). @@ -186,7 +186,7 @@ extension PixelKit { Self.Constants.enrollmentDateKey: experimentData.enrollmentDate.toYYYYMMDDInET() ] let event = ExperimentEvent(name: eventName, parameters: parameters) - let eventStoreKey = eventName + "_" + parameters.escapedString() + let eventStoreKey = eventName + "_" + parameters.description // Check if user is in conversion window // if not don't send pixel and remove related action from the store @@ -204,10 +204,10 @@ extension PixelKit { let actualActionNumber = ExperimentConfig.store.integer(forKey: eventStoreKey) + 1 ExperimentConfig.store.set(actualActionNumber, forKey: eventStoreKey) if actualActionNumber >= numberOfAction { - ExperimentConfig.fireFunction(event, .uniqueIncludingParameters, false) + ExperimentConfig.fireFunction(event, .uniqueByNameAndParameters, false) } } else { - ExperimentConfig.fireFunction(event, .uniqueIncludingParameters, false) + ExperimentConfig.fireFunction(event, .uniqueByNameAndParameters, false) } } @@ -228,14 +228,6 @@ extension PixelKit { } } -extension Dictionary where Key: Comparable { - func escapedString(pairSeparator: String = ":", entrySeparator: String = ",") -> String { - return self.sorted { $0.key < $1.key } - .map { "\("\($0.key)".replacingOccurrences(of: entrySeparator, with: "\\\(entrySeparator)"))\(pairSeparator)\("\($0.value)".replacingOccurrences(of: entrySeparator, with: "\\\(entrySeparator)"))" } - .joined(separator: entrySeparator) - } -} - extension UserDefaults: ExperimentActionPixelStore {} extension Date { diff --git a/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift b/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift new file mode 100644 index 000000000..d98ea2daf --- /dev/null +++ b/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift @@ -0,0 +1,26 @@ +// +// Disctionary+PixelKit.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +public extension Dictionary where Key: Comparable { + func toString(pairSeparator: String = ":", entrySeparator: String = ",") -> String { + return self.sorted(by: { $0.key < $1.key }) + .map { "\($0.key)\(pairSeparator)\($0.value)" } + .joined(separator: entrySeparator) + } +} diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index a3b027e5e..b342996d3 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -36,7 +36,7 @@ public final class PixelKit { case unique /// Sent only once ever (based on pixel name AND parameters). The timestamp for this pixel is stored. - case uniqueIncludingParameters + case uniqueByNameAndParameters /// [Legacy] Used in Pixel.fire(...) as .daily but without the `_d` automatically added to the name case legacyDaily @@ -70,8 +70,8 @@ public final class PixelKit { "Legacy Daily and Count" case .dailyAndCount: "Daily and Count" - case .uniqueIncludingParameters: - "Unique Including Parameters" + case .uniqueByNameAndParameters: + "Unique By Name And Parameters" } } } @@ -227,8 +227,8 @@ public final class PixelKit { } else { printDebugInfo(pixelName: pixelName, frequency: frequency, parameters: newParams, skipped: true) } - case .uniqueIncludingParameters: - let pixelNameAndParams = pixelName + newParams.description + case .uniqueByNameAndParameters: + let pixelNameAndParams = pixelName + newParams.toString() if !pixelHasBeenFiredEver(pixelNameAndParams) { fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) updatePixelLastFireDate(pixelName: pixelNameAndParams) diff --git a/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift b/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift index bdb5bb2cc..6b7ad1a6c 100644 --- a/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift +++ b/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift @@ -161,7 +161,7 @@ public extension XCTestCase { case .dailyAndCount: expectedPixelNames.append(originalName.appending("_daily")) expectedPixelNames.append(originalName.appending("_count")) - case .uniqueIncludingParameters: + case .uniqueByNameAndParameters: expectedPixelNames.append(originalName) } return expectedPixelNames diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index 40992caf8..b420b2a9e 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -65,7 +65,7 @@ final class PixelExperimentKitTests: XCTestCase { // THEN XCTAssertEqual(firedEvent[0].name, expectedEventName) XCTAssertEqual(firedEvent[0].parameters, expectedParameters) - XCTAssertEqual(firedFrequency[0], .uniqueIncludingParameters) + XCTAssertEqual(firedFrequency[0], .uniqueByNameAndParameters) XCTAssertFalse(firedIncludeAppVersion[0]) } @@ -93,7 +93,7 @@ final class PixelExperimentKitTests: XCTestCase { // THEN XCTAssertEqual(firedEvent[0].name, expectedEventName) XCTAssertEqual(firedEvent[0].parameters, expectedParameters) - XCTAssertEqual(firedFrequency[0], .uniqueIncludingParameters) + XCTAssertEqual(firedFrequency[0], .uniqueByNameAndParameters) XCTAssertFalse(firedIncludeAppVersion[0]) } @@ -120,7 +120,7 @@ final class PixelExperimentKitTests: XCTestCase { // THEN XCTAssertEqual(firedEvent[0].name, expectedEventName) XCTAssertEqual(firedEvent[0].parameters, expectedParameters) - XCTAssertEqual(firedFrequency[0], .uniqueIncludingParameters) + XCTAssertEqual(firedFrequency[0], .uniqueByNameAndParameters) XCTAssertFalse(firedIncludeAppVersion[0]) } @@ -159,7 +159,7 @@ final class PixelExperimentKitTests: XCTestCase { // THEN XCTAssertEqual(firedEvent[0].name, expectedEventName) XCTAssertEqual(firedEvent[0].parameters, expectedParameters) - XCTAssertEqual(firedFrequency[0], .uniqueIncludingParameters) + XCTAssertEqual(firedFrequency[0], .uniqueByNameAndParameters) XCTAssertFalse(firedIncludeAppVersion[0]) } @@ -219,7 +219,7 @@ final class PixelExperimentKitTests: XCTestCase { "value": value, "enrollmentDate": enrollmentDate.toYYYYMMDDInET() ] - let eventStoreKey = expectedEventName + "_" + expectedParameters.escapedString() + let eventStoreKey = expectedEventName + "_" + expectedParameters.toString() print(eventStoreKey) mockPixelStore.store = [eventStoreKey: 2] diff --git a/Tests/PixelKitTests/PixelKitTests.swift b/Tests/PixelKitTests/PixelKitTests.swift index 2a6b98acc..50e1f2d9a 100644 --- a/Tests/PixelKitTests/PixelKitTests.swift +++ b/Tests/PixelKitTests/PixelKitTests.swift @@ -415,6 +415,56 @@ final class PixelKitTests: XCTestCase { wait(for: [fireCallbackCalled], timeout: 0.5) } + func testUniqueNyNameAndParameterPixel() { + // Prepare test parameters + let appVersion = "1.0.5" + let headers = ["a": "2", "b": "3", "c": "2000"] + let event = TestEventV2.uniqueEvent + let userDefaults = userDefaults() + + let timeMachine = TimeMachine() + + // Set expectations + let fireCallbackCalled = expectation(description: "Expect the pixel firing callback to be called") + fireCallbackCalled.expectedFulfillmentCount = 3 + fireCallbackCalled.assertForOverFulfill = true + + let pixelKit = PixelKit(dryRun: false, + appVersion: appVersion, + defaultHeaders: headers, + dailyPixelCalendar: nil, + dateGenerator: timeMachine.now, + defaults: userDefaults) { _, _, _, _, _, _ in + fireCallbackCalled.fulfill() + } + + // Run test + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["a": "100"]) // Fired + timeMachine.travel(by: .hour, value: 2) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["b": "200"]) // Fired + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["a": "100"]) // Skipped (already fired) + + timeMachine.travel(by: .day, value: 1) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["a": "100", "c": "300"]) // Fired + timeMachine.travel(by: .hour, value: 2) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["a": "100"]) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["b": "200"]) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["c": "300", "a": "100"]) // Skipped (already fired) + + timeMachine.travel(by: .hour, value: 10) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["a": "100"]) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["b": "200"]) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["a": "100", "c": "300"]) // Skipped (already fired) + + timeMachine.travel(by: .day, value: 1) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["a": "100"]) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["b": "200"]) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByNameAndParameters, withAdditionalParameters: ["a": "100", "c": "300"]) // Skipped (already fired) + + // Wait for expectations to be fulfilled + wait(for: [fireCallbackCalled], timeout: 0.5) + } + func testVPNCohort() { XCTAssertEqual(PixelKit.cohort(from: nil), "") assertCohortEqual(.init(year: 2023, month: 1, day: 1), reportAs: "week-1") From 7747bdc811db80b66587b7af929d5b4dce8fd960 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 29 Nov 2024 16:14:16 +0100 Subject: [PATCH 31/49] remove unwanted code --- .../PixelExperimentKit.swift | 3 ++- Sources/PixelKit/PixelKitEventV2.swift | 22 ------------------- .../PixelExperimentKitTests.swift | 1 + 3 files changed, 3 insertions(+), 23 deletions(-) diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index ce3fbe809..071b0fe24 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -186,11 +186,12 @@ extension PixelKit { Self.Constants.enrollmentDateKey: experimentData.enrollmentDate.toYYYYMMDDInET() ] let event = ExperimentEvent(name: eventName, parameters: parameters) - let eventStoreKey = eventName + "_" + parameters.description + let eventStoreKey = eventName + "_" + parameters.toString() // Check if user is in conversion window // if not don't send pixel and remove related action from the store guard isUserInConversionWindow(conversionWindowDays, enrollmentDate: experimentData.enrollmentDate) else { + print(eventStoreKey) ExperimentConfig.store.removeObject(forKey: eventStoreKey) return } diff --git a/Sources/PixelKit/PixelKitEventV2.swift b/Sources/PixelKit/PixelKitEventV2.swift index b0070b972..dc641454c 100644 --- a/Sources/PixelKit/PixelKitEventV2.swift +++ b/Sources/PixelKit/PixelKitEventV2.swift @@ -43,28 +43,6 @@ public protocol PixelFiring { func fire(_ event: PixelKitEventV2, frequency: PixelKit.Frequency) - - static func fire(_ event: PixelKitEvent, - frequency: PixelKit.Frequency, - withHeaders headers: [String: String], - withAdditionalParameters parameters: [String: String]?, - withError error: Error?, - allowedQueryReservedCharacters: CharacterSet?, - includeAppVersionParameter: Bool, - onComplete: (Bool, Error?) -> Void) -} - -extension PixelFiring { - public static func fire(_ event: PixelKitEvent, - frequency: PixelKit.Frequency = .standard, - withHeaders headers: [String: String] = [:], - withAdditionalParameters parameters: [String: String]? = nil, - withError error: Error? = nil, - allowedQueryReservedCharacters: CharacterSet? = nil, - includeAppVersionParameter: Bool = true, - onComplete: (Bool, Error?) -> Void = { _, _ in }) { - return fire(event, frequency: frequency, withHeaders: headers, withAdditionalParameters: parameters, withError: error, allowedQueryReservedCharacters: allowedQueryReservedCharacters, includeAppVersionParameter: includeAppVersionParameter, onComplete: onComplete) - } } extension PixelKit: PixelFiring { diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index b420b2a9e..1fa3c836a 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -230,6 +230,7 @@ final class PixelExperimentKitTests: XCTestCase { XCTAssertTrue(firedEvent.isEmpty) XCTAssertTrue(firedFrequency.isEmpty) XCTAssertTrue(firedIncludeAppVersion.isEmpty) + print(mockPixelStore.store) XCTAssertEqual(mockPixelStore.store.count, 0) } From 92bb24f959b2d819c2867bf2086888e9fad921ef Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Fri, 29 Nov 2024 18:13:57 +0100 Subject: [PATCH 32/49] refactor --- .../ExperimentActionPixelManager.swift | 55 +++++++++++++++++++ .../PixelExperimentKit.swift | 24 ++------ .../PixelExperimentKitTests.swift | 6 +- 3 files changed, 63 insertions(+), 22 deletions(-) create mode 100644 Sources/PixelExperimentKit/ExperimentActionPixelManager.swift diff --git a/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift b/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift new file mode 100644 index 000000000..0099f030f --- /dev/null +++ b/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift @@ -0,0 +1,55 @@ +// +// ExperimentActionPixelManager.swift +// DuckDuckGo +// +// Copyright © 2024 DuckDuckGo. All rights reserved. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. +// + +import Foundation + +public protocol ExperimentActionPixelStore { + func removeObject(forKey defaultName: String) + func integer(forKey defaultName: String) -> Int + func set(_ value: Int, forKey defaultName: String) + } + +public protocol ExperimentActionPixelManaging { + func incrementAndCheck(forKey key: String, threshold: Int) -> Bool +} + +public struct ExperimentActionPixelManager: ExperimentActionPixelManaging { + let store: ExperimentActionPixelStore + private let syncQueue = DispatchQueue(label: "com.pixelkit.experimentActionSyncQueue") + + public init(store: ExperimentActionPixelStore = UserDefaults.standard) { + self.store = store + } + + public func incrementAndCheck(forKey key: String, threshold: Int) -> Bool { + syncQueue.sync { + let currentCount = store.integer(forKey: key) + let newCount = currentCount + 1 + store.set(newCount, forKey: key) + + if newCount >= threshold { + store.removeObject(forKey: key) + return true + } + return false + } + } +} + +extension UserDefaults: ExperimentActionPixelStore {} diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 071b0fe24..7035132e6 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -26,12 +26,6 @@ struct ExperimentEvent: PixelKitEvent { var parameters: [String : String]? } -public protocol ExperimentActionPixelStore { - func removeObject(forKey defaultName: String) - func integer(forKey defaultName: String) -> Int - func set(_ value: Int, forKey defaultName: String) -} - extension PixelKit { struct Constants { @@ -48,7 +42,7 @@ extension PixelKit { // Static property to hold shared dependencies struct ExperimentConfig { static var featureFlagger: FeatureFlagger? - static var store: ExperimentActionPixelStore = UserDefaults.standard + static var store: ExperimentActionPixelManaging = ExperimentActionPixelManager() static var fireFunction: (PixelKitEvent, PixelKit.Frequency, Bool) -> Void = { event, frequency, includeAppVersion in fire(event, frequency: frequency, includeAppVersionParameter: includeAppVersion) } @@ -57,7 +51,7 @@ extension PixelKit { // Setup method to initialize dependencies public static func configureExperimentKit( featureFlagger: FeatureFlagger, - store: ExperimentActionPixelStore = UserDefaults.standard, + store: ExperimentActionPixelManaging = ExperimentActionPixelManager(), fire: @escaping (PixelKitEvent, PixelKit.Frequency, Bool) -> Void = { event, frequency, includeAppVersion in fire(event, frequency: frequency, includeAppVersionParameter: includeAppVersion) } @@ -119,7 +113,7 @@ extension PixelKit { return } featureFlagger.getAllActiveExperiments().forEach { experiment in - fireExperimentPixelsfor( + fireExperimentPixelsFor( experiment.key, experimentData: experiment.value, metric: Self.Constants.searchMetricValue, @@ -148,7 +142,7 @@ extension PixelKit { return } featureFlagger.getAllActiveExperiments().forEach { experiment in - fireExperimentPixelsfor( + fireExperimentPixelsFor( experiment.key, experimentData: experiment.value, metric: Self.Constants.appUseMetricValue, @@ -157,7 +151,7 @@ extension PixelKit { } } - private static func fireExperimentPixelsfor( + private static func fireExperimentPixelsFor( _ experiment: SubfeatureID, experimentData: ExperimentData, metric: String, @@ -191,8 +185,6 @@ extension PixelKit { // Check if user is in conversion window // if not don't send pixel and remove related action from the store guard isUserInConversionWindow(conversionWindowDays, enrollmentDate: experimentData.enrollmentDate) else { - print(eventStoreKey) - ExperimentConfig.store.removeObject(forKey: eventStoreKey) return } @@ -202,9 +194,7 @@ extension PixelKit { // if not increase the count of the action // if value is not a number send the pixel if let numberOfAction = Int(value), numberOfAction > 1 { - let actualActionNumber = ExperimentConfig.store.integer(forKey: eventStoreKey) + 1 - ExperimentConfig.store.set(actualActionNumber, forKey: eventStoreKey) - if actualActionNumber >= numberOfAction { + if ExperimentConfig.store.incrementAndCheck(forKey: eventStoreKey, threshold: numberOfAction) { ExperimentConfig.fireFunction(event, .uniqueByNameAndParameters, false) } } else { @@ -229,8 +219,6 @@ extension PixelKit { } } -extension UserDefaults: ExperimentActionPixelStore {} - extension Date { public func toYYYYMMDDInET() -> String { let formatter = DateFormatter() diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index 1fa3c836a..940ab3bee 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -35,7 +35,7 @@ final class PixelExperimentKitTests: XCTestCase { super.setUp() mockPixelStore = MockExperimentActionPixelStore() mockFeatureFlagger = MockFeatureFlagger() - PixelKit.configureExperimentKit(featureFlagger: mockFeatureFlagger, store: mockPixelStore, fire: { event, frequency, includeAppVersion in + PixelKit.configureExperimentKit(featureFlagger: mockFeatureFlagger, store: ExperimentActionPixelManager(store: mockPixelStore), fire: { event, frequency, includeAppVersion in self.firedEvent.append(event) self.firedFrequency.append(frequency) self.firedIncludeAppVersion.append(includeAppVersion) @@ -220,8 +220,6 @@ final class PixelExperimentKitTests: XCTestCase { "enrollmentDate": enrollmentDate.toYYYYMMDDInET() ] let eventStoreKey = expectedEventName + "_" + expectedParameters.toString() - print(eventStoreKey) - mockPixelStore.store = [eventStoreKey: 2] // WHEN PixelKit.fireExperimentPixel(for: subfeatureID, metric: "someMetric", conversionWindowDays: conversionWindow, value: value) @@ -231,7 +229,6 @@ final class PixelExperimentKitTests: XCTestCase { XCTAssertTrue(firedFrequency.isEmpty) XCTAssertTrue(firedIncludeAppVersion.isEmpty) print(mockPixelStore.store) - XCTAssertEqual(mockPixelStore.store.count, 0) } func testFireSearchExperimentPixels_WithMultipleExperiments() { @@ -364,6 +361,7 @@ final class PixelExperimentKitTests: XCTestCase { class MockExperimentActionPixelStore: ExperimentActionPixelStore { + var store: [String: Int] = [:] func removeObject(forKey defaultName: String) { From e7801f9d29f481fbb1ff325e4bd565acf7d76c70 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Mon, 2 Dec 2024 11:58:32 +0100 Subject: [PATCH 33/49] refactor and add more tests --- .../ExperimentActionPixelManager.swift | 13 +++++- .../PixelExperimentKit.swift | 43 ++++++++++--------- 2 files changed, 33 insertions(+), 23 deletions(-) diff --git a/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift b/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift index 0099f030f..36a7aad12 100644 --- a/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift +++ b/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift @@ -26,7 +26,7 @@ public protocol ExperimentActionPixelStore { } public protocol ExperimentActionPixelManaging { - func incrementAndCheck(forKey key: String, threshold: Int) -> Bool + func incrementAndCheckThreshold(forKey key: String, threshold: Int, isInWindow: Bool) -> Bool } public struct ExperimentActionPixelManager: ExperimentActionPixelManaging { @@ -37,12 +37,20 @@ public struct ExperimentActionPixelManager: ExperimentActionPixelManaging { self.store = store } - public func incrementAndCheck(forKey key: String, threshold: Int) -> Bool { + public func incrementAndCheckThreshold(forKey key: String, threshold: Int, isInWindow: Bool) -> Bool { syncQueue.sync { + // Remove the key if is not in window + guard isInWindow else { + store.removeObject(forKey: key) + return false + } + + // Increment the current count let currentCount = store.integer(forKey: key) let newCount = currentCount + 1 store.set(newCount, forKey: key) + // Check if the threshold is exceeded if newCount >= threshold { store.removeObject(forKey: key) return true @@ -50,6 +58,7 @@ public struct ExperimentActionPixelManager: ExperimentActionPixelManaging { return false } } + } extension UserDefaults: ExperimentActionPixelStore {} diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 7035132e6..35614d7ce 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -180,42 +180,43 @@ extension PixelKit { Self.Constants.enrollmentDateKey: experimentData.enrollmentDate.toYYYYMMDDInET() ] let event = ExperimentEvent(name: eventName, parameters: parameters) - let eventStoreKey = eventName + "_" + parameters.toString() + let eventStoreKey = "\(eventName)_\(parameters.toString())" - // Check if user is in conversion window - // if not don't send pixel and remove related action from the store - guard isUserInConversionWindow(conversionWindowDays, enrollmentDate: experimentData.enrollmentDate) else { - return - } + // Determine if the user is within the conversion window + let isInWindow = isUserInConversionWindow(conversionWindowDays, enrollmentDate: experimentData.enrollmentDate) // Check if value is a number - // if so check if the action for the given experiment and parameter has been performed a number of time >= than the required - // if so send the pixel - // if not increase the count of the action - // if value is not a number send the pixel if let numberOfAction = Int(value), numberOfAction > 1 { - if ExperimentConfig.store.incrementAndCheck(forKey: eventStoreKey, threshold: numberOfAction) { + // Increment or remove based on conversion window status + let shouldSendPixel = ExperimentConfig.store.incrementAndCheckThreshold( + forKey: eventStoreKey, + threshold: numberOfAction, + isInWindow: isInWindow + ) + + // Send the pixel only if conditions are met + if shouldSendPixel { ExperimentConfig.fireFunction(event, .uniqueByNameAndParameters, false) } - } else { + } else if isInWindow { + // If value is not a number, send the pixel only if within the window ExperimentConfig.fireFunction(event, .uniqueByNameAndParameters, false) } } - private static func isUserInConversionWindow(_ conversionWindowDays: ClosedRange, enrollmentDate: Date) -> Bool { + private static func isUserInConversionWindow( + _ conversionWindowRange: ClosedRange, + enrollmentDate: Date + ) -> Bool { let calendar = Calendar.current - guard let startOfWindowDate = enrollmentDate.addDays(conversionWindowDays.lowerBound), - let endOfWindowDate = enrollmentDate.addDays(conversionWindowDays.upperBound) else { + guard let startOfWindow = enrollmentDate.addDays(conversionWindowRange.lowerBound), + let endOfWindow = enrollmentDate.addDays(conversionWindowRange.upperBound) else { return false } - // Normalize dates to the start of the day - let normalizedStart = calendar.startOfDay(for: startOfWindowDate) - let normalizedEnd = calendar.startOfDay(for: endOfWindowDate) let currentDate = calendar.startOfDay(for: Date()) - - // Check if the current date falls within the normalized range - return currentDate >= normalizedStart && currentDate <= normalizedEnd + return currentDate >= calendar.startOfDay(for: startOfWindow) && + currentDate <= calendar.startOfDay(for: endOfWindow) } } From fc1c31e9e7c9466cc899ea3e41f1a5d89a06642f Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Mon, 2 Dec 2024 12:03:29 +0100 Subject: [PATCH 34/49] add tests --- .../PixelExperimentKitTests.swift | 162 +++++++++++++++++- 1 file changed, 156 insertions(+), 6 deletions(-) diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index 940ab3bee..f1bd048e7 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -27,6 +27,7 @@ final class PixelExperimentKitTests: XCTestCase { var featureJson: Data = "{}".data(using: .utf8)! var mockPixelStore: MockExperimentActionPixelStore! var mockFeatureFlagger: MockFeatureFlagger! + var firedEventSet = Set() var firedEvent = [PixelKitEvent]() var firedFrequency = [PixelKit.Frequency]() var firedIncludeAppVersion = [Bool]() @@ -36,6 +37,7 @@ final class PixelExperimentKitTests: XCTestCase { mockPixelStore = MockExperimentActionPixelStore() mockFeatureFlagger = MockFeatureFlagger() PixelKit.configureExperimentKit(featureFlagger: mockFeatureFlagger, store: ExperimentActionPixelManager(store: mockPixelStore), fire: { event, frequency, includeAppVersion in + self.firedEventSet.insert(event.name + "_" + (event.parameters?.toString() ?? "")) self.firedEvent.append(event) self.firedFrequency.append(frequency) self.firedIncludeAppVersion.append(includeAppVersion) @@ -95,13 +97,14 @@ final class PixelExperimentKitTests: XCTestCase { XCTAssertEqual(firedEvent[0].parameters, expectedParameters) XCTAssertEqual(firedFrequency[0], .uniqueByNameAndParameters) XCTAssertFalse(firedIncludeAppVersion[0]) + XCTAssertEqual(mockPixelStore.store.count, 0) } func testFireExperimentPixel_WithValidExperimentAndConversionWindowAndValue1() { // GIVEN let subfeatureID = "credentialsSaving" let cohort = "control" - let enrollmentDate = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago + let enrollmentDate = Date().addingTimeInterval(-3 * 24 * 60 * 60) // 5 days ago let conversionWindow = 3...7 let value = "1" let expectedEventName = "experiment_metrics_\(subfeatureID)_\(cohort)" @@ -122,13 +125,14 @@ final class PixelExperimentKitTests: XCTestCase { XCTAssertEqual(firedEvent[0].parameters, expectedParameters) XCTAssertEqual(firedFrequency[0], .uniqueByNameAndParameters) XCTAssertFalse(firedIncludeAppVersion[0]) + XCTAssertEqual(mockPixelStore.store.count, 0) } func testFireExperimentPixel_WithValidExperimentAndConversionWindowAndValueN() { // GIVEN let subfeatureID = "credentialsSaving" let cohort = "control" - let enrollmentDate = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago + let enrollmentDate = Date().addingTimeInterval(-7 * 24 * 60 * 60) // 5 days ago let conversionWindow = 3...7 let randomNumber = Int.random(in: 1...100) let value = "\(randomNumber)" @@ -161,6 +165,7 @@ final class PixelExperimentKitTests: XCTestCase { XCTAssertEqual(firedEvent[0].parameters, expectedParameters) XCTAssertEqual(firedFrequency[0], .uniqueByNameAndParameters) XCTAssertFalse(firedIncludeAppVersion[0]) + XCTAssertEqual(mockPixelStore.store.count, 0) } func testFireExperimentPixel_WithInvalidExperimentAndValidConversionWindowAndValue1() { @@ -185,8 +190,7 @@ final class PixelExperimentKitTests: XCTestCase { // GIVEN let subfeatureID = "credentialsSaving" let cohort = "control" - let enrollmentDate = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago - print(enrollmentDate) + let enrollmentDate = Date().addingTimeInterval(-7 * 24 * 60 * 60) // 7 days ago let conversionWindow = 8...11 let value = "3" let experimentData = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate) @@ -206,7 +210,7 @@ final class PixelExperimentKitTests: XCTestCase { // GIVEN let subfeatureID = "credentialsSaving" let cohort = "control" - let enrollmentDate = Date().addingTimeInterval(-6 * 24 * 60 * 60) // 5 days ago + let enrollmentDate = Date().addingTimeInterval(-6 * 24 * 60 * 60) // 6 days ago print(enrollmentDate) let conversionWindow = 3...5 let value = "3" @@ -220,6 +224,7 @@ final class PixelExperimentKitTests: XCTestCase { "enrollmentDate": enrollmentDate.toYYYYMMDDInET() ] let eventStoreKey = expectedEventName + "_" + expectedParameters.toString() + mockPixelStore.store = [eventStoreKey: 2] // WHEN PixelKit.fireExperimentPixel(for: subfeatureID, metric: "someMetric", conversionWindowDays: conversionWindow, value: value) @@ -228,7 +233,145 @@ final class PixelExperimentKitTests: XCTestCase { XCTAssertTrue(firedEvent.isEmpty) XCTAssertTrue(firedFrequency.isEmpty) XCTAssertTrue(firedIncludeAppVersion.isEmpty) - print(mockPixelStore.store) + XCTAssertEqual(mockPixelStore.store.count, 0) + } + + func testFireSearchExperimentPixels_WithValue1() { + let subfeatureID = "credentialsSaving" + let cohort = "control" + let enrollmentDate0 = Date() + let enrollmentDate1 = Date().addingTimeInterval(-1 * 24 * 60 * 60) // 1 days ago + let enrollmentDate2 = Date().addingTimeInterval(-2 * 24 * 60 * 60) // 2 days ago + let enrollmentDate3 = Date().addingTimeInterval(-3 * 24 * 60 * 60) // 3 days ago + let enrollmentDate4 = Date().addingTimeInterval(-4 * 24 * 60 * 60) // 4 days ago + let enrollmentDate5 = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago + let enrollmentDate6 = Date().addingTimeInterval(-6 * 24 * 60 * 60) // 6 days ago + let enrollmentDate7 = Date().addingTimeInterval(-7 * 24 * 60 * 60) // 7 days ago + let enrollmentDate8 = Date().addingTimeInterval(-8 * 24 * 60 * 60) // 8 days ago + let experimentData0 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate0) + let experimentData1 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate1) + let experimentData2 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate2) + let experimentData3 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate3) + let experimentData4 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate4) + let experimentData5 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate5) + let experimentData6 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate6) + let experimentData7 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate7) + let experimentData8 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate8) + + mockFeatureFlagger.experiments = [subfeatureID: experimentData0] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 1) // Fires 1 0-0 + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData1] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 1) // Fires 1 1-1 + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData2] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 1) // Fires 1 2-2 + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData3] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 1) // Fires 1 3-3 + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData4] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 1) // Fires 1 4-4 + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData5] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 2) // Fires 1 5-5 and 1 5-7 + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData6] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 2) // Fires 1 6-6 and 1 5-7 + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData7] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 2) // Fires 1 7-7 and 1 5-7 + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData8] + PixelKit.fireSearchExperimentPixels() + XCTAssertEqual(firedEvent.count, 0) // Nothing + } + + func testFireSearchExperimentPixels_WithValue4() { + let subfeatureID = "credentialsSaving" + let cohort = "control" + let enrollmentDate4 = Date().addingTimeInterval(-4 * 24 * 60 * 60) // 4 days ago + let enrollmentDate5 = Date().addingTimeInterval(-5 * 24 * 60 * 60) // 5 days ago + let enrollmentDate7 = Date().addingTimeInterval(-7 * 24 * 60 * 60) // 7 days ago + let enrollmentDate8 = Date().addingTimeInterval(-8 * 24 * 60 * 60) // 8 days ago + let enrollmentDate15 = Date().addingTimeInterval(-15 * 24 * 60 * 60) // 15 days ago + let enrollmentDate16 = Date().addingTimeInterval(-16 * 24 * 60 * 60) // 16 days ago + let experimentData4 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate4) + let experimentData5 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate5) + let experimentData7 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate7) + let experimentData8 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate8) + let experimentData15 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate15) + let experimentData16 = ExperimentData(parentID: "autofill", cohortID: cohort, enrollmentDate: enrollmentDate16) + + mockFeatureFlagger.experiments = [subfeatureID: experimentData4] + PixelKit.fireSearchExperimentPixels() // Fires 1 4-4 + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + XCTAssertEqual(firedEventSet.count, 1) + PixelKit.fireSearchExperimentPixels() // Nothing + XCTAssertEqual(firedEventSet.count, 1) + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData5] + PixelKit.fireSearchExperimentPixels() // Fires 1 5-5 + 1 5-7 + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + XCTAssertEqual(firedEventSet.count, 2) + PixelKit.fireSearchExperimentPixels() // Fires + 4 5-7 + XCTAssertEqual(firedEventSet.count, 3) + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData7] + PixelKit.fireSearchExperimentPixels() // Fires 1 7-7 + 1 5-7 + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + XCTAssertEqual(firedEventSet.count, 2) + PixelKit.fireSearchExperimentPixels() // Fires + 4 5-7 + XCTAssertEqual(firedEventSet.count, 3) + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData8] + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + XCTAssertEqual(firedEventSet.count, 0) + PixelKit.fireSearchExperimentPixels() // Fires 4 8-15 + XCTAssertEqual(firedEventSet.count, 1) + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData15] + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + XCTAssertEqual(firedEventSet.count, 0) + PixelKit.fireSearchExperimentPixels() // Fires 4 8-15 + XCTAssertEqual(firedEventSet.count, 1) + clearEvents() + + mockFeatureFlagger.experiments = [subfeatureID: experimentData16] + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + PixelKit.fireSearchExperimentPixels() // Nothing + XCTAssertEqual(firedEventSet.count, 0) + PixelKit.fireSearchExperimentPixels() // Nothing + XCTAssertEqual(firedEventSet.count, 0) + clearEvents() } func testFireSearchExperimentPixels_WithMultipleExperiments() { @@ -357,6 +500,13 @@ final class PixelExperimentKitTests: XCTestCase { ) } + private func clearEvents() { + firedEvent = [] + firedEventSet = [] + firedFrequency = [] + firedIncludeAppVersion = [] + } + } From f24740a5db1060f325e94f94c6903a6c7d21ed9d Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Mon, 2 Dec 2024 12:15:03 +0100 Subject: [PATCH 35/49] fix linting --- .../ExperimentActionPixelManager.swift | 1 - Sources/PixelExperimentKit/PixelExperimentKit.swift | 4 ++-- .../PixelExperimentKitTests.swift | 10 +++------- 3 files changed, 5 insertions(+), 10 deletions(-) diff --git a/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift b/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift index 36a7aad12..5b3503848 100644 --- a/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift +++ b/Sources/PixelExperimentKit/ExperimentActionPixelManager.swift @@ -1,6 +1,5 @@ // // ExperimentActionPixelManager.swift -// DuckDuckGo // // Copyright © 2024 DuckDuckGo. All rights reserved. // diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 35614d7ce..4bfac1d67 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -23,7 +23,7 @@ import Foundation struct ExperimentEvent: PixelKitEvent { var name: String - var parameters: [String : String]? + var parameters: [String: String]? } extension PixelKit { @@ -170,7 +170,7 @@ extension PixelKit { } } - private static func fireExperimentPixelForActiveExperiment(_ subfeatureID: SubfeatureID, experimentData: ExperimentData,metric: String, conversionWindowDays: ClosedRange, value: String) { + private static func fireExperimentPixelForActiveExperiment(_ subfeatureID: SubfeatureID, experimentData: ExperimentData, metric: String, conversionWindowDays: ClosedRange, value: String) { // Set parameters, event name, store key let eventName = "\(Self.Constants.metricsEventPrefix)_\(subfeatureID)_\(experimentData.cohortID)" let parameters: [String: String] = [ diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index f1bd048e7..7b4817e33 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -1,6 +1,5 @@ // // PixelExperimentKitTests.swift -// DuckDuckGo // // Copyright © 2024 DuckDuckGo. All rights reserved. // @@ -417,7 +416,6 @@ final class PixelExperimentKitTests: XCTestCase { } ) - // Verify no pixel fired for the second experiment (outside conversion window) XCTAssertNotNil(mockPixelStore.store) XCTAssertFalse( @@ -480,7 +478,6 @@ final class PixelExperimentKitTests: XCTestCase { } ) - // Verify no pixel fired for the second experiment (outside conversion window) XCTAssertNotNil(mockPixelStore.store) XCTAssertFalse( @@ -509,9 +506,8 @@ final class PixelExperimentKitTests: XCTestCase { } - class MockExperimentActionPixelStore: ExperimentActionPixelStore { - + var store: [String: Int] = [:] func removeObject(forKey defaultName: String) { @@ -533,11 +529,11 @@ class MockFeatureFlagger: FeatureFlagger { var internalUserDecider: any InternalUserDecider = MockInternalUserDecider() var localOverrides: (any BrowserServicesKit.FeatureFlagLocalOverriding)? - + func getCohortIfEnabled(for featureFlag: Flag) -> (any FlagCohort)? where Flag: FeatureFlagExperimentDescribing { return nil } - + func getAllActiveExperiments() -> Experiments { return experiments } From aef82ddaa542726a30a6b561f2a8760da9e70627 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Mon, 2 Dec 2024 12:41:28 +0100 Subject: [PATCH 36/49] linting --- .../PixelExperimentKit.swift | 2 +- .../Extensions/Disctionary+PixelKit.swift | 2 +- Sources/PixelKit/PixelKit.swift | 198 ++++++++++++------ 3 files changed, 133 insertions(+), 69 deletions(-) diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 4bfac1d67..0b67631ec 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -1,6 +1,5 @@ // // PixelExperimentKit.swift -// DuckDuckGo // // Copyright © 2024 DuckDuckGo. All rights reserved. // @@ -17,6 +16,7 @@ // limitations under the License. // + import PixelKit import BrowserServicesKit import Foundation diff --git a/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift b/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift index d98ea2daf..f8d76a43e 100644 --- a/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift +++ b/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift @@ -1,6 +1,5 @@ // // Disctionary+PixelKit.swift -// DuckDuckGo // // Copyright © 2024 DuckDuckGo. All rights reserved. // @@ -17,6 +16,7 @@ // limitations under the License. // + public extension Dictionary where Key: Comparable { func toString(pairSeparator: String = ":", entrySeparator: String = ",") -> String { return self.sorted(by: { $0.key < $1.key }) diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index b342996d3..ede2f5dc6 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -202,82 +202,146 @@ public final class PixelKit { switch frequency { case .standard: - reportErrorIf(pixel: pixelName, endsWith: "_u") - reportErrorIf(pixel: pixelName, endsWith: "_d") - fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) + handleStandardFrequency(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .legacyInitial: - reportErrorIf(pixel: pixelName, endsWith: "_u") - reportErrorIf(pixel: pixelName, endsWith: "_d") - if !pixelHasBeenFiredEver(pixelName) { - fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) - updatePixelLastFireDate(pixelName: pixelName) - } else { - printDebugInfo(pixelName: pixelName, frequency: frequency, parameters: newParams, skipped: true) - } + handleLegacyInitial(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .unique: - reportErrorIf(pixel: pixelName, endsWith: "_d") - guard pixelName.hasSuffix("_u") else { - assertionFailure("Unique pixel: must end with _u") - onComplete(false, nil) - return - } - if !pixelHasBeenFiredEver(pixelName) { - fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) - updatePixelLastFireDate(pixelName: pixelName) - } else { - printDebugInfo(pixelName: pixelName, frequency: frequency, parameters: newParams, skipped: true) - } + handleUnique(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .uniqueByNameAndParameters: - let pixelNameAndParams = pixelName + newParams.toString() - if !pixelHasBeenFiredEver(pixelNameAndParams) { - fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) - updatePixelLastFireDate(pixelName: pixelNameAndParams) - } else { - printDebugInfo(pixelName: pixelName, frequency: frequency, parameters: newParams, skipped: true) - } + handleUniqueByNameAndParameters(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .legacyDaily: - reportErrorIf(pixel: pixelName, endsWith: "_u") - reportErrorIf(pixel: pixelName, endsWith: "_d") - if !pixelHasBeenFiredToday(pixelName) { - fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) - updatePixelLastFireDate(pixelName: pixelName) - } else { - printDebugInfo(pixelName: pixelName, frequency: frequency, parameters: newParams, skipped: true) - } + handleLegacyDaily(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .daily: - reportErrorIf(pixel: pixelName, endsWith: "_u") - reportErrorIf(pixel: pixelName, endsWith: "_d") // Because is added automatically - if !pixelHasBeenFiredToday(pixelName) { - fireRequestWrapper(pixelName + "_d", headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) - updatePixelLastFireDate(pixelName: pixelName) - } else { - printDebugInfo(pixelName: pixelName + "_d", frequency: frequency, parameters: newParams, skipped: true) - } + handleDaily(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .legacyDailyAndCount: - reportErrorIf(pixel: pixelName, endsWith: "_u") - reportErrorIf(pixel: pixelName, endsWith: "_d") // Because is added automatically - reportErrorIf(pixel: pixelName, endsWith: "_c") // Because is added automatically - if !pixelHasBeenFiredToday(pixelName) { - fireRequestWrapper(pixelName + "_d", headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) - updatePixelLastFireDate(pixelName: pixelName) - } else { - printDebugInfo(pixelName: pixelName + "_d", frequency: frequency, parameters: newParams, skipped: true) - } - - fireRequestWrapper(pixelName + "_c", headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) + handleLegacyDailyAndCount(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .dailyAndCount: - reportErrorIf(pixel: pixelName, endsWith: "_u") - reportErrorIf(pixel: pixelName, endsWith: "_daily") // Because is added automatically - reportErrorIf(pixel: pixelName, endsWith: "_count") // Because is added automatically - if !pixelHasBeenFiredToday(pixelName) { - fireRequestWrapper(pixelName + "_daily", headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) - updatePixelLastFireDate(pixelName: pixelName) - } else { - printDebugInfo(pixelName: pixelName + "_daily", frequency: frequency, parameters: newParams, skipped: true) - } + handleDailyAndCount(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) + } + } + + private func handleStandardFrequency(_ pixelName: String, + _ headers: [String: String], + _ params: [String: String], + _ allowedQueryReservedCharacters: CharacterSet?, + _ onComplete: @escaping CompletionBlock) { + reportErrorIf(pixel: pixelName, endsWith: "_u") + reportErrorIf(pixel: pixelName, endsWith: "_d") + fireRequestWrapper(pixelName, headers, params, allowedQueryReservedCharacters, true, .standard, onComplete) + } + + private func handleLegacyInitial(_ pixelName: String, + _ headers: [String: String], + _ newParams: [String: String], + _ allowedQueryReservedCharacters: CharacterSet?, + _ onComplete: @escaping CompletionBlock) { + reportErrorIf(pixel: pixelName, endsWith: "_u") + reportErrorIf(pixel: pixelName, endsWith: "_d") + if !pixelHasBeenFiredEver(pixelName) { + fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, .legacyInitial, onComplete) + updatePixelLastFireDate(pixelName: pixelName) + } else { + printDebugInfo(pixelName: pixelName, frequency: .legacyInitial, parameters: newParams, skipped: true) + } + } + + private func handleUnique(_ pixelName: String, + _ headers: [String: String], + _ newParams: [String: String], + _ allowedQueryReservedCharacters: CharacterSet?, + _ onComplete: @escaping CompletionBlock) { + reportErrorIf(pixel: pixelName, endsWith: "_d") + guard pixelName.hasSuffix("_u") else { + assertionFailure("Unique pixel: must end with _u") + onComplete(false, nil) + return + } + if !pixelHasBeenFiredEver(pixelName) { + fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, .unique, onComplete) + updatePixelLastFireDate(pixelName: pixelName) + } else { + printDebugInfo(pixelName: pixelName, frequency: .unique, parameters: newParams, skipped: true) + } + } + + private func handleUniqueByNameAndParameters(_ pixelName: String, + _ headers: [String: String], + _ newParams: [String: String], + _ allowedQueryReservedCharacters: CharacterSet?, + _ onComplete: @escaping CompletionBlock) { + let pixelNameAndParams = pixelName + newParams.toString() + if !pixelHasBeenFiredEver(pixelNameAndParams) { + fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, .uniqueByNameAndParameters, onComplete) + updatePixelLastFireDate(pixelName: pixelNameAndParams) + } else { + printDebugInfo(pixelName: pixelName, frequency: .uniqueByNameAndParameters, parameters: newParams, skipped: true) + } + } + + private func handleLegacyDaily(_ pixelName: String, + _ headers: [String: String], + _ newParams: [String: String], + _ allowedQueryReservedCharacters: CharacterSet?, + _ onComplete: @escaping CompletionBlock) { + reportErrorIf(pixel: pixelName, endsWith: "_u") + reportErrorIf(pixel: pixelName, endsWith: "_d") + if !pixelHasBeenFiredToday(pixelName) { + fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, .legacyDaily, onComplete) + updatePixelLastFireDate(pixelName: pixelName) + } else { + printDebugInfo(pixelName: pixelName, frequency: .legacyDaily, parameters: newParams, skipped: true) + } + } - fireRequestWrapper(pixelName + "_count", headers, newParams, allowedQueryReservedCharacters, true, frequency, onComplete) + private func handleDaily(_ pixelName: String, + _ headers: [String: String], + _ newParams: [String: String], + _ allowedQueryReservedCharacters: CharacterSet?, + _ onComplete: @escaping CompletionBlock) { + reportErrorIf(pixel: pixelName, endsWith: "_u") + reportErrorIf(pixel: pixelName, endsWith: "_d") // Because is added automatically + if !pixelHasBeenFiredToday(pixelName) { + fireRequestWrapper(pixelName + "_d", headers, newParams, allowedQueryReservedCharacters, true, .daily, onComplete) + updatePixelLastFireDate(pixelName: pixelName) + } else { + printDebugInfo(pixelName: pixelName + "_d", frequency: .daily, parameters: newParams, skipped: true) + } + } + + private func handleLegacyDailyAndCount(_ pixelName: String, + _ headers: [String: String], + _ newParams: [String: String], + _ allowedQueryReservedCharacters: CharacterSet?, + _ onComplete: @escaping CompletionBlock) { + reportErrorIf(pixel: pixelName, endsWith: "_u") + reportErrorIf(pixel: pixelName, endsWith: "_d") // Because is added automatically + reportErrorIf(pixel: pixelName, endsWith: "_c") // Because is added automatically + if !pixelHasBeenFiredToday(pixelName) { + fireRequestWrapper(pixelName + "_d", headers, newParams, allowedQueryReservedCharacters, true, .legacyDailyAndCount, onComplete) + updatePixelLastFireDate(pixelName: pixelName) + } else { + printDebugInfo(pixelName: pixelName + "_d", frequency: .legacyDailyAndCount, parameters: newParams, skipped: true) } + + fireRequestWrapper(pixelName + "_c", headers, newParams, allowedQueryReservedCharacters, true, .legacyDailyAndCount, onComplete) + } + + private func handleDailyAndCount(_ pixelName: String, + _ headers: [String: String], + _ newParams: [String: String], + _ allowedQueryReservedCharacters: CharacterSet?, + _ onComplete: @escaping CompletionBlock) { + reportErrorIf(pixel: pixelName, endsWith: "_u") + reportErrorIf(pixel: pixelName, endsWith: "_daily") // Because is added automatically + reportErrorIf(pixel: pixelName, endsWith: "_count") // Because is added automatically + if !pixelHasBeenFiredToday(pixelName) { + fireRequestWrapper(pixelName + "_daily", headers, newParams, allowedQueryReservedCharacters, true, .dailyAndCount, onComplete) + updatePixelLastFireDate(pixelName: pixelName) + } else { + printDebugInfo(pixelName: pixelName + "_daily", frequency: .dailyAndCount, parameters: newParams, skipped: true) + } + + fireRequestWrapper(pixelName + "_count", headers, newParams, allowedQueryReservedCharacters, true, .dailyAndCount, onComplete) } /// If the pixel name ends with the forbiddenString then an error is logged or an assertion failure is fired in debug From 9b65cc7cbe76d769bcc8ee16b6924a05760aac27 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Mon, 2 Dec 2024 12:44:09 +0100 Subject: [PATCH 37/49] linting --- Sources/PixelExperimentKit/PixelExperimentKit.swift | 1 - Sources/PixelKit/Extensions/Disctionary+PixelKit.swift | 1 - 2 files changed, 2 deletions(-) diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 0b67631ec..190aa9933 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -16,7 +16,6 @@ // limitations under the License. // - import PixelKit import BrowserServicesKit import Foundation diff --git a/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift b/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift index f8d76a43e..84f3491b7 100644 --- a/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift +++ b/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift @@ -16,7 +16,6 @@ // limitations under the License. // - public extension Dictionary where Key: Comparable { func toString(pairSeparator: String = ":", entrySeparator: String = ",") -> String { return self.sorted(by: { $0.key < $1.key }) From 4992efed02d05d3120ece7ddd6f9148890bf16db Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Tue, 3 Dec 2024 14:08:24 +0100 Subject: [PATCH 38/49] implement firing enrolment pixel --- .../ExperimentCohortsManager.swift | 6 +++- .../PixelExperimentKit.swift | 5 +++- .../ExperimentCohortsManagerTests.swift | 29 +++++++++++++++++-- .../FeatureFlaggerExperimentsTests.swift | 2 +- 4 files changed, 37 insertions(+), 5 deletions(-) diff --git a/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift index 60a03f8c4..2d0943379 100644 --- a/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift +++ b/Sources/BrowserServicesKit/FeatureFlagger/ExperimentCohortsManager.swift @@ -72,6 +72,7 @@ public class ExperimentCohortsManager: ExperimentCohortsManaging { private var store: ExperimentsDataStoring private let randomizer: (Range) -> Double private let queue = DispatchQueue(label: "com.ExperimentCohortsManager.queue") + private let fireCohortAssigned: (_ subfeatureID: SubfeatureID, _ experiment: ExperimentData) -> Void public var experiments: Experiments? { get { @@ -81,9 +82,11 @@ public class ExperimentCohortsManager: ExperimentCohortsManaging { } } - public init(store: ExperimentsDataStoring = ExperimentsDataStore(), randomizer: @escaping (Range) -> Double = Double.random(in:)) { + public init(store: ExperimentsDataStoring = ExperimentsDataStore(), randomizer: @escaping (Range) -> Double = Double.random(in:), + fireCohortAssigned: @escaping (_ subfeatureID: SubfeatureID, _ experiment: ExperimentData) -> Void) { self.store = store self.randomizer = randomizer + self.fireCohortAssigned = fireCohortAssigned } public func resolveCohort(for experiment: ExperimentSubfeature, allowCohortReassignment: Bool) -> CohortID? { @@ -113,6 +116,7 @@ extension ExperimentCohortsManager { cumulativeWeight += Double(cohort.weight) if randomValue < cumulativeWeight { saveCohort(cohort.name, in: subfeature.subfeatureID, parentID: subfeature.parentID) + fireCohortAssigned(subfeature.subfeatureID, ExperimentData(parentID: subfeature.parentID, cohortID: cohort.name, enrollmentDate: Date())) return cohort.name } } diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 190aa9933..1650b60f7 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -172,9 +172,12 @@ extension PixelKit { private static func fireExperimentPixelForActiveExperiment(_ subfeatureID: SubfeatureID, experimentData: ExperimentData, metric: String, conversionWindowDays: ClosedRange, value: String) { // Set parameters, event name, store key let eventName = "\(Self.Constants.metricsEventPrefix)_\(subfeatureID)_\(experimentData.cohortID)" + let conversionWindowValue = (conversionWindowDays.lowerBound != conversionWindowDays.upperBound) ? + "\(conversionWindowDays.lowerBound)-\(conversionWindowDays.upperBound)" : + "\(conversionWindowDays.lowerBound)" let parameters: [String: String] = [ Self.Constants.metricKey: metric, - Self.Constants.conversionWindowDaysKey: "\(conversionWindowDays.lowerBound.description)-\(conversionWindowDays.upperBound.description)", + Self.Constants.conversionWindowDaysKey: conversionWindowValue, Self.Constants.valueKey: value, Self.Constants.enrollmentDateKey: experimentData.enrollmentDate.toYYYYMMDDInET() ] diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift index 48fc85355..7a2be7abc 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/ExperimentCohortsManagerTests.swift @@ -41,6 +41,9 @@ final class ExperimentCohortsManagerTests: XCTestCase { let subfeatureName4 = "TestSubfeature4" var experimentData4: ExperimentData! + var firedSubfeatureID: SubfeatureID? + var firedExperimentData: ExperimentData? + let encoder: JSONEncoder = { let encoder = JSONEncoder() encoder.dateEncodingStrategy = .secondsSince1970 @@ -50,8 +53,12 @@ final class ExperimentCohortsManagerTests: XCTestCase { override func setUp() { super.setUp() mockStore = MockExperimentDataStore() + experimentCohortsManager = ExperimentCohortsManager( - store: mockStore + store: mockStore, fireCohortAssigned: {subfeatureID, experimentData in + self.firedSubfeatureID = subfeatureID + self.firedExperimentData = experimentData + } ) let expectedDate1 = Date() @@ -87,6 +94,8 @@ final class ExperimentCohortsManagerTests: XCTestCase { XCTAssertEqual(experiments?[subfeatureName1], experimentData1) XCTAssertEqual(experiments?[subfeatureName2], experimentData2) XCTAssertNil(experiments?[subfeatureName3]) + XCTAssertNil(firedSubfeatureID) + XCTAssertNil(firedExperimentData) } func testCohortReturnsCohortIDIfExistsForMultipleSubfeatures() { @@ -100,6 +109,8 @@ final class ExperimentCohortsManagerTests: XCTestCase { // THEN XCTAssertEqual(result1, experimentData1.cohortID) XCTAssertEqual(result2, experimentData2.cohortID) + XCTAssertNil(firedSubfeatureID) + XCTAssertNil(firedExperimentData) } func testCohortAssignIfEnabledWhenNoCohortExists() { @@ -114,6 +125,10 @@ final class ExperimentCohortsManagerTests: XCTestCase { // THEN XCTAssertNotNil(result) XCTAssertEqual(result, experimentData1.cohortID) + XCTAssertEqual(firedSubfeatureID, subfeatureName1) + XCTAssertEqual(firedExperimentData?.cohortID, experimentData1.cohortID) + XCTAssertEqual(firedExperimentData?.parentID, experimentData1.parentID) + XCTAssertEqual(firedExperimentData?.enrollmentDate.daySinceReferenceDate, experimentData1.enrollmentDate.daySinceReferenceDate) } func testCohortDoesNotAssignIfAssignIfEnabledIsFalse() { @@ -127,6 +142,8 @@ final class ExperimentCohortsManagerTests: XCTestCase { // THEN XCTAssertNil(result) + XCTAssertNil(firedSubfeatureID) + XCTAssertNil(firedExperimentData) } func testCohortDoesNotAssignIfAssignIfEnabledIsTrueButNoCohortsAvailable() { @@ -139,6 +156,8 @@ final class ExperimentCohortsManagerTests: XCTestCase { // THEN XCTAssertNil(result) + XCTAssertNil(firedSubfeatureID) + XCTAssertNil(firedExperimentData) } func testCohortReassignsCohortIfAssignedCohortDoesNotExistAndAssignIfEnabledIsTrue() { @@ -150,6 +169,10 @@ final class ExperimentCohortsManagerTests: XCTestCase { // THEN XCTAssertEqual(result1, experimentData3.cohortID) + XCTAssertEqual(firedSubfeatureID, subfeatureName1) + XCTAssertEqual(firedExperimentData?.cohortID, experimentData3.cohortID) + XCTAssertEqual(firedExperimentData?.parentID, experimentData3.parentID) + XCTAssertEqual(firedExperimentData?.enrollmentDate.daySinceReferenceDate, experimentData3.enrollmentDate.daySinceReferenceDate) } func testCohortDoesNotReassignsCohortIfAssignedCohortDoesNotExistAndAssignIfEnabledIsTrue() { @@ -161,6 +184,8 @@ final class ExperimentCohortsManagerTests: XCTestCase { // THEN XCTAssertNil(result1) + XCTAssertNil(firedSubfeatureID) + XCTAssertNil(firedExperimentData) } func testCohortAssignsBasedOnWeight() { @@ -173,7 +198,7 @@ final class ExperimentCohortsManagerTests: XCTestCase { experimentCohortsManager = ExperimentCohortsManager( store: mockStore, - randomizer: randomizer + randomizer: randomizer, fireCohortAssigned: { _, _ in } ) // WHEN diff --git a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift index b6931ef22..933eb11b3 100644 --- a/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift +++ b/Tests/BrowserServicesKitTests/FeatureFlagging/FeatureFlaggerExperimentsTests.swift @@ -81,7 +81,7 @@ final class FeatureFlaggerExperimentsTests: XCTestCase { mockEmbeddedData = MockEmbeddedDataProvider(data: featureJson, etag: "test") let mockInternalUserStore = MockInternalUserStoring() mockStore = MockExperimentDataStore() - experimentManager = ExperimentCohortsManager(store: mockStore) + experimentManager = ExperimentCohortsManager(store: mockStore, fireCohortAssigned: { _, _ in }) manager = PrivacyConfigurationManager(fetchedETag: nil, fetchedData: nil, embeddedDataProvider: mockEmbeddedData, From b202b440e2f3481294207c8004e50f146ef3e9bb Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Tue, 3 Dec 2024 14:37:56 +0100 Subject: [PATCH 39/49] fix tests --- Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index 7b4817e33..380ca4afb 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -81,7 +81,7 @@ final class PixelExperimentKitTests: XCTestCase { let expectedEventName = "experiment_metrics_\(subfeatureID)_\(cohort)" let expectedParameters = [ "metric": "someMetric", - "conversionWindowDays": "3-3", + "conversionWindowDays": "3", "value": value, "enrollmentDate": enrollmentDate.toYYYYMMDDInET() ] @@ -412,7 +412,7 @@ final class PixelExperimentKitTests: XCTestCase { ) XCTAssertTrue( firedEvent.contains { - $0.parameters?[PixelKit.Constants.conversionWindowDaysKey] == "6-6" + $0.parameters?[PixelKit.Constants.conversionWindowDaysKey] == "6" } ) @@ -474,7 +474,7 @@ final class PixelExperimentKitTests: XCTestCase { ) XCTAssertTrue( firedEvent.contains { - $0.parameters?[PixelKit.Constants.conversionWindowDaysKey] == "6-6" + $0.parameters?[PixelKit.Constants.conversionWindowDaysKey] == "6" } ) From c62fac4c79969c47c5396b23686c400df30b48eb Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Tue, 3 Dec 2024 17:16:30 +0100 Subject: [PATCH 40/49] generalise headers and prefix name --- Sources/PixelKit/PixelKit.swift | 27 +++++++++++++++++++++++++-- 1 file changed, 25 insertions(+), 2 deletions(-) diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index ede2f5dc6..6ee4ae905 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -195,7 +195,18 @@ public final class PixelKit { var headers = headers ?? defaultHeaders headers[Header.moreInfo] = "See " + Self.duckDuckGoMorePrivacyInfo.absoluteString - headers[Header.client] = "macOS" + if let source { + switch source { + case Source.iOS.rawValue: + headers[Header.client] = "iOS" + case Source.iPadOS.rawValue: + headers[Header.client] = "iPadOS" + case Source.macDMG.rawValue, Source.macStore.rawValue: + headers[Header.client] = "macOS" + default: + break + } + } // The event name can't contain `.` reportErrorIf(pixel: pixelName, contains: ".") @@ -395,7 +406,19 @@ public final class PixelKit { // Special kind of pixel event that don't follow the standard naming conventions return nonStandardEvent.name } else { - return "m_mac_\(event.name)" + if let source { + switch source { + case Source.iOS.rawValue: + return "m_ios_\(event.name)" + case Source.iPadOS.rawValue: + return "m_ipad_\(event.name)" + case Source.macDMG.rawValue, Source.macStore.rawValue: + return "m_mac_\(event.name)" + default: + break + } + } + return event.name } } From cc6c1b66508d7da0f651cfc319712f27489a79ee Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Wed, 4 Dec 2024 09:55:28 +0100 Subject: [PATCH 41/49] address comments --- .../BrowserServicesKit-Package.xcscheme | 30 +++++++++++--- ...ger.swift => ExperimentEventTracker.swift} | 19 +++++++-- .../PixelExperimentKit.swift | 40 +++++++++---------- ...xelKit.swift => DictionaryExtension.swift} | 4 +- Sources/PixelKit/PixelKit.swift | 12 +++--- .../XCTestCase+PixelKit.swift | 2 +- .../PixelExperimentKitTests.swift | 2 +- Tests/PixelKitTests/PixelKitTests.swift | 12 +++--- 8 files changed, 76 insertions(+), 45 deletions(-) rename Sources/PixelExperimentKit/{ExperimentActionPixelManager.swift => ExperimentEventTracker.swift} (64%) rename Sources/PixelKit/Extensions/{Disctionary+PixelKit.swift => DictionaryExtension.swift} (91%) diff --git a/.swiftpm/xcode/xcshareddata/xcschemes/BrowserServicesKit-Package.xcscheme b/.swiftpm/xcode/xcshareddata/xcschemes/BrowserServicesKit-Package.xcscheme index 28a867b7d..bcddddeb8 100644 --- a/.swiftpm/xcode/xcshareddata/xcschemes/BrowserServicesKit-Package.xcscheme +++ b/.swiftpm/xcode/xcshareddata/xcschemes/BrowserServicesKit-Package.xcscheme @@ -547,15 +547,26 @@ buildForAnalyzing = "YES"> + + + + + + + + Bool } -public struct ExperimentActionPixelManager: ExperimentActionPixelManaging { +public struct ExperimentEventTracker: ExperimentEventTracking { let store: ExperimentActionPixelStore private let syncQueue = DispatchQueue(label: "com.pixelkit.experimentActionSyncQueue") diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 1650b60f7..7577fad38 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -41,7 +41,7 @@ extension PixelKit { // Static property to hold shared dependencies struct ExperimentConfig { static var featureFlagger: FeatureFlagger? - static var store: ExperimentActionPixelManaging = ExperimentActionPixelManager() + static var eventTracker: ExperimentEventTracking = ExperimentEventTracker() static var fireFunction: (PixelKitEvent, PixelKit.Frequency, Bool) -> Void = { event, frequency, includeAppVersion in fire(event, frequency: frequency, includeAppVersionParameter: includeAppVersion) } @@ -50,13 +50,13 @@ extension PixelKit { // Setup method to initialize dependencies public static func configureExperimentKit( featureFlagger: FeatureFlagger, - store: ExperimentActionPixelManaging = ExperimentActionPixelManager(), + eventTracker: ExperimentEventTracking = ExperimentEventTracker(), fire: @escaping (PixelKitEvent, PixelKit.Frequency, Bool) -> Void = { event, frequency, includeAppVersion in fire(event, frequency: frequency, includeAppVersionParameter: includeAppVersion) } ) { ExperimentConfig.featureFlagger = featureFlagger - ExperimentConfig.store = store + ExperimentConfig.eventTracker = eventTracker ExperimentConfig.fireFunction = fire } @@ -65,8 +65,8 @@ extension PixelKit { /// - subfeatureID: Identifier for the subfeature associated with the experiment. /// - experiment: Data about the experiment like cohort and enrollment date public static func fireExperimentEnrollmentPixel(subfeatureID: SubfeatureID, experiment: ExperimentData) { - let eventName = "\(Self.Constants.enrollmentEventPrefix)_\(subfeatureID)_\(experiment.cohortID)" - let event = ExperimentEvent(name: eventName, parameters: [Self.Constants.enrollmentDateKey: experiment.enrollmentDate.toYYYYMMDDInET()]) + let eventName = "\(Constants.enrollmentEventPrefix)_\(subfeatureID)_\(experiment.cohortID)" + let event = ExperimentEvent(name: eventName, parameters: [Constants.enrollmentDateKey: experiment.enrollmentDate.toYYYYMMDDInET()]) ExperimentConfig.fireFunction(event, .uniqueByNameAndParameters, false) } @@ -89,7 +89,7 @@ extension PixelKit { } guard let experimentData = featureFlagger.getAllActiveExperiments()[subfeatureID] else { return } - Self.fireExperimentPixelForActiveExperiment(subfeatureID, experimentData: experimentData, metric: metric, conversionWindowDays: conversionWindowDays, value: value) + fireExperimentPixelForActiveExperiment(subfeatureID, experimentData: experimentData, metric: metric, conversionWindowDays: conversionWindowDays, value: value) } /// Fires search-related experiment pixels for all active experiments. @@ -112,10 +112,10 @@ extension PixelKit { return } featureFlagger.getAllActiveExperiments().forEach { experiment in - fireExperimentPixelsFor( + fireExperimentPixels(for: experiment.key, experimentData: experiment.value, - metric: Self.Constants.searchMetricValue, + metric: Constants.searchMetricValue, valueConversionDictionary: valueConversionDictionary ) } @@ -141,17 +141,17 @@ extension PixelKit { return } featureFlagger.getAllActiveExperiments().forEach { experiment in - fireExperimentPixelsFor( - experiment.key, + fireExperimentPixels( + for: experiment.key, experimentData: experiment.value, - metric: Self.Constants.appUseMetricValue, + metric: Constants.appUseMetricValue, valueConversionDictionary: valueConversionDictionary ) } } - private static func fireExperimentPixelsFor( - _ experiment: SubfeatureID, + private static func fireExperimentPixels( + for experiment: SubfeatureID, experimentData: ExperimentData, metric: String, valueConversionDictionary: [Int: [ClosedRange]] @@ -171,15 +171,15 @@ extension PixelKit { private static func fireExperimentPixelForActiveExperiment(_ subfeatureID: SubfeatureID, experimentData: ExperimentData, metric: String, conversionWindowDays: ClosedRange, value: String) { // Set parameters, event name, store key - let eventName = "\(Self.Constants.metricsEventPrefix)_\(subfeatureID)_\(experimentData.cohortID)" + let eventName = "\(Constants.metricsEventPrefix)_\(subfeatureID)_\(experimentData.cohortID)" let conversionWindowValue = (conversionWindowDays.lowerBound != conversionWindowDays.upperBound) ? "\(conversionWindowDays.lowerBound)-\(conversionWindowDays.upperBound)" : "\(conversionWindowDays.lowerBound)" let parameters: [String: String] = [ - Self.Constants.metricKey: metric, - Self.Constants.conversionWindowDaysKey: conversionWindowValue, - Self.Constants.valueKey: value, - Self.Constants.enrollmentDateKey: experimentData.enrollmentDate.toYYYYMMDDInET() + Constants.metricKey: metric, + Constants.conversionWindowDaysKey: conversionWindowValue, + Constants.valueKey: value, + Constants.enrollmentDateKey: experimentData.enrollmentDate.toYYYYMMDDInET() ] let event = ExperimentEvent(name: eventName, parameters: parameters) let eventStoreKey = "\(eventName)_\(parameters.toString())" @@ -190,7 +190,7 @@ extension PixelKit { // Check if value is a number if let numberOfAction = Int(value), numberOfAction > 1 { // Increment or remove based on conversion window status - let shouldSendPixel = ExperimentConfig.store.incrementAndCheckThreshold( + let shouldSendPixel = ExperimentConfig.eventTracker.incrementAndCheckThreshold( forKey: eventStoreKey, threshold: numberOfAction, isInWindow: isInWindow @@ -231,6 +231,6 @@ extension Date { } func addDays(_ days: Int) -> Date? { - return Calendar.current.date(byAdding: .day, value: days, to: self) + Calendar.current.date(byAdding: .day, value: days, to: self) } } diff --git a/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift b/Sources/PixelKit/Extensions/DictionaryExtension.swift similarity index 91% rename from Sources/PixelKit/Extensions/Disctionary+PixelKit.swift rename to Sources/PixelKit/Extensions/DictionaryExtension.swift index 84f3491b7..a905a9854 100644 --- a/Sources/PixelKit/Extensions/Disctionary+PixelKit.swift +++ b/Sources/PixelKit/Extensions/DictionaryExtension.swift @@ -1,5 +1,5 @@ // -// Disctionary+PixelKit.swift +// DictionaryExtension.swift // // Copyright © 2024 DuckDuckGo. All rights reserved. // @@ -18,7 +18,7 @@ public extension Dictionary where Key: Comparable { func toString(pairSeparator: String = ":", entrySeparator: String = ",") -> String { - return self.sorted(by: { $0.key < $1.key }) + sorted(by: { $0.key < $1.key }) .map { "\($0.key)\(pairSeparator)\($0.value)" } .joined(separator: entrySeparator) } diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index 6ee4ae905..99657a3c8 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -33,7 +33,7 @@ public final class PixelKit { /// Sent only once ever (based on pixel name only.) The timestamp for this pixel is stored. /// Note: This is the only pixel that MUST end with `_u`, Name for pixels of this type must end with if it doesn't an assertion is fired. - case unique + case uniqueByName /// Sent only once ever (based on pixel name AND parameters). The timestamp for this pixel is stored. case uniqueByNameAndParameters @@ -60,7 +60,7 @@ public final class PixelKit { "Standard" case .legacyInitial: "Legacy Initial" - case .unique: + case .uniqueByName: "Unique" case .legacyDaily: "Legacy Daily" @@ -216,7 +216,7 @@ public final class PixelKit { handleStandardFrequency(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .legacyInitial: handleLegacyInitial(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) - case .unique: + case .uniqueByName: handleUnique(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) case .uniqueByNameAndParameters: handleUniqueByNameAndParameters(pixelName, headers, newParams, allowedQueryReservedCharacters, onComplete) @@ -268,10 +268,10 @@ public final class PixelKit { return } if !pixelHasBeenFiredEver(pixelName) { - fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, .unique, onComplete) + fireRequestWrapper(pixelName, headers, newParams, allowedQueryReservedCharacters, true, .uniqueByName, onComplete) updatePixelLastFireDate(pixelName: pixelName) } else { - printDebugInfo(pixelName: pixelName, frequency: .unique, parameters: newParams, skipped: true) + printDebugInfo(pixelName: pixelName, frequency: .uniqueByName, parameters: newParams, skipped: true) } } @@ -437,7 +437,7 @@ public final class PixelKit { if frequency == .daily, pixelHasBeenFiredToday(pixelName) { onComplete(false, nil) return - } else if frequency == .unique, pixelHasBeenFiredEver(pixelName) { + } else if frequency == .uniqueByName, pixelHasBeenFiredEver(pixelName) { onComplete(false, nil) return } diff --git a/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift b/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift index 6b7ad1a6c..8ae6efb40 100644 --- a/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift +++ b/Sources/PixelKitTestingUtilities/XCTestCase+PixelKit.swift @@ -149,7 +149,7 @@ public extension XCTestCase { expectedPixelNames.append(originalName) case .legacyInitial: expectedPixelNames.append(originalName) - case .unique: + case .uniqueByName: expectedPixelNames.append(originalName) case .legacyDaily: expectedPixelNames.append(originalName) diff --git a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift index 380ca4afb..30334f6e6 100644 --- a/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift +++ b/Tests/PixelExperimentKitTests/PixelExperimentKitTests.swift @@ -35,7 +35,7 @@ final class PixelExperimentKitTests: XCTestCase { super.setUp() mockPixelStore = MockExperimentActionPixelStore() mockFeatureFlagger = MockFeatureFlagger() - PixelKit.configureExperimentKit(featureFlagger: mockFeatureFlagger, store: ExperimentActionPixelManager(store: mockPixelStore), fire: { event, frequency, includeAppVersion in + PixelKit.configureExperimentKit(featureFlagger: mockFeatureFlagger, eventTracker: ExperimentEventTracker(store: mockPixelStore), fire: { event, frequency, includeAppVersion in self.firedEventSet.insert(event.name + "_" + (event.parameters?.toString() ?? "")) self.firedEvent.append(event) self.firedFrequency.append(frequency) diff --git a/Tests/PixelKitTests/PixelKitTests.swift b/Tests/PixelKitTests/PixelKitTests.swift index 50e1f2d9a..c678c3f9d 100644 --- a/Tests/PixelKitTests/PixelKitTests.swift +++ b/Tests/PixelKitTests/PixelKitTests.swift @@ -82,7 +82,7 @@ final class PixelKitTests: XCTestCase { case .testEvent, .testEventWithoutParameters, .nameWithDot: return .standard case .uniqueEvent: - return .unique + return .uniqueByName case .dailyEvent, .dailyEventWithoutParameters: return .daily case .dailyAndContinuousEvent, .dailyAndContinuousEventWithoutParameters: @@ -397,19 +397,19 @@ final class PixelKitTests: XCTestCase { } // Run test - pixelKit.fire(event, frequency: .unique) // Fired + pixelKit.fire(event, frequency: .uniqueByName) // Fired timeMachine.travel(by: .hour, value: 2) - pixelKit.fire(event, frequency: .unique) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByName) // Skipped (already fired) timeMachine.travel(by: .day, value: 1) timeMachine.travel(by: .hour, value: 2) - pixelKit.fire(event, frequency: .unique) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByName) // Skipped (already fired) timeMachine.travel(by: .hour, value: 10) - pixelKit.fire(event, frequency: .unique) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByName) // Skipped (already fired) timeMachine.travel(by: .day, value: 1) - pixelKit.fire(event, frequency: .unique) // Skipped (already fired) + pixelKit.fire(event, frequency: .uniqueByName) // Skipped (already fired) // Wait for expectations to be fulfilled wait(for: [fireCallbackCalled], timeout: 0.5) From b567a222b3a28693239f46f1a9e370e574814524 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Wed, 4 Dec 2024 10:02:59 +0100 Subject: [PATCH 42/49] add isDry for new unique params --- Sources/PixelKit/PixelKit.swift | 8 ++++++++ 1 file changed, 8 insertions(+) diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index 99657a3c8..37f68d395 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -455,6 +455,14 @@ public final class PixelKit { newParams = nil } + if !dryRun, let newParams { + let pixelNameAndParams = pixelName + newParams.toString() + if frequency == .uniqueByNameAndParameters, pixelHasBeenFiredEver(pixelNameAndParams) { + onComplete(false, nil) + return + } + } + let newError: Error? if let event = event as? PixelKitEventV2, From 2b0f43493b4956e4ea6f2505b67b219f071ae080 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Wed, 4 Dec 2024 10:05:26 +0100 Subject: [PATCH 43/49] fix linting --- Sources/PixelKit/PixelKit.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index 37f68d395..59368e705 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -457,7 +457,7 @@ public final class PixelKit { if !dryRun, let newParams { let pixelNameAndParams = pixelName + newParams.toString() - if frequency == .uniqueByNameAndParameters, pixelHasBeenFiredEver(pixelNameAndParams) { + if frequency == .uniqueByNameAndParameters, pixelHasBeenFiredEver(pixelNameAndParams) { onComplete(false, nil) return } From 15eed7b59e994a6c2bb45f24fa9c2cc08839a533 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Wed, 4 Dec 2024 10:24:27 +0100 Subject: [PATCH 44/49] fix tests --- Tests/PixelKitTests/PixelKitParametersTests.swift | 2 +- Tests/PixelKitTests/PixelKitTests.swift | 3 +++ 2 files changed, 4 insertions(+), 1 deletion(-) diff --git a/Tests/PixelKitTests/PixelKitParametersTests.swift b/Tests/PixelKitTests/PixelKitParametersTests.swift index 4d661d333..4fa3e1f6c 100644 --- a/Tests/PixelKitTests/PixelKitParametersTests.swift +++ b/Tests/PixelKitTests/PixelKitParametersTests.swift @@ -66,7 +66,7 @@ final class PixelKitParametersTests: XCTestCase { fire(TestEvent.errorEvent(error: topLevelError), frequency: .standard, - and: .expect(pixelName: "m_mac_error_event", + and: .expect(pixelName: "error_event", error: topLevelError, underlyingErrors: [underlyingError2, underlyingError3]), file: #filePath, diff --git a/Tests/PixelKitTests/PixelKitTests.swift b/Tests/PixelKitTests/PixelKitTests.swift index c678c3f9d..5225fbdf4 100644 --- a/Tests/PixelKitTests/PixelKitTests.swift +++ b/Tests/PixelKitTests/PixelKitTests.swift @@ -209,6 +209,7 @@ final class PixelKitTests: XCTestCase { // Prepare mock to validate expectations let pixelKit = PixelKit(dryRun: false, appVersion: appVersion, + source: PixelKit.Source.macDMG.rawValue, defaultHeaders: headers, dailyPixelCalendar: nil, defaults: userDefaults) { firedPixelName, firedHeaders, parameters, _, _, _ in @@ -254,6 +255,7 @@ final class PixelKitTests: XCTestCase { // Prepare mock to validate expectations let pixelKit = PixelKit(dryRun: false, appVersion: appVersion, + source: PixelKit.Source.macDMG.rawValue, defaultHeaders: headers, dailyPixelCalendar: nil, defaults: userDefaults) { firedPixelName, firedHeaders, parameters, _, _, _ in @@ -300,6 +302,7 @@ final class PixelKitTests: XCTestCase { // Prepare mock to validate expectations let pixelKit = PixelKit(dryRun: false, appVersion: appVersion, + source: PixelKit.Source.macDMG.rawValue, defaultHeaders: headers, dailyPixelCalendar: nil, defaults: userDefaults) { firedPixelName, firedHeaders, parameters, _, _, _ in From 8db91da8441a247dece2bfe8ba9f0db0fbbf1eeb Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Mon, 9 Dec 2024 12:48:02 +0100 Subject: [PATCH 45/49] fix mac pixel names --- Sources/PixelKit/PixelKit.swift | 8 +++++--- 1 file changed, 5 insertions(+), 3 deletions(-) diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index 59368e705..1ec919e37 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -195,6 +195,7 @@ public final class PixelKit { var headers = headers ?? defaultHeaders headers[Header.moreInfo] = "See " + Self.duckDuckGoMorePrivacyInfo.absoluteString + // Needs to be updated/generalised when fully adopted by iOS if let source { switch source { case Source.iOS.rawValue: @@ -204,7 +205,7 @@ public final class PixelKit { case Source.macDMG.rawValue, Source.macStore.rawValue: headers[Header.client] = "macOS" default: - break + headers[Header.client] = "macOS" } } @@ -395,6 +396,7 @@ public final class PixelKit { fireRequest(pixelName, headers, parameters, allowedQueryReservedCharacters, callBackOnMainThread, onComplete) } + // Needs to be updated when fully adopted by iOS private func prefixedName(for event: Event) -> String { if event.name.hasPrefix("m_mac_") { // Can be a debug event or not, if already prefixed the name remains unchanged @@ -415,10 +417,10 @@ public final class PixelKit { case Source.macDMG.rawValue, Source.macStore.rawValue: return "m_mac_\(event.name)" default: - break + return "m_mac_\(event.name)" } } - return event.name + return "m_mac_\(event.name)" } } From 8c2a9c6154c1991f59cbb3aea3f54b0e50882035 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Wed, 4 Dec 2024 14:03:45 +0100 Subject: [PATCH 46/49] fix test --- Tests/PixelKitTests/PixelKitParametersTests.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Tests/PixelKitTests/PixelKitParametersTests.swift b/Tests/PixelKitTests/PixelKitParametersTests.swift index 4fa3e1f6c..4d661d333 100644 --- a/Tests/PixelKitTests/PixelKitParametersTests.swift +++ b/Tests/PixelKitTests/PixelKitParametersTests.swift @@ -66,7 +66,7 @@ final class PixelKitParametersTests: XCTestCase { fire(TestEvent.errorEvent(error: topLevelError), frequency: .standard, - and: .expect(pixelName: "error_event", + and: .expect(pixelName: "m_mac_error_event", error: topLevelError, underlyingErrors: [underlyingError2, underlyingError3]), file: #filePath, From e0e45ca614ca0cf2b0bee5f897ff7985e0d7a48a Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Wed, 4 Dec 2024 17:40:37 +0100 Subject: [PATCH 47/49] fix iOS pixel naming --- Sources/PixelKit/PixelKit.swift | 10 +++++----- 1 file changed, 5 insertions(+), 5 deletions(-) diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index 1ec919e37..a2a67c907 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -397,7 +397,7 @@ public final class PixelKit { } // Needs to be updated when fully adopted by iOS - private func prefixedName(for event: Event) -> String { + private func prefixedAndSuffixedName(for event: Event) -> String { if event.name.hasPrefix("m_mac_") { // Can be a debug event or not, if already prefixed the name remains unchanged return event.name @@ -411,9 +411,9 @@ public final class PixelKit { if let source { switch source { case Source.iOS.rawValue: - return "m_ios_\(event.name)" + return "m_\(event.name)_ios_phone" case Source.iPadOS.rawValue: - return "m_ipad_\(event.name)" + return "m_\(event.name)_ios_tablet" case Source.macDMG.rawValue, Source.macStore.rawValue: return "m_mac_\(event.name)" default: @@ -433,7 +433,7 @@ public final class PixelKit { includeAppVersionParameter: Bool = true, onComplete: @escaping CompletionBlock = { _, _ in }) { - let pixelName = prefixedName(for: event) + let pixelName = prefixedAndSuffixedName(for: event) if !dryRun { if frequency == .daily, pixelHasBeenFiredToday(pixelName) { @@ -540,7 +540,7 @@ public final class PixelKit { } public func pixelLastFireDate(event: Event) -> Date? { - pixelLastFireDate(pixelName: prefixedName(for: event)) + pixelLastFireDate(pixelName: prefixedAndSuffixedName(for: event)) } private func updatePixelLastFireDate(pixelName: String) { From cd6d89ad149b0047dabed51dcd7107981b032623 Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Thu, 5 Dec 2024 10:48:14 +0100 Subject: [PATCH 48/49] address comments and remove m_ from experiments --- .../ExperimentEventTracker.swift | 10 ++++-- .../PixelExperimentKit.swift | 29 ++++++++++----- Sources/PixelKit/PixelKit.swift | 35 +++++++++++-------- 3 files changed, 49 insertions(+), 25 deletions(-) diff --git a/Sources/PixelExperimentKit/ExperimentEventTracker.swift b/Sources/PixelExperimentKit/ExperimentEventTracker.swift index 51d11a053..e38f39702 100644 --- a/Sources/PixelExperimentKit/ExperimentEventTracker.swift +++ b/Sources/PixelExperimentKit/ExperimentEventTracker.swift @@ -18,6 +18,10 @@ import Foundation +public typealias ThresholdCheckResult = Bool +public typealias ExprimentPixelNameAndParameters = String +public typealias NumberOfActions = Int + public protocol ExperimentActionPixelStore { func removeObject(forKey defaultName: String) func integer(forKey defaultName: String) -> Int @@ -38,18 +42,18 @@ public protocol ExperimentEventTracking { /// - threshold: The count threshold that triggers a return of `true`. /// - isInWindow: A flag indicating if the count should be considered (e.g., within a time window). /// - Returns: `true` if the threshold is exceeded and the count is reset, otherwise `false`. - func incrementAndCheckThreshold(forKey key: String, threshold: Int, isInWindow: Bool) -> Bool + func incrementAndCheckThreshold(forKey key: ExprimentPixelNameAndParameters, threshold: NumberOfActions, isInWindow: Bool) -> ThresholdCheckResult } public struct ExperimentEventTracker: ExperimentEventTracking { - let store: ExperimentActionPixelStore + private let store: ExperimentActionPixelStore private let syncQueue = DispatchQueue(label: "com.pixelkit.experimentActionSyncQueue") public init(store: ExperimentActionPixelStore = UserDefaults.standard) { self.store = store } - public func incrementAndCheckThreshold(forKey key: String, threshold: Int, isInWindow: Bool) -> Bool { + public func incrementAndCheckThreshold(forKey key: ExprimentPixelNameAndParameters, threshold: NumberOfActions, isInWindow: Bool) -> ThresholdCheckResult { syncQueue.sync { // Remove the key if is not in window guard isInWindow else { diff --git a/Sources/PixelExperimentKit/PixelExperimentKit.swift b/Sources/PixelExperimentKit/PixelExperimentKit.swift index 7577fad38..d0962f791 100644 --- a/Sources/PixelExperimentKit/PixelExperimentKit.swift +++ b/Sources/PixelExperimentKit/PixelExperimentKit.swift @@ -20,6 +20,8 @@ import PixelKit import BrowserServicesKit import Foundation +public typealias ConversionWindow = ClosedRange + struct ExperimentEvent: PixelKitEvent { var name: String var parameters: [String: String]? @@ -81,7 +83,10 @@ extension PixelKit { /// 1. Validates if the experiment is active. /// 2. Ensures the user is within the specified conversion window. /// 3. Tracks actions performed and sends the pixel once the target value is reached (if applicable). - public static func fireExperimentPixel(for subfeatureID: SubfeatureID, metric: String, conversionWindowDays: ClosedRange, value: String) { + public static func fireExperimentPixel(for subfeatureID: SubfeatureID, + metric: String, + conversionWindowDays: ConversionWindow, + value: String) { // Check is active experiment for user guard let featureFlagger = ExperimentConfig.featureFlagger else { assertionFailure("PixelKit is not configured for experiments") @@ -89,7 +94,11 @@ extension PixelKit { } guard let experimentData = featureFlagger.getAllActiveExperiments()[subfeatureID] else { return } - fireExperimentPixelForActiveExperiment(subfeatureID, experimentData: experimentData, metric: metric, conversionWindowDays: conversionWindowDays, value: value) + fireExperimentPixelForActiveExperiment(subfeatureID, + experimentData: experimentData, + metric: metric, + conversionWindowDays: conversionWindowDays, + value: value) } /// Fires search-related experiment pixels for all active experiments. @@ -99,7 +108,7 @@ extension PixelKit { /// - The value and conversion windows define when and how many search actions /// must occur before the pixel is fired. public static func fireSearchExperimentPixels() { - let valueConversionDictionary: [Int: [ClosedRange]] = [ + let valueConversionDictionary: [NumberOfActions: [ConversionWindow]] = [ 1: [0...0, 1...1, 2...2, 3...3, 4...4, 5...5, 6...6, 7...7, 5...7], 4: [5...7, 8...15], 6: [5...7, 8...15], @@ -128,7 +137,7 @@ extension PixelKit { /// - The value and conversion windows define when and how many app usage actions /// must occur before the pixel is fired. public static func fireAppRetentionExperimentPixels() { - let valueConversionDictionary: [Int: [ClosedRange]] = [ + let valueConversionDictionary: [NumberOfActions: [ConversionWindow]] = [ 1: [1...1, 2...2, 3...3, 4...4, 5...5, 6...6, 7...7, 5...7], 4: [5...7, 8...15], 6: [5...7, 8...15], @@ -154,7 +163,7 @@ extension PixelKit { for experiment: SubfeatureID, experimentData: ExperimentData, metric: String, - valueConversionDictionary: [Int: [ClosedRange]] + valueConversionDictionary: [NumberOfActions: [ConversionWindow]] ) { valueConversionDictionary.forEach { value, ranges in ranges.forEach { range in @@ -169,7 +178,11 @@ extension PixelKit { } } - private static func fireExperimentPixelForActiveExperiment(_ subfeatureID: SubfeatureID, experimentData: ExperimentData, metric: String, conversionWindowDays: ClosedRange, value: String) { + private static func fireExperimentPixelForActiveExperiment(_ subfeatureID: SubfeatureID, + experimentData: ExperimentData, + metric: String, + conversionWindowDays: ConversionWindow, + value: String) { // Set parameters, event name, store key let eventName = "\(Constants.metricsEventPrefix)_\(subfeatureID)_\(experimentData.cohortID)" let conversionWindowValue = (conversionWindowDays.lowerBound != conversionWindowDays.upperBound) ? @@ -188,7 +201,7 @@ extension PixelKit { let isInWindow = isUserInConversionWindow(conversionWindowDays, enrollmentDate: experimentData.enrollmentDate) // Check if value is a number - if let numberOfAction = Int(value), numberOfAction > 1 { + if let numberOfAction = NumberOfActions(value), numberOfAction > 1 { // Increment or remove based on conversion window status let shouldSendPixel = ExperimentConfig.eventTracker.incrementAndCheckThreshold( forKey: eventStoreKey, @@ -207,7 +220,7 @@ extension PixelKit { } private static func isUserInConversionWindow( - _ conversionWindowRange: ClosedRange, + _ conversionWindowRange: ConversionWindow, enrollmentDate: Date ) -> Bool { let calendar = Calendar.current diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index a2a67c907..1db02aedc 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -152,7 +152,7 @@ public final class PixelKit { } private var dryRun: Bool - private let source: String? + public let source: String? private let pixelCalendar: Calendar public init(dryRun: Bool, @@ -396,8 +396,11 @@ public final class PixelKit { fireRequest(pixelName, headers, parameters, allowedQueryReservedCharacters, callBackOnMainThread, onComplete) } - // Needs to be updated when fully adopted by iOS + // Only set up for macOS and for Experiments private func prefixedAndSuffixedName(for event: Event) -> String { + if event.name.hasPrefix("experiment") { + return addPlatformSuffix(to: event.name) + } if event.name.hasPrefix("m_mac_") { // Can be a debug event or not, if already prefixed the name remains unchanged return event.name @@ -408,22 +411,26 @@ public final class PixelKit { // Special kind of pixel event that don't follow the standard naming conventions return nonStandardEvent.name } else { - if let source { - switch source { - case Source.iOS.rawValue: - return "m_\(event.name)_ios_phone" - case Source.iPadOS.rawValue: - return "m_\(event.name)_ios_tablet" - case Source.macDMG.rawValue, Source.macStore.rawValue: - return "m_mac_\(event.name)" - default: - return "m_mac_\(event.name)" - } - } return "m_mac_\(event.name)" } } + private func addPlatformSuffix(to name: String) -> String { + if let source { + switch source { + case Source.iOS.rawValue: + return "\(name)_ios_phone" + case Source.iPadOS.rawValue: + return "\(name)_ios_tablet" + case Source.macStore.rawValue, Source.macDMG.rawValue: + return "\(name)_mac" + default: + return name + } + } + return name + } + public func fire(_ event: Event, frequency: Frequency = .standard, withHeaders headers: [String: String]? = nil, From 956812378687d28d045a090b06afa261e293405e Mon Sep 17 00:00:00 2001 From: Sabrina Tardio Date: Thu, 5 Dec 2024 13:32:07 +0100 Subject: [PATCH 49/49] remove unwanted change --- Sources/PixelKit/PixelKit.swift | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/Sources/PixelKit/PixelKit.swift b/Sources/PixelKit/PixelKit.swift index 1db02aedc..2e719dd26 100644 --- a/Sources/PixelKit/PixelKit.swift +++ b/Sources/PixelKit/PixelKit.swift @@ -152,7 +152,7 @@ public final class PixelKit { } private var dryRun: Bool - public let source: String? + private let source: String? private let pixelCalendar: Calendar public init(dryRun: Bool,