-
-
Notifications
You must be signed in to change notification settings - Fork 203
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
fix(3741): refactor remote-feature-flag controller with generateDeterministicRandomNumber and getMetaMetricsId #5097
Changes from 2 commits
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -14,7 +14,7 @@ import type { | |
RemoteFeatureFlagControllerStateChangeEvent, | ||
} from './remote-feature-flag-controller'; | ||
import type { FeatureFlags } from './remote-feature-flag-controller-types'; | ||
import * as userSegmentationUtils from './utils/user-segmentation-utils'; | ||
import { ClientType } from './remote-feature-flag-controller-types'; | ||
|
||
const MOCK_FLAGS: FeatureFlags = { | ||
feature1: true, | ||
|
@@ -42,7 +42,6 @@ const MOCK_FLAGS_WITH_THRESHOLD = { | |
}; | ||
|
||
const MOCK_METRICS_ID = 'f9e8d7c6-b5a4-4210-9876-543210fedcba'; | ||
const MOCK_METRICS_ID_2 = '987fcdeb-51a2-4c4b-9876-543210fedcba'; | ||
|
||
/** | ||
* Creates a controller instance with default parameters for testing | ||
|
@@ -51,6 +50,7 @@ const MOCK_METRICS_ID_2 = '987fcdeb-51a2-4c4b-9876-543210fedcba'; | |
* @param options.state - The initial controller state | ||
* @param options.clientConfigApiService - The client config API service instance | ||
* @param options.disabled - Whether the controller should start disabled | ||
* @param options.getMetaMetricsId - The function to get the metaMetricsId | ||
* @returns A configured RemoteFeatureFlagController instance | ||
*/ | ||
function createController( | ||
|
@@ -59,7 +59,7 @@ function createController( | |
state: Partial<RemoteFeatureFlagControllerState>; | ||
clientConfigApiService: AbstractClientConfigApiService; | ||
disabled: boolean; | ||
getMetaMetricsId: Promise<string | undefined>; | ||
getMetaMetricsId: () => Promise<string> | string; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Note that this function is synchronous on both extension and mobile. It's typed as asynchronous on mobile by mistake, but it's not. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Oh that's good to know. @joaoloureirop could you make adjustments in your side of implementation when initializing the controller to adapt to synchronous pattern? changes can be found: 15428da There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Thanks Mark & Danica. I'll update mobile PR |
||
}> = {}, | ||
) { | ||
return new RemoteFeatureFlagController({ | ||
|
@@ -68,7 +68,8 @@ function createController( | |
clientConfigApiService: | ||
options.clientConfigApiService ?? buildClientConfigApiService(), | ||
disabled: options.disabled, | ||
getMetaMetricsId: options.getMetaMetricsId, | ||
getMetaMetricsId: | ||
options.getMetaMetricsId ?? (() => Promise.resolve(MOCK_METRICS_ID)), | ||
}); | ||
} | ||
|
||
|
@@ -219,6 +220,7 @@ describe('RemoteFeatureFlagController', () => { | |
}); | ||
|
||
const clientConfigApiService = { | ||
getClient: () => ClientType.Mobile, | ||
fetchRemoteFeatureFlags: fetchSpy, | ||
} as AbstractClientConfigApiService; | ||
|
||
|
@@ -271,7 +273,7 @@ describe('RemoteFeatureFlagController', () => { | |
}); | ||
const controller = createController({ | ||
clientConfigApiService, | ||
getMetaMetricsId: Promise.resolve(MOCK_METRICS_ID), | ||
getMetaMetricsId: () => Promise.resolve(MOCK_METRICS_ID), | ||
}); | ||
await controller.updateRemoteFeatureFlags(); | ||
|
||
|
@@ -289,7 +291,7 @@ describe('RemoteFeatureFlagController', () => { | |
}); | ||
const controller = createController({ | ||
clientConfigApiService, | ||
getMetaMetricsId: Promise.resolve(MOCK_METRICS_ID), | ||
getMetaMetricsId: () => Promise.resolve(MOCK_METRICS_ID), | ||
}); | ||
await controller.updateRemoteFeatureFlags(); | ||
|
||
|
@@ -298,23 +300,22 @@ describe('RemoteFeatureFlagController', () => { | |
expect(nonThresholdFlags).toStrictEqual(MOCK_FLAGS); | ||
}); | ||
|
||
it('uses a fallback metaMetricsId if none is provided', async () => { | ||
jest | ||
.spyOn(userSegmentationUtils, 'generateFallbackMetaMetricsId') | ||
.mockReturnValue(MOCK_METRICS_ID_2); | ||
it('handles synchronous metaMetricsId', async () => { | ||
const clientConfigApiService = buildClientConfigApiService({ | ||
remoteFeatureFlags: MOCK_FLAGS_WITH_THRESHOLD, | ||
}); | ||
const controller = createController({ | ||
clientConfigApiService, | ||
getMetaMetricsId: () => MOCK_METRICS_ID, | ||
}); | ||
|
||
await controller.updateRemoteFeatureFlags(); | ||
|
||
expect( | ||
controller.state.remoteFeatureFlags.testFlagForThreshold, | ||
).toStrictEqual({ | ||
name: 'groupA', | ||
value: 'valueA', | ||
name: 'groupC', | ||
value: 'valueC', | ||
}); | ||
}); | ||
}); | ||
|
@@ -399,18 +400,22 @@ function getControllerMessenger( | |
* @param options.remoteFeatureFlags - Optional feature flags data to return | ||
* @param options.cacheTimestamp - Optional timestamp to use for the cache | ||
* @param options.error - Optional error to simulate API failure | ||
* @param options.client - The client type to return from getClient | ||
* @returns A mock client config API service | ||
*/ | ||
function buildClientConfigApiService({ | ||
remoteFeatureFlags, | ||
cacheTimestamp, | ||
error, | ||
client, | ||
}: { | ||
remoteFeatureFlags?: FeatureFlags; | ||
cacheTimestamp?: number; | ||
error?: Error; | ||
client?: ClientType; | ||
} = {}): AbstractClientConfigApiService { | ||
return { | ||
getClient: jest.fn().mockReturnValue(client ?? ClientType.Mobile), | ||
fetchRemoteFeatureFlags: jest.fn().mockImplementation(() => { | ||
if (error) { | ||
return Promise.reject(error); | ||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -14,7 +14,6 @@ import type { | |
import { | ||
generateDeterministicRandomNumber, | ||
isFeatureFlagWithScopeValue, | ||
generateFallbackMetaMetricsId, | ||
} from './utils/user-segmentation-utils'; | ||
|
||
// === GENERAL === | ||
|
@@ -103,7 +102,7 @@ export class RemoteFeatureFlagController extends BaseController< | |
|
||
#inProgressFlagUpdate?: Promise<ServiceResponse>; | ||
|
||
#getMetaMetricsId?: Promise<string | undefined>; | ||
#getMetaMetricsId: () => string | Promise<string>; | ||
|
||
/** | ||
* Constructs a new RemoteFeatureFlagController instance. | ||
|
@@ -127,7 +126,7 @@ export class RemoteFeatureFlagController extends BaseController< | |
messenger: RemoteFeatureFlagControllerMessenger; | ||
state?: Partial<RemoteFeatureFlagControllerState>; | ||
clientConfigApiService: AbstractClientConfigApiService; | ||
getMetaMetricsId?: Promise<string | undefined>; | ||
getMetaMetricsId: () => string | Promise<string>; | ||
fetchInterval?: number; | ||
disabled?: boolean; | ||
}) { | ||
|
@@ -209,9 +208,17 @@ export class RemoteFeatureFlagController extends BaseController< | |
remoteFeatureFlags: FeatureFlags, | ||
): Promise<FeatureFlags> { | ||
const processedRemoteFeatureFlags: FeatureFlags = {}; | ||
const metaMetricsIdResult = this.#getMetaMetricsId(); | ||
const metaMetricsId = | ||
(await this.#getMetaMetricsId) || generateFallbackMetaMetricsId(); | ||
const thresholdValue = generateDeterministicRandomNumber(metaMetricsId); | ||
metaMetricsIdResult instanceof Promise | ||
? await metaMetricsIdResult | ||
: metaMetricsIdResult; | ||
|
||
const clientType = this.#clientConfigApiService.getClient(); | ||
const thresholdValue = generateDeterministicRandomNumber( | ||
clientType, | ||
metaMetricsId, | ||
); | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Q: i don't really understand what you did here. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Yes definitely! We no longer handle undefined values - the signature doesn't actually include undefined. It's about handling both synchronous and asynchronous returns in a type-safe way. |
||
|
||
for (const [ | ||
remoteFeatureFlagName, | ||
|
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
What is this method for?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
this one is to get the client type we passed in the service configuration. Then when we can decide which algorithm we use inside
generateDeterministicRandomNumber