457 lines
20 KiB
Markdown
457 lines
20 KiB
Markdown
---
|
|
stage: Release
|
|
group: Release
|
|
info: To determine the technical writer assigned to the Stage/Group associated with this page, see https://about.gitlab.com/handbook/product/ux/technical-writing/#assignments
|
|
---
|
|
|
|
# Feature flags **(FREE)**
|
|
|
|
> [Moved](https://gitlab.com/gitlab-org/gitlab/-/issues/212318) from GitLab Premium to GitLab Free in 13.5.
|
|
|
|
With feature flags, you can deploy your application's new features to production in smaller batches.
|
|
You can toggle a feature on and off to subsets of users, helping you achieve Continuous Delivery.
|
|
Feature flags help reduce risk, allowing you to do controlled testing, and separate feature
|
|
delivery from customer launch.
|
|
|
|
<i class="fa fa-youtube-play youtube" aria-hidden="true"></i>
|
|
For an example of feature flags in action, see [GitLab for deploys, feature flags, and error tracking](https://www.youtube.com/watch?v=5tw2p6lwXxo).
|
|
|
|
NOTE:
|
|
To contribute to the development of the GitLab product, view
|
|
[this feature flag content](../development/feature_flags/index.md) instead.
|
|
|
|
## How it works
|
|
|
|
GitLab uses [Unleash](https://github.com/Unleash/unleash), a feature
|
|
toggle service.
|
|
|
|
By enabling or disabling a flag in GitLab, your application
|
|
can determine which features to enable or disable.
|
|
|
|
You can create feature flags in GitLab and use the API from your application
|
|
to get the list of feature flags and their statuses. The application must be configured to communicate
|
|
with GitLab, so it's up to developers to use a compatible client library and
|
|
[integrate the feature flags in your app](#integrate-feature-flags-with-your-application).
|
|
|
|
## Create a feature flag
|
|
|
|
To create and enable a feature flag:
|
|
|
|
1. On the top bar, select **Main menu > Projects** and find your project.
|
|
1. On the left sidebar, select **Deployments > Feature Flags**.
|
|
1. Select **New feature flag**.
|
|
1. Enter a name that starts with a letter and contains only lowercase letters, digits, underscores (`_`),
|
|
or dashes (`-`), and does not end with a dash (`-`) or underscore (`_`).
|
|
1. Optional. Enter a description (255 characters maximum).
|
|
1. Add feature flag [**Strategies**](#feature-flag-strategies) to define how the flag should be applied. For each strategy, include the **Type** (defaults to [**All users**](#all-users))
|
|
and **Environments** (defaults to all environments).
|
|
1. Select **Create feature flag**.
|
|
|
|
To change these settings, select **Edit** (**{pencil}**).
|
|
next to any feature flag in the list.
|
|
|
|
## Maximum number of feature flags
|
|
|
|
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/254379) in GitLab 13.5.
|
|
|
|
The maximum number of feature flags per project on self-managed GitLab instances
|
|
is 200. For GitLab SaaS, the maximum number is determined by [tier](https://about.gitlab.com/pricing/):
|
|
|
|
| Tier | Feature flags per project (SaaS) | Feature flags per project (self-managed) |
|
|
|----------|----------------------------------|------------------------------------------|
|
|
| Free | 50 | 200 |
|
|
| Premium | 150 | 200 |
|
|
| Ultimate | 200 | 200 |
|
|
|
|
## Feature flag strategies
|
|
|
|
> - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/35555) in GitLab 13.0.
|
|
> - It was deployed behind a feature flag, disabled by default.
|
|
> - It became [enabled by default](https://gitlab.com/gitlab-org/gitlab/-/issues/214684) in GitLab 13.2.
|
|
> - It's recommended for production use.
|
|
> - It's enabled on GitLab.com.
|
|
|
|
You can apply a feature flag strategy across multiple environments, without defining
|
|
the strategy multiple times.
|
|
|
|
GitLab feature flags use [Unleash](https://docs.getunleash.io/) as the feature flag
|
|
engine. In Unleash, there are [strategies](https://docs.getunleash.io/user_guide/activation_strategy)
|
|
for granular feature flag controls. GitLab feature flags can have multiple strategies,
|
|
and the supported strategies are:
|
|
|
|
- [All users](#all-users)
|
|
- [Percent of Users](#percent-of-users)
|
|
- [User IDs](#user-ids)
|
|
- [User List](#user-list)
|
|
|
|
Strategies can be added to feature flags when [creating a feature flag](#create-a-feature-flag),
|
|
or by editing an existing feature flag after creation by navigating to **Deployments > Feature Flags**
|
|
and selecting **Edit** (**{pencil}**).
|
|
|
|
### All users
|
|
|
|
Enables the feature for all users. It uses the Standard (`default`) Unleash activation [strategy](https://docs.getunleash.io/user_guide/activation_strategy#standard).
|
|
|
|
### Percent Rollout
|
|
|
|
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/merge_requests/43340) in GitLab 13.5.
|
|
|
|
Enables the feature for a percentage of page views, with configurable consistency
|
|
of behavior. This consistency is also known as stickiness. It uses the
|
|
Gradual Rollout (`flexibleRollout`) Unleash activation [strategy](https://docs.getunleash.io/user_guide/activation_strategy#gradual-rollout).
|
|
|
|
You can configure the consistency to be based on:
|
|
|
|
- **User IDs**: Each user ID has a consistent behavior, ignoring session IDs.
|
|
- **Session IDs**: Each session ID has a consistent behavior, ignoring user IDs.
|
|
- **Random**: Consistent behavior is not guaranteed. The feature is enabled for the
|
|
selected percentage of page views randomly. User IDs and session IDs are ignored.
|
|
- **Available ID**: Consistent behavior is attempted based on the status of the user:
|
|
- If the user is logged in, make behavior consistent based on user ID.
|
|
- If the user is anonymous, make the behavior consistent based on the session ID.
|
|
- If there is no user ID or session ID, then the feature is enabled for the selected
|
|
percentage of page view randomly.
|
|
|
|
For example, set a value of 15% based on **Available ID** to enable the feature for 15% of page views. For
|
|
authenticated users this is based on their user ID. For anonymous users with a session ID it would be based on their
|
|
session ID instead as they do not have a user ID. Then if no session ID is provided, it falls back to random.
|
|
|
|
The rollout percentage can be from 0% to 100%.
|
|
|
|
Selecting a consistency based on User IDs functions the same as the [percent of Users](#percent-of-users) rollout.
|
|
|
|
WARNING:
|
|
Selecting **Random** provides inconsistent application behavior for individual users.
|
|
|
|
### Percent of Users
|
|
|
|
Enables the feature for a percentage of authenticated users. It uses the Unleash activation strategy
|
|
[`gradualRolloutUserId`](https://docs.getunleash.io/user_guide/activation_strategy#gradual-rollout).
|
|
|
|
For example, set a value of 15% to enable the feature for 15% of authenticated users.
|
|
|
|
The rollout percentage can be from 0% to 100%.
|
|
|
|
Stickiness (consistent application behavior for the same user) is guaranteed for authenticated users,
|
|
but not anonymous users.
|
|
|
|
Note that [percent rollout](#percent-rollout) with a consistency based on **User IDs** has the same
|
|
behavior. We recommend using percent rollout because it's more flexible than percent of users
|
|
|
|
WARNING:
|
|
If the percent of users strategy is selected, then the Unleash client **must** be given a user
|
|
ID for the feature to be enabled. See the [Ruby example](#ruby-application-example) below.
|
|
|
|
### User IDs
|
|
|
|
> - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/8240) in GitLab 12.2.
|
|
> - [Updated](https://gitlab.com/gitlab-org/gitlab/-/issues/34363) to be defined per environment in GitLab 12.6.
|
|
|
|
Enables the feature for a list of target users. It is implemented
|
|
using the Unleash UserIDs (`userWithId`) activation [strategy](https://docs.getunleash.io/user_guide/activation_strategy#userids).
|
|
|
|
Enter user IDs as a comma-separated list of values (for example,
|
|
`user@example.com, user2@example.com`, or `username1,username2,username3`, and so on). Note that
|
|
user IDs are identifiers for your application users. They do not need to be GitLab users.
|
|
|
|
WARNING:
|
|
The Unleash client **must** be given a user ID for the feature to be enabled for
|
|
target users. See the [Ruby example](#ruby-application-example) below.
|
|
|
|
### User List
|
|
|
|
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/35930) in GitLab 13.1.
|
|
|
|
Enables the feature for lists of users created [in the feature flags UI](#create-a-user-list), or with the [feature flag user list API](../api/feature_flag_user_lists.md).
|
|
Similar to [User IDs](#user-ids), it uses the Unleash UsersIDs (`userWithId`) activation [strategy](https://docs.getunleash.io/user_guide/activation_strategy#userids).
|
|
|
|
It's not possible to *disable* a feature for members of a user list, but you can achieve the same
|
|
effect by enabling a feature for a user list that doesn't contain the excluded users.
|
|
|
|
For example:
|
|
|
|
- `Full-user-list` = `User1A, User1B, User2A, User2B, User3A, User3B, ...`
|
|
- `Full-user-list-excluding-B-users` = `User1A, User2A, User3A, ...`
|
|
|
|
#### Create a user list
|
|
|
|
> - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/13308) in GitLab 13.3.
|
|
> - [Updated](https://gitlab.com/gitlab-org/gitlab/-/issues/322425) in GitLab 14.0.
|
|
|
|
To create a user list:
|
|
|
|
1. On the top bar, select **Main menu > Projects** and find your project.
|
|
1. On the left sidebar, select **Deployments > Feature Flags**.
|
|
1. Select **View user lists**
|
|
1. Select **New user list**.
|
|
1. Enter a name for the list.
|
|
1. Select **Create**.
|
|
|
|
You can view a list's User IDs by selecting **Edit** (**{pencil}**) next to it.
|
|
When viewing a list, you can rename it by selecting **Edit** (**{pencil}**).
|
|
|
|
#### Add users to a user list
|
|
|
|
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/13308) in GitLab 13.3.
|
|
|
|
To add users to a user list:
|
|
|
|
1. On the top bar, select **Main menu > Projects** and find your project.
|
|
1. On the left sidebar, select **Deployments > Feature Flags**.
|
|
1. Select **Edit** (**{pencil}**) next to the list you want to add users to.
|
|
1. Select **Add Users**.
|
|
1. Enter the user IDs as a comma-separated list of values. For example,
|
|
`user@example.com, user2@example.com`, or `username1,username2,username3`, and so on.
|
|
1. Select **Add**.
|
|
|
|
#### Remove users from a user list
|
|
|
|
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/13308) in GitLab 13.3.
|
|
|
|
To remove users from a user list:
|
|
|
|
1. On the top bar, select **Main menu > Projects** and find your project.
|
|
1. On the left sidebar, select **Deployments > Feature Flags**.
|
|
1. Select **Edit** (**{pencil}**) next to the list you want to change.
|
|
1. Select **Remove** (**{remove}**) next to the ID you want to remove.
|
|
|
|
## Search for Code References **(PREMIUM)**
|
|
|
|
> [Introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/300299) in GitLab 14.4.
|
|
|
|
Search your project and find any references of a feature flag in your
|
|
code so that you can clean it up when it's time to remove the feature flag.
|
|
|
|
To search for code references of a feature flag:
|
|
|
|
1. On the top bar, select **Main menu > Projects** and find your project.
|
|
1. On the left sidebar, select **Deployments > Feature Flags**.
|
|
1. Edit the feature flag you want to remove.
|
|
1. Select **More actions** (**{ellipsis_v}**).
|
|
1. Select **Search code references**.
|
|
|
|
## Disable a feature flag for a specific environment
|
|
|
|
In [GitLab 13.0 and earlier](https://gitlab.com/gitlab-org/gitlab/-/issues/8621),
|
|
to disable a feature flag for a specific environment:
|
|
|
|
1. On the top bar, select **Main menu > Projects** and find your project.
|
|
1. On the left sidebar, select **Deployments > Feature Flags**.
|
|
1. For the feature flag you want to disable, select **Edit** (**{pencil}**).
|
|
1. To disable the flag:
|
|
|
|
- In GitLab 13.0 and earlier: Slide the Status toggle for the environment. Or, to delete the
|
|
environment spec, on the right, select **Remove (X)**.
|
|
- In GitLab 13.1 and later: For each strategy it applies to, under **Environments**, delete the environment.
|
|
|
|
1. Select **Save changes**.
|
|
|
|
## Disable a feature flag for all environments
|
|
|
|
To disable a feature flag for all environments:
|
|
|
|
1. On the top bar, select **Main menu > Projects** and find your project.
|
|
1. On the left sidebar, select **Deployments > Feature Flags**.
|
|
1. For the feature flag you want to disable, slide the Status toggle to **Disabled**.
|
|
|
|
The feature flag is displayed on the **Disabled** tab.
|
|
|
|
## Integrate feature flags with your application
|
|
|
|
To use feature flags with your application, get access credentials from GitLab.
|
|
Then prepare your application with a client library.
|
|
|
|
### Get access credentials
|
|
|
|
To get the access credentials that your application needs to communicate with GitLab:
|
|
|
|
1. On the top bar, select **Main menu > Projects** and find your project.
|
|
1. On the left sidebar, select **Deployments > Feature Flags**.
|
|
1. Select **Configure** to view the following:
|
|
- **API URL**: URL where the client (application) connects to get a list of feature flags.
|
|
- **Instance ID**: Unique token that authorizes the retrieval of the feature flags.
|
|
- **Application name**: The name of the *environment* the application runs in
|
|
(not the name of the application itself).
|
|
|
|
For example, if the application runs for a production server, the **Application name**
|
|
could be `production` or similar. This value is used for the environment spec evaluation.
|
|
|
|
Note that the meaning of these fields might change over time. For example, we're not sure if
|
|
**Instance ID** is a single token or multiple tokens, assigned to the **Environment**. Also,
|
|
**Application name** could describe the application version instead of the running environment.
|
|
|
|
### Choose a client library
|
|
|
|
GitLab implements a single backend that is compatible with Unleash clients.
|
|
|
|
With the Unleash client, developers can define, in the application code, the default values for flags.
|
|
Each feature flag evaluation can express the desired outcome if the flag isn't present in the
|
|
provided configuration file.
|
|
|
|
Unleash currently [offers many SDKs for various languages and frameworks](https://github.com/Unleash/unleash#client-implementations).
|
|
|
|
### Feature flags API information
|
|
|
|
For API content, see:
|
|
|
|
- [Feature flags API](../api/feature_flags.md)
|
|
- [Feature flag specs API](../api/feature_flag_specs.md) (Deprecated and [scheduled for removal](https://gitlab.com/gitlab-org/gitlab/-/issues/213369) in GitLab 14.0.)
|
|
- [Feature flag user lists API](../api/feature_flag_user_lists.md)
|
|
|
|
### Golang application example
|
|
|
|
Here's an example of how to integrate feature flags in a Golang application:
|
|
|
|
```golang
|
|
package main
|
|
|
|
import (
|
|
"io"
|
|
"log"
|
|
"net/http"
|
|
|
|
"github.com/Unleash/unleash-client-go/v3"
|
|
)
|
|
|
|
type metricsInterface struct {
|
|
}
|
|
|
|
func init() {
|
|
unleash.Initialize(
|
|
unleash.WithUrl("https://gitlab.com/api/v4/feature_flags/unleash/42"),
|
|
unleash.WithInstanceId("29QmjsW6KngPR5JNPMWx"),
|
|
unleash.WithAppName("production"), // Set to the running environment of your application
|
|
unleash.WithListener(&metricsInterface{}),
|
|
)
|
|
}
|
|
|
|
func helloServer(w http.ResponseWriter, req *http.Request) {
|
|
if unleash.IsEnabled("my_feature_name") {
|
|
io.WriteString(w, "Feature enabled\n")
|
|
} else {
|
|
io.WriteString(w, "hello, world!\n")
|
|
}
|
|
}
|
|
|
|
func main() {
|
|
http.HandleFunc("/", helloServer)
|
|
log.Fatal(http.ListenAndServe(":8080", nil))
|
|
}
|
|
```
|
|
|
|
### Ruby application example
|
|
|
|
Here's an example of how to integrate feature flags in a Ruby application.
|
|
|
|
The Unleash client is given a user ID for use with a **Percent rollout (logged in users)** rollout strategy or a list of **Target Users**.
|
|
|
|
```ruby
|
|
#!/usr/bin/env ruby
|
|
|
|
require 'unleash'
|
|
require 'unleash/context'
|
|
|
|
unleash = Unleash::Client.new({
|
|
url: 'http://gitlab.com/api/v4/feature_flags/unleash/42',
|
|
app_name: 'production', # Set to the running environment of your application
|
|
instance_id: '29QmjsW6KngPR5JNPMWx'
|
|
})
|
|
|
|
unleash_context = Unleash::Context.new
|
|
# Replace "123" with the ID of an authenticated user.
|
|
# Note that the context's user ID must be a string:
|
|
# https://unleash.github.io/docs/unleash_context
|
|
unleash_context.user_id = "123"
|
|
|
|
if unleash.is_enabled?("my_feature_name", unleash_context)
|
|
puts "Feature enabled"
|
|
else
|
|
puts "hello, world!"
|
|
end
|
|
```
|
|
|
|
### Unleash Proxy example
|
|
|
|
As of [Unleash Proxy](https://docs.getunleash.io/sdks/unleash-proxy) version
|
|
0.2, the proxy is compatible with feature flags. To run a Docker container to
|
|
connect to your project's feature flags, run the following command:
|
|
|
|
```shell
|
|
docker run \
|
|
-e UNLEASH_PROXY_SECRETS=<secret> \
|
|
-e UNLEASH_URL=<project feature flags URL> \
|
|
-e UNLEASH_INSTANCE_ID=<project feature flags instance ID> \
|
|
-e UNLEASH_APP_NAME=<project environment> \
|
|
-e UNLEASH_API_TOKEN=<tokenNotUsed> \
|
|
-p 3000:3000 \
|
|
unleashorg/unleash-proxy
|
|
```
|
|
|
|
| Variable | Value |
|
|
| --------------------------- | ------------------------------------------------------------------------------------------------------------------------------------ |
|
|
| `UNLEASH_PROXY_SECRETS` | Shared secret used to configure an [Unleash Proxy client](https://docs.getunleash.io/sdks/unleash-proxy#how-to-connect-to-the-proxy). |
|
|
| `UNLEASH_URL` | Your project's API URL. For more details, read [Get access credentials](#get-access-credentials). |
|
|
| `UNLEASH_INSTANCE_ID` | Your project's Instance ID. For more details, read [Get access credentials](#get-access-credentials). |
|
|
| `UNLEASH_APP_NAME` | The name of the environment the application runs in. For more details, read [Get access credentials](#get-access-credentials). |
|
|
| `UNLEASH_API_TOKEN` | Required to start the Unleash Proxy, but not used to connect to GitLab. Can be set to any value. |
|
|
|
|
There is a limitation when using the Unleash Proxy where each proxy instance can request flags only for the environment named in `UNLEASH_APP_NAME`. The Proxy sends
|
|
this to GitLab on behalf of the client, which means the client can't override it.
|
|
|
|
## Feature flag related issues **(PREMIUM)**
|
|
|
|
> - [Introduced](https://gitlab.com/gitlab-org/gitlab/-/merge_requests/36617) in GitLab 13.2.
|
|
> - [Feature flag removed](https://gitlab.com/gitlab-org/gitlab/-/issues/251234) in GitLab 13.5.
|
|
> - Showing related feature flags in issues [introduced](https://gitlab.com/gitlab-org/gitlab/-/issues/220333) in GitLab 14.1.
|
|
|
|
You can link related issues to a feature flag. In the feature flag **Linked issues** section,
|
|
select the `+` button and input the issue reference number or the full URL of the issue.
|
|
The issues then appear in the related feature flag and the other way round.
|
|
|
|
This feature is similar to the [linked issues](../user/project/issues/related_issues.md) feature.
|
|
|
|
## Performance factors
|
|
|
|
In general, GitLab feature flags can be used in any applications,
|
|
however, if it's a large application, it could require an additional configuration in advance.
|
|
This section explains the performance factors to help your organization to identify
|
|
what's needed to be done before using the feature.
|
|
Please read [How it works](#how-it-works) section before diving into the details.
|
|
|
|
### Maximum supported clients in application nodes
|
|
|
|
GitLab accepts client requests as much as possible until it hits the [rate limiting](../security/rate_limits.md).
|
|
At the moment, the feature flag API falls into **Unauthenticated traffic (from a given IP address)**
|
|
in the [GitLab.com specific limits](../user/gitlab_com/index.md),
|
|
so it's **500 requests per minute**.
|
|
|
|
The polling rate is configurable in SDKs. Provided that all clients are requesting from the same IP:
|
|
|
|
- Request once per minute ... 500 clients can be supported.
|
|
- Request once per 15 sec ... 125 clients can be supported.
|
|
|
|
For applications looking for more scalable solution, we recommend to use [Unleash Proxy](#unleash-proxy-example).
|
|
This proxy server sits between the server and clients. It requests to the server as a behalf of the client groups,
|
|
so the number of outbound requests can be greatly reduced.
|
|
|
|
There is also an [issue](https://gitlab.com/gitlab-org/gitlab/-/issues/295472) to give more
|
|
capacity to the current rate limit.
|
|
|
|
### Recovering from network errors
|
|
|
|
In general, [Unleash clients](https://github.com/Unleash/unleash#unleash-sdks) have
|
|
a fall-back mechanism when the server returns an error code.
|
|
For example, `unleash-ruby-client` reads flag data from the local backup so that
|
|
application can keep running in the current state.
|
|
|
|
Please reads the documentation in a SDK project for more information.
|
|
|
|
### Self-managed GitLab
|
|
|
|
Functionality-wise, there are no differences. Both SaaS and self-managed behave the same.
|
|
|
|
In terms of scalability, it's up to the spec of the GitLab instance.
|
|
For example, GitLab.com runs on HA architecture so that it can handle a lot of requests concurrently,
|
|
however, a self-managed instance runs on a low spec machine can't expect the same result.
|
|
Please see [Reference architectures](../administration/reference_architectures/index.md)
|
|
for more information.
|