Skip to content
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

feat: add custom context fields to Unleash #11730

Open
wants to merge 1 commit into
base: main
Choose a base branch
from

Conversation

dblandin
Copy link
Member

@dblandin dblandin commented Mar 19, 2025

Description

This is meant as a conversation starter – not something on the critical path for feature development.

Reflecting on recent Unleash-backed A/B experiments, it would be nice to update the targeting for an experiment so that a certain group is always given the experiment variant. For example the members of a specific product team or all Artsy staff members (as identified via the team role).

We've recently built some handling into Metaphysics (artsy/metaphysics#6297 and artsy/metaphysics#6465) to support returning Home View content only when the client is requesting from a supported app version. It seems like we can do similar targeting for Unleash feature flags (including experiments), if we pass along an appVersion context field.

  • appPlatformOS: "ios" | "android" | "web" | # to allow targeting based on platform
  • appVersion: "8.67.0" # to allow targeting based on semVer constraints
  • userRoles: "user,team" # to allow targeting based on user role (for example, rolling out to all Artsy staff)

The updated API request params to the Unleash backend look a bit like this:

sessionId: 128917516
appName: eigen
environment: default
properties["appPlatformOS"]: ios
properties["userRoles"]: user,team
userId: <user ID>

PR Checklist

  • I have tested my changes on iOS and Android.
  • I hid my changes behind a feature flag, or they don't need one.
  • I have included screenshots or videos, or I have not changed the UI.
  • I have added tests, or my changes don't require any.
  • I added an app state migration, or my changes do not require one.
  • I have documented any follow-up work that this PR will require, or it does not require any.
  • I have added a changelog entry below, or my changes do not require one.

To the reviewers 👀

  • I would like at least one of the reviewers to run this PR on the simulator or device.
Changelog updates

Changelog updates

Cross-platform user-facing changes

iOS user-facing changes

Android user-facing changes

Dev changes

  • feat: add custom context fields to Unleash

Need help with something? Have a look at our docs, or get in touch with us.

- appPlatformOS: "ios" | "android" | "web" | # to allow targeting based on platform
- appVersion: "8.67.0" # to allow targeting based on semVer constraints
- userRoles: "user,team" # to allow targeting based on user role (for example, rolling out to all Artsy staff)
@dblandin dblandin self-assigned this Mar 19, 2025
@@ -11,6 +13,12 @@ export const WrappedFlagProvider: React.FC = ({ children }) => {
const storageName = (name: string) => `unleash-values:${name}`

const config: IConfig = {
context: {
properties: {
appVersion: DeviceInfo.getVersion(),
Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Should we be using the version attribute within app.json instead? Looks we use that when generating our user agent string. When running in a simulator, DeviceInfo.getVersion() doesn't properly return a known version number but we still use this utility function in a few places, including the About screen.

Copy link
Contributor

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

DeviceInfo.getVersion() retrieves the version of the app installed from the store. That's probably why we don't see a correct version when running in a simulator.

Do we use version number from app.json anywhere?

Copy link
Member Author

@dblandin dblandin Mar 20, 2025

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We're currently using the app.json value to construct the user agent string (source). Maybe not very significant before but we're now using that value in Metaphysics to alter the homeView content returned.

Copy link
Member Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

FYI – I opened another PR to bring more consistency to how we determine the current app version, preferring the app.json value based on this Slack conversation.

Holding on this change until that PR is resolved – then I'll rebase here.

Copy link
Member

@MounirDhahri MounirDhahri left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I am definitely in for more granularity with Unleash. In many situations it would have been great to target a specific target group instead of injecting user ids. Do you know how we can set such groups/strategies? does unleash allow for that

Copy link
Member

@anandaroop anandaroop left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

I like it!

Is the idea that the custom userRoles context variable being set here would show up in the Unleash targeting UI?

Screenshot 2025-03-19 at 8 53 51 PM

@dblandin
Copy link
Member Author

I like it!

Is the idea that the custom userRoles context variable being set here would show up in the Unleash targeting UI?

Screenshot 2025-03-19 at 8 53 51 PM

Yes, exactly. You have to define these properties in the Unleash admin UI for them to show up as targeting options. And while Unleash doesn't directly support arrays of strings, we can use a comma-separated value in an "includes" clause for role targeting.

Also interesting to find that they have specific semVer constraints that you can apply to a semVer version value.

@dblandin
Copy link
Member Author

I am definitely in for more granularity with Unleash. In many situations it would have been great to target a specific target group instead of injecting user ids. Do you know how we can set such groups/strategies? does unleash allow for that

I know you can define segments in the Unleash UI which can persist a group of constraints and give it a name. So for Onyx, we can define a named Onyx segment with those userIDs preconfigured.

But with user roles we can also define an "Artsy Staff" segment.

@gkartalis
Copy link
Member

I am in favour of this approach - will be really useful

@dblandin dblandin marked this pull request as ready for review March 20, 2025 12:56
@ArtsyOpenSource
Copy link
Contributor

Warnings
⚠️

An error occurred while validating your changelog, please make sure you provided a valid changelog.

Generated by 🚫 dangerJS against 4124593

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment
Labels
None yet
Projects
None yet
Development

Successfully merging this pull request may close these issues.

6 participants