Skip to content

Commit

Permalink
Privacy Pro Free Trials Core Implementation (#3760)
Browse files Browse the repository at this point in the history
Task/Issue URL: https://app.asana.com/0/0/1208767141940868/f
Tech Design URL:
https://app.asana.com/0/1206488453854252/1208916720468103/f

**Description**: This PR adds the core implementation of Privacy Pro
Free Trials. Specifically:
1. Adds a `FreeTrialsFeatureFlagExperimenting` type and associated
concrete implementation
2. Adds a `Bucketer` type and associated concrete implementation 
3. Updates `SubscriptionPagesUseSubscriptionFeature` to fetch Free Trial
Subscriptions based on experiment enrollment
  • Loading branch information
aataraxiaa authored Jan 10, 2025
1 parent cb2b109 commit bac8c86
Show file tree
Hide file tree
Showing 11 changed files with 1,229 additions and 33 deletions.
6 changes: 3 additions & 3 deletions Core/FeatureFlag.swift
Original file line number Diff line number Diff line change
Expand Up @@ -60,7 +60,7 @@ public enum FeatureFlag: String {
case crashReportOptInStatusResetting

/// https://app.asana.com/0/0/1208767141940869/f
case freeTrials
case privacyProFreeTrialJan25
}

extension FeatureFlag: FeatureFlagDescribing {
Expand Down Expand Up @@ -136,8 +136,8 @@ extension FeatureFlag: FeatureFlagDescribing {
return .remoteReleasable(.feature(.adAttributionReporting))
case .crashReportOptInStatusResetting:
return .internalOnly()
case .freeTrials:
return .remoteDevelopment(.subfeature(PrivacyProSubfeature.freeTrials))
case .privacyProFreeTrialJan25:
return .remoteDevelopment(.subfeature(PrivacyProSubfeature.privacyProFreeTrialJan25))
case .aiChat:
return .remoteReleasable(.feature(.aiChat))
}
Expand Down
54 changes: 45 additions & 9 deletions DuckDuckGo.xcodeproj/project.pbxproj

Large diffs are not rendered by default.

Original file line number Diff line number Diff line change
Expand Up @@ -32,8 +32,8 @@
"kind" : "remoteSourceControl",
"location" : "https://github.com/DuckDuckGo/BrowserServicesKit",
"state" : {
"revision" : "6379f0de589e656dd715a707600824e895bd951a",
"version" : "224.4.0"
"revision" : "1700c54067b6676974676ca9a81d654317a093f1",
"version" : "224.5.0"
}
},
{
Expand Down
Original file line number Diff line number Diff line change
@@ -0,0 +1,305 @@
//
// FreeTrialsFeatureFlagExperiment.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
import BrowserServicesKit
import PixelExperimentKit
import PixelKit
import Persistence

/// A protocol that defines a method for firing experiment-related analytics pixels.
///
/// Types conforming to this protocol can be used to send experiment data (e.g., metrics and user actions).
/// This protocol is particularly useful for injecting dependencies to enable testing.
protocol ExperimentPixelFiring {
/// Fires an experiment pixel with the specified parameters.
///
/// - Parameters:
/// - subfeatureID: The unique identifier of the subfeature associated with the experiment.
/// - metric: The name of the metric being tracked (e.g., impressions, clicks, conversions).
/// - conversionWindowDays: The time range (in days) to associate the pixel with conversion events.
/// - value: A string representing the value associated with the metric, such as counts or statuses.
static func fireExperimentPixel(for subfeatureID: SubfeatureID,
metric: String,
conversionWindowDays: ConversionWindow,
value: String)
}

/// Conforming `PixelKit` to the `ExperimentPixelFiring` protocol.
///
/// `PixelKit` provides the concrete implementation for firing experiment pixels. By extending
/// `PixelKit` to conform to `ExperimentPixelFiring`, its functionality can be injected and mocked
/// for testing purposes.
extension PixelKit: ExperimentPixelFiring {}

/// Protocol defining the functionality required for a feature flag experiment related to free trials.
protocol FreeTrialsFeatureFlagExperimenting: FeatureFlagExperimentDescribing {

/// Retrieves the cohort assigned to the user for the experiment.
///
/// This method determines the cohort for the experiment if it is enabled, allowing
/// for differentiation of behavior or configurations.
///
/// - Returns: The user's cohort, or `nil` if the experiment is not enabled.
func getCohortIfEnabled() -> (any FlagCohort)?

/// Provides experiment-specific parameters if applicable.
///
/// This method returns parameters associated with the experiment and cohort, based on
/// certain criteria. Implementations can use this to provide experiment-specific data
/// to be used for analytics or feature configuration.
///
/// - Parameter cohort: The cohort assigned to the user.
/// - Returns: A dictionary of experiment-specific parameters, or `nil` if parameters
/// are not applicable.
func freeTrialParametersIfNotPreviouslyReturned(for cohort: any FlagCohort) -> [String: String]?

/// Increments the count of paywall views if the user's enrollment date is within the conversion window.
func incrementPaywallViewCountIfWithinConversionWindow()

/// Fires a pixel tracking the impression of the paywall.
func firePaywallImpressionPixel()

/// Fires a pixel when the monthly subscription offer is selected.
func fireOfferSelectionMonthlyPixel()

/// Fires a pixel when the yearly subscription offer is selected.
func fireOfferSelectionYearlyPixel()

/// Fires a pixel when a monthly subscription is started.
func fireSubscriptionStartedMonthlyPixel()

/// Fires a pixel when a yearly subscription is started.
func fireSubscriptionStartedYearlyPixel()
}

/// Implementation of a feature flag experiment for monitoring and optimizing the impact of free trial offers.
final class FreeTrialsFeatureFlagExperiment: FreeTrialsFeatureFlagExperimenting {

/// Represents the cohorts in the experiment.
typealias CohortType = Cohort
enum Cohort: String, FlagCohort {
/// Control cohort with no changes applied.
case control
/// Treatment cohort where the experiment modifications are applied.
case treatment
}

/// Constants used in the experiment.
enum Constants {
/// Unique identifier for the subfeature being tested.
static let subfeatureIdentifier = "privacyProFreeTrialJan25"

/// Metric identifiers for various user actions during the experiment.
static let metricPaywallImpressions = "paywallImpressions"
static let metricStartClickedMonthly = "startClickedMonthly"
static let metricStartClickedYearly = "startClickedYearly"
static let metricSubscriptionStartedMonthly = "subscriptionStartedMonthly"
static let metricSubscriptionStartedYearly = "subscriptionStartedYearly"

/// Conversion window in days for tracking user actions.
static let conversionWindowDays = 0...3

/// Key used to store the paywall view count in persistent storage.
static let paywallViewCountKey = "\(subfeatureIdentifier)_paywallViewCount"
static let hasReturnedFreeTrialParametersKey = "\(subfeatureIdentifier)_hasReturnedFreeTrialParameters"

/// Experiment name key included in the free trial parameters.
static let freeTrialParameterExperimentName = "experimentName"

/// Cohort value key included in the free trial parameters.
static let freeTrialParameterExperimentCohort = "experimentCohort"

/// Key used to store the override status for the feature flag in persistent storage.
static let featureFlagOverrideKey = "\(subfeatureIdentifier)_featureFlagOverride"
}

/// Identifier for the experiment.
let rawValue = Constants.subfeatureIdentifier

/// Source of the feature flag, defining how it is retrieved and enabled.
let source: FeatureFlagSource = .remoteReleasable(.subfeature(PrivacyProSubfeature.privacyProFreeTrialJan25))

/// Persistent storage for experiment-related data.
private let storage: KeyValueStoring

/// A type responsible for firing experiment-related analytics pixels.
private let experimentPixelFirer: ExperimentPixelFiring.Type

/// A bucketer responsible for categorizing values into predefined ranges.
private let bucketer: any Bucketer

/// A feature flagging service for managing feature flag experiments.
private let featureFlagger: FeatureFlagger

/// Initializes the experiment with the specified storage.
/// - Parameter storage: The persistent storage to use. Defaults to `UserDefaults.standard`.
init(storage: KeyValueStoring = UserDefaults.standard,
experimentPixelFirer: ExperimentPixelFiring.Type = PixelKit.self,
bucketer: any Bucketer = PaywallViewBucketer(),
featureFlagger: FeatureFlagger = AppDependencyProvider.shared.featureFlagger) {
self.storage = storage
self.experimentPixelFirer = experimentPixelFirer
self.bucketer = bucketer
self.featureFlagger = featureFlagger
}

/// Retrieves the cohort associated with the experiment if the feature flag is enabled or if the override is active.
///
/// This method determines whether the feature flag for the experiment is enabled
/// or overridden. If the override flag stored in persistent storage is set to `true`,
/// it forces the method to return the `treatment` cohort regardless of the feature flag's actual state.
/// If no override is active, it checks the feature flag and returns the cohort assigned to the user
/// based on the feature flag configuration.
///
/// - Returns: The `treatment` cohort if the override is enabled, the assigned cohort if the feature flag is enabled, or `nil` otherwise.
func getCohortIfEnabled() -> (any FlagCohort)? {
let isFlagOverrideEnabled = storage.object(forKey: Constants.featureFlagOverrideKey) as? Bool ?? false
if isFlagOverrideEnabled {
return FreeTrialsFeatureFlagExperiment.Cohort.treatment
}

return featureFlagger.getCohortIfEnabled(for: self)
as? FreeTrialsFeatureFlagExperiment.Cohort
}

/// Provides free trial experiment parameters if they haven't been returned before.
///
/// This method checks whether the experiment parameters for the user's cohort have already been returned.
/// If not, it returns a dictionary of parameters containing the experiment name and cohort.
/// If the user is outside the conversion window, `_outside` is appended to the cohort name.
/// This ensures parameters are provided only once per user.
///
/// - Parameter cohort: The cohort to which the user is assigned.
/// - Returns: A dictionary containing the experiment name and cohort, or `nil` if the parameters
/// have already been returned.
func freeTrialParametersIfNotPreviouslyReturned(for cohort: any FlagCohort) -> [String: String]? {
let hasReturnedParameters = storage.object(forKey: Constants.hasReturnedFreeTrialParametersKey) as? Bool ?? false

// Return parameters only if they haven't been returned before
guard !hasReturnedParameters else {
return nil
}

storage.set(true, forKey: Constants.hasReturnedFreeTrialParametersKey)

let cohortValue: String
if userIsInConversionWindow {
cohortValue = cohort.rawValue
} else {
cohortValue = "\(cohort.rawValue)_outside"
}

return [
Constants.freeTrialParameterExperimentName: rawValue,
Constants.freeTrialParameterExperimentCohort: cohortValue
]
}

/// Increments the count of paywall views if the user's enrollment date is within the conversion window.
func incrementPaywallViewCountIfWithinConversionWindow() {
guard userIsInConversionWindow else { return }
paywallViewCount += 1
}

/// Fires a pixel tracking the impression of the paywall.
func firePaywallImpressionPixel() {
let bucket = bucketer.bucket(for: paywallViewCount)
experimentPixelFirer.fireExperimentPixel(for: Constants.subfeatureIdentifier,
metric: Constants.metricPaywallImpressions,
conversionWindowDays: Constants.conversionWindowDays,
value: bucket)
}

/// Fires a pixel when the monthly subscription offer is selected.
func fireOfferSelectionMonthlyPixel() {
let bucket = bucketer.bucket(for: paywallViewCount)
experimentPixelFirer.fireExperimentPixel(for: Constants.subfeatureIdentifier,
metric: Constants.metricStartClickedMonthly,
conversionWindowDays: Constants.conversionWindowDays,
value: bucket)
}

/// Fires a pixel when the yearly subscription offer is selected.
func fireOfferSelectionYearlyPixel() {
let bucket = bucketer.bucket(for: paywallViewCount)
experimentPixelFirer.fireExperimentPixel(for: Constants.subfeatureIdentifier,
metric: Constants.metricStartClickedYearly,
conversionWindowDays: Constants.conversionWindowDays,
value: bucket)
}

/// Fires a pixel when a monthly subscription is started.
func fireSubscriptionStartedMonthlyPixel() {
let bucket = bucketer.bucket(for: paywallViewCount)
experimentPixelFirer.fireExperimentPixel(for: Constants.subfeatureIdentifier,
metric: Constants.metricSubscriptionStartedMonthly,
conversionWindowDays: Constants.conversionWindowDays,
value: bucket)
}

/// Fires a pixel when a yearly subscription is started.
func fireSubscriptionStartedYearlyPixel() {
let bucket = bucketer.bucket(for: paywallViewCount)
experimentPixelFirer.fireExperimentPixel(for: Constants.subfeatureIdentifier,
metric: Constants.metricSubscriptionStartedYearly,
conversionWindowDays: Constants.conversionWindowDays,
value: bucket)
}
}

private extension FreeTrialsFeatureFlagExperiment {
/// Computed property for managing the paywall view count in persistent storage.
var paywallViewCount: Int {
get {
storage.object(forKey: Constants.paywallViewCountKey) as? Int ?? 0
}
set {
storage.set(newValue, forKey: Constants.paywallViewCountKey)
}
}

/// Determines if the user is within the conversion window for the experiment.
var userIsInConversionWindow: Bool {
guard let enrollmentDate = featureFlagger.getAllActiveExperiments()[rawValue]?.enrollmentDate else {
return false
}

let startOfWindow = enrollmentDate.addingDays(Constants.conversionWindowDays.lowerBound)
let endOfWindow = enrollmentDate.addingDays(Constants.conversionWindowDays.upperBound)

let today = Date().startOfDay()
return today >= startOfWindow && today <= endOfWindow
}
}

private extension Date {
/// Returns a new `Date` by adding the specified number of days to the current date.
/// - Parameter days: The number of days to add. Negative values subtract days.
/// - Returns: A new `Date` instance.
func addingDays(_ days: Int) -> Date {
Calendar.current.date(byAdding: .day, value: days, to: self) ?? self
}

/// Returns the start of the day for the current date.
/// - Returns: A `Date` representing the beginning of the day.
func startOfDay() -> Date {
Calendar.current.startOfDay(for: self)
}
}
Original file line number Diff line number Diff line change
@@ -0,0 +1,54 @@
//
// PaywallViewBucketer.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

/// A protocol for defining a bucketing system.
/// - Note: The value to bucket is now explicitly an `Int`, removing the need for an associated type.
protocol Bucketer {
/// Determines the bucket for the given value.
/// - Parameter value: The integer value to bucket.
/// - Returns: A string label representing the bucket the value belongs to.
func bucket(for value: Int) -> String
}

/// An implementation of the `Bucketer` protocol for bucketing paywall view counts.
struct PaywallViewBucketer: Bucketer {
/// A list of bucket ranges and their corresponding labels.
private let buckets: [(range: ClosedRange<Int>, label: String)] = [
(1...1, "1"),
(2...2, "2"),
(3...3, "3"),
(4...4, "4"),
(5...5, "5"),
(6...10, "6-10"),
(11...50, "11-50"),
(51...Int.max, "51+")
]

/// Finds the bucket label for a given value.
/// - Parameter value: The number of paywall views.
/// - Returns: A string label representing the bucket, or "Unknown" if no bucket matches.
func bucket(for value: Int) -> String {
for bucket in buckets where bucket.range.contains(value) {
return bucket.label
}
return "Unknown"
}
}
Loading

0 comments on commit bac8c86

Please sign in to comment.