Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
G
gitlab-ce
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
0
Issues
0
List
Boards
Labels
Milestones
Merge Requests
1
Merge Requests
1
Analytics
Analytics
Repository
Value Stream
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Commits
Issue Boards
Open sidebar
nexedi
gitlab-ce
Commits
29508607
Commit
29508607
authored
Apr 15, 2020
by
Alex Buijs
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
Use chatops for setting experiment ratio
parent
8e4aca65
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
51 additions
and
83 deletions
+51
-83
doc/development/experiment_guide/index.md
doc/development/experiment_guide/index.md
+12
-8
lib/gitlab/experimentation.rb
lib/gitlab/experimentation.rb
+31
-30
spec/lib/gitlab/experimentation_spec.rb
spec/lib/gitlab/experimentation_spec.rb
+8
-45
No files found.
doc/development/experiment_guide/index.md
View file @
29508607
...
@@ -32,10 +32,9 @@ The author then adds a comment to this piece of code and adds a link to the issu
...
@@ -32,10 +32,9 @@ The author then adds a comment to this piece of code and adds a link to the issu
#...
#...
},
},
# Add your experiment here:
# Add your experiment here:
sign_up_flow:
{
signup_flow:
{
feature_toggle: :experimental_sign_up_flow
,
# Feature flag that will be used
environment:
::
Gitlab
.
dev_env_or_com?
,
# Target environment, defaults to enabled for development and GitLab.com
environment:
::
Gitlab
.
dev_env_or_com?
,
# Target environment
tracking_category:
'Growth::Acquisition::Experiment::SignUpFlow'
# Used for providing the category when setting up tracking data
enabled_ratio:
0.1
# Percentage of users that will be part of the experiment. 10% of the users would be part of this experiments.
}
}
}.
freeze
}.
freeze
```
```
...
@@ -46,7 +45,7 @@ The author then adds a comment to this piece of code and adds a link to the issu
...
@@ -46,7 +45,7 @@ The author then adds a comment to this piece of code and adds a link to the issu
class
RegistrationController
<
Applicationcontroller
class
RegistrationController
<
Applicationcontroller
def
show
def
show
# experiment_enabled?(:feature_name) is also available in views and helpers
# experiment_enabled?(:feature_name) is also available in views and helpers
if
experiment_enabled?
(
:sign
_
up_flow
)
if
experiment_enabled?
(
:signup_flow
)
# render the experiment
# render the experiment
else
else
# render the original version
# render the original version
...
@@ -57,11 +56,16 @@ The author then adds a comment to this piece of code and adds a link to the issu
...
@@ -57,11 +56,16 @@ The author then adds a comment to this piece of code and adds a link to the issu
-
Track necessary events. See the
[
telemetry guide
](
../../telemetry/index.md
)
for details.
-
Track necessary events. See the
[
telemetry guide
](
../../telemetry/index.md
)
for details.
-
After the merge request is merged, use
[
`chatops`
](
../../ci/chatops/README.md
)
in the
-
After the merge request is merged, use
[
`chatops`
](
../../ci/chatops/README.md
)
in the
[
appropriate channel
](
../feature_flags/controls.md#where-to-run-commands
)
to enable the feature flag and start the experiment.
[
appropriate channel
](
../feature_flags/controls.md#where-to-run-commands
)
to start the experiment for 10% of the users.
The feature flag should have the name of the experiment with the
`_experiment_percentage`
suffix appended.
For visibility, please also share any commands run against production in the
`#s_growth`
channel:
For visibility, please also share any commands run against production in the
`#s_growth`
channel:
```
shell
```
shell
/chatops run feature
set
--project
=
gitlab-org/gitlab experimental_sign_up_flow
true
/chatops run feature
set
signup_flow_experiment_percentage 10
```
```
If you notice issues with the experiment, you can disable the experiment by setting the feature flag to
`false`
again.
If you notice issues with the experiment, you can disable the experiment by removing the feature flag:
```
shell
/chatops run feature delete signup_flow_experiment_percentage
```
lib/gitlab/experimentation.rb
View file @
29508607
...
@@ -2,48 +2,44 @@
...
@@ -2,48 +2,44 @@
# == Experimentation
# == Experimentation
#
#
# Utility module used for A/B testing experimental features. Define your experiments in the `EXPERIMENTS` constant.
# Utility module for A/B testing experimental features. Define your experiments in the `EXPERIMENTS` constant.
# The feature_toggle and environment keys are optional. If the feature_toggle is not set, a feature with the name of
# Experiment options:
# the experiment will be checked, with a default value of true. The enabled_ratio is required and should be
# - environment (optional, defaults to enabled for development and GitLab.com)
# the ratio for the number of users for which this experiment is enabled. For example: a ratio of 0.1 will
# - tracking_category (optional, used to set the category when tracking an experiment event)
# enable the experiment for 10% of the users (determined by the `experimentation_subject_index`).
#
# The experiment is controlled by a Feature Flag (https://docs.gitlab.com/ee/development/feature_flags/controls.html),
# which is named "#{key}_experiment_percentage" and *must* be set with a percentage and not be used for other purposes.
# To enable the experiment for 10% of the users (determined by the `experimentation_subject_index` value from a cookie):
#
# chatops: `/chatops run feature set key_experiment_percentage 10`
# console: `Feature.get(:key_experiment_percentage).enable_percentage_of_time(10)`
#
# To disable the experiment:
#
# chatops: `/chatops run feature delete key_experiment_percentage`
# console: `Feature.get(:key_experiment_percentage).remove`
#
#
module
Gitlab
module
Gitlab
module
Experimentation
module
Experimentation
EXPERIMENTS
=
{
EXPERIMENTS
=
{
signup_flow:
{
signup_flow:
{
feature_toggle: :experimental_separate_sign_up_flow
,
environment:
::
Gitlab
.
dev_env_or_com?
,
enabled_ratio:
1
,
tracking_category:
'Growth::Acquisition::Experiment::SignUpFlow'
tracking_category:
'Growth::Acquisition::Experiment::SignUpFlow'
},
},
paid_signup_flow:
{
paid_signup_flow:
{
feature_toggle: :paid_signup_flow
,
environment:
::
Gitlab
.
dev_env_or_com?
,
enabled_ratio:
1
,
tracking_category:
'Growth::Acquisition::Experiment::PaidSignUpFlow'
tracking_category:
'Growth::Acquisition::Experiment::PaidSignUpFlow'
},
},
suggest_pipeline:
{
suggest_pipeline:
{
feature_toggle: :suggest_pipeline
,
environment:
::
Gitlab
.
dev_env_or_com?
,
enabled_ratio:
0.1
,
tracking_category:
'Growth::Expansion::Experiment::SuggestPipeline'
tracking_category:
'Growth::Expansion::Experiment::SuggestPipeline'
},
},
ci_notification_dot:
{
ci_notification_dot:
{
feature_toggle: :ci_notification_dot
,
environment:
::
Gitlab
.
dev_env_or_com?
,
enabled_ratio:
0.1
,
tracking_category:
'Growth::Expansion::Experiment::CiNotificationDot'
tracking_category:
'Growth::Expansion::Experiment::CiNotificationDot'
},
},
buy_ci_minutes_version_a:
{
buy_ci_minutes_version_a:
{
feature_toggle: :buy_ci_minutes_version_a
,
environment:
::
Gitlab
.
dev_env_or_com?
,
enabled_ratio:
0.2
,
tracking_category:
'Growth::Expansion::Experiment::BuyCiMinutesVersionA'
tracking_category:
'Growth::Expansion::Experiment::BuyCiMinutesVersionA'
}
}
}.
freeze
}.
freeze
# Controller concern that checks if an
experimentation_subject_id cookie
is present and sets it if absent.
# Controller concern that checks if an
`experimentation_subject_id cookie`
is present and sets it if absent.
# Used for A/B testing of experimental features. Exposes the `experiment_enabled?(experiment_name)` method
# Used for A/B testing of experimental features. Exposes the `experiment_enabled?(experiment_name)` method
# to controllers and views. It returns true when the experiment is enabled and the user is selected as part
# to controllers and views. It returns true when the experiment is enabled and the user is selected as part
# of the experimental group.
# of the experimental group.
...
@@ -144,7 +140,7 @@ module Gitlab
...
@@ -144,7 +140,7 @@ module Gitlab
return
false
unless
EXPERIMENTS
.
key?
(
experiment_key
)
return
false
unless
EXPERIMENTS
.
key?
(
experiment_key
)
experiment
=
experiment
(
experiment_key
)
experiment
=
experiment
(
experiment_key
)
experiment
.
feature_toggle_
enabled?
&&
experiment
.
enabled_for_environment?
experiment
.
enabled?
&&
experiment
.
enabled_for_environment?
end
end
def
enabled_for_user?
(
experiment_key
,
experimentation_subject_index
)
def
enabled_for_user?
(
experiment_key
,
experimentation_subject_index
)
...
@@ -153,23 +149,28 @@ module Gitlab
...
@@ -153,23 +149,28 @@ module Gitlab
end
end
end
end
Experiment
=
Struct
.
new
(
:key
,
:feature_toggle
,
:environment
,
:enabled_ratio
,
:tracking_category
,
keyword_init:
true
)
do
Experiment
=
Struct
.
new
(
:key
,
:environment
,
:tracking_category
,
keyword_init:
true
)
do
def
feature_toggle_enabled?
def
enabled?
return
Feature
.
enabled?
(
key
,
default_enabled:
true
)
if
feature_toggle
.
nil?
experiment_percentage
.
positive?
Feature
.
enabled?
(
feature_toggle
)
end
end
def
enabled_for_environment?
def
enabled_for_environment?
return
true
if
environment
.
nil?
return
::
Gitlab
.
dev_env_or_com?
if
environment
.
nil?
environment
environment
end
end
def
enabled_for_experimentation_subject?
(
experimentation_subject_index
)
def
enabled_for_experimentation_subject?
(
experimentation_subject_index
)
return
false
if
enabled_ratio
.
nil?
||
experimentation_subject_index
.
blank?
return
false
if
experimentation_subject_index
.
blank?
experimentation_subject_index
<=
experiment_percentage
end
private
experimentation_subject_index
<=
enabled_ratio
*
100
# When a feature does not exist, the `percentage_of_time_value` method will return 0
def
experiment_percentage
@experiment_percentage
||=
Feature
.
get
(
:"
#{
key
}
_experiment_percentage"
).
percentage_of_time_value
end
end
end
end
end
end
...
...
spec/lib/gitlab/experimentation_spec.rb
View file @
29508607
...
@@ -6,19 +6,16 @@ describe Gitlab::Experimentation do
...
@@ -6,19 +6,16 @@ describe Gitlab::Experimentation do
before
do
before
do
stub_const
(
'Gitlab::Experimentation::EXPERIMENTS'
,
{
stub_const
(
'Gitlab::Experimentation::EXPERIMENTS'
,
{
test_experiment:
{
test_experiment:
{
feature_toggle:
feature_toggle
,
environment:
environment
,
environment:
environment
,
enabled_ratio:
enabled_ratio
,
tracking_category:
'Team'
tracking_category:
'Team'
}
}
})
})
stub_feature_flags
(
feature_toggle
=>
tru
e
)
allow
(
Feature
).
to
receive
(
:get
).
with
(
:test_experiment_experiment_percentage
).
and_return
double
(
percentage_of_time_value:
enabled_percentag
e
)
end
end
let
(
:feature_toggle
)
{
:test_experiment_toggle
}
let
(
:environment
)
{
Rails
.
env
.
test?
}
let
(
:environment
)
{
Rails
.
env
.
test?
}
let
(
:enabled_
ratio
)
{
0.1
}
let
(
:enabled_
percentage
)
{
10
}
describe
Gitlab
::
Experimentation
::
ControllerConcern
,
type: :controller
do
describe
Gitlab
::
Experimentation
::
ControllerConcern
,
type: :controller
do
controller
(
ApplicationController
)
do
controller
(
ApplicationController
)
do
...
@@ -251,44 +248,16 @@ describe Gitlab::Experimentation do
...
@@ -251,44 +248,16 @@ describe Gitlab::Experimentation do
end
end
end
end
describe
'feature toggle'
do
describe
'experiment is disabled'
do
context
'feature toggle is not set'
do
let
(
:enabled_percentage
)
{
0
}
let
(
:feature_toggle
)
{
nil
}
it
{
is_expected
.
to
be_truthy
}
it
{
is_expected
.
to
be_falsey
}
end
context
'feature toggle is not set, but a feature with the experiment key as name does exist'
do
before
do
stub_feature_flags
(
test_experiment:
false
)
end
let
(
:feature_toggle
)
{
nil
}
it
{
is_expected
.
to
be_falsey
}
end
context
'feature toggle is disabled'
do
before
do
stub_feature_flags
(
feature_toggle
=>
false
)
end
it
{
is_expected
.
to
be_falsey
}
end
end
end
describe
'environment'
do
describe
'we are on the wrong environment'
do
context
'environment is not set'
do
let
(
:environment
)
{
::
Gitlab
.
com?
}
let
(
:environment
)
{
nil
}
it
{
is_expected
.
to
be_truthy
}
end
context
'we are on the wrong environment'
do
let
(
:environment
)
{
::
Gitlab
.
com?
}
it
{
is_expected
.
to
be_falsey
}
it
{
is_expected
.
to
be_falsey
}
end
end
end
end
end
...
@@ -312,12 +281,6 @@ describe Gitlab::Experimentation do
...
@@ -312,12 +281,6 @@ describe Gitlab::Experimentation do
it
{
is_expected
.
to
be_truthy
}
it
{
is_expected
.
to
be_truthy
}
context
'enabled ratio is not set'
do
let
(
:enabled_ratio
)
{
nil
}
it
{
is_expected
.
to
be_falsey
}
end
describe
'experimentation_subject_index'
do
describe
'experimentation_subject_index'
do
context
'experimentation_subject_index is not set'
do
context
'experimentation_subject_index is not set'
do
let
(
:experimentation_subject_index
)
{
nil
}
let
(
:experimentation_subject_index
)
{
nil
}
...
...
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment