flagsmith 2.0.0 → 3.0.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/.rspec +2 -0
- data/.rubocop.yml +4 -1
- data/.rubocop_todo.yml +0 -1
- data/.ruby-version +1 -0
- data/Gemfile.lock +35 -6
- data/LICENCE +4 -5
- data/README.md +8 -64
- data/Rakefile +5 -1
- data/example/.env.development +5 -0
- data/example/.env.test +4 -0
- data/example/.gitignore +5 -0
- data/example/.hanamirc +3 -0
- data/example/.rspec +2 -0
- data/example/Gemfile +25 -0
- data/example/Gemfile.lock +269 -0
- data/example/README.md +29 -0
- data/example/Rakefile +9 -0
- data/example/apps/web/application.rb +162 -0
- data/example/apps/web/config/routes.rb +1 -0
- data/example/apps/web/controllers/.gitkeep +0 -0
- data/example/apps/web/controllers/home/index.rb +32 -0
- data/example/apps/web/templates/application.html.slim +7 -0
- data/example/apps/web/templates/home/index.html.slim +10 -0
- data/example/apps/web/views/application_layout.rb +7 -0
- data/example/apps/web/views/home/index.rb +29 -0
- data/example/config/boot.rb +2 -0
- data/example/config/environment.rb +17 -0
- data/example/config/initializers/.gitkeep +0 -0
- data/example/config/initializers/flagsmith.rb +9 -0
- data/example/config/puma.rb +15 -0
- data/example/config.ru +3 -0
- data/example/spec/example/entities/.gitkeep +0 -0
- data/example/spec/example/mailers/.gitkeep +0 -0
- data/example/spec/example/repositories/.gitkeep +0 -0
- data/example/spec/features_helper.rb +12 -0
- data/example/spec/spec_helper.rb +103 -0
- data/example/spec/support/.gitkeep +0 -0
- data/example/spec/support/capybara.rb +8 -0
- data/example/spec/web/controllers/.gitkeep +0 -0
- data/example/spec/web/controllers/home/index_spec.rb +9 -0
- data/example/spec/web/features/.gitkeep +0 -0
- data/example/spec/web/views/application_layout_spec.rb +10 -0
- data/example/spec/web/views/home/index_spec.rb +10 -0
- data/lib/flagsmith/engine/core.rb +88 -0
- data/lib/flagsmith/engine/environments/models.rb +61 -0
- data/lib/flagsmith/engine/features/models.rb +173 -0
- data/lib/flagsmith/engine/identities/models.rb +115 -0
- data/lib/flagsmith/engine/organisations/models.rb +28 -0
- data/lib/flagsmith/engine/projects/models.rb +31 -0
- data/lib/flagsmith/engine/segments/constants.rb +41 -0
- data/lib/flagsmith/engine/segments/evaluator.rb +68 -0
- data/lib/flagsmith/engine/segments/models.rb +121 -0
- data/lib/flagsmith/engine/utils/hash_func.rb +34 -0
- data/lib/flagsmith/hash_slice.rb +12 -0
- data/lib/flagsmith/sdk/analytics_processor.rb +39 -0
- data/lib/flagsmith/sdk/api_client.rb +47 -0
- data/lib/flagsmith/sdk/config.rb +91 -0
- data/lib/flagsmith/sdk/errors.rb +9 -0
- data/lib/flagsmith/sdk/instance_methods.rb +137 -0
- data/lib/flagsmith/sdk/intervals.rb +24 -0
- data/lib/flagsmith/sdk/models/flag.rb +62 -0
- data/lib/flagsmith/sdk/models/flags/collection.rb +105 -0
- data/lib/flagsmith/sdk/pooling_manager.rb +31 -0
- data/lib/flagsmith/version.rb +5 -0
- data/lib/flagsmith.rb +79 -101
- metadata +104 -6
- data/.gitignore +0 -57
- data/flagsmith.gemspec +0 -22
@@ -0,0 +1,162 @@
|
|
1
|
+
require 'hanami/helpers'
|
2
|
+
require 'hanami/assets'
|
3
|
+
|
4
|
+
require_relative '../../../lib/flagsmith'
|
5
|
+
|
6
|
+
module Web
|
7
|
+
class Application < Hanami::Application
|
8
|
+
configure do
|
9
|
+
root __dir__
|
10
|
+
|
11
|
+
load_paths << %w[
|
12
|
+
controllers
|
13
|
+
views
|
14
|
+
]
|
15
|
+
|
16
|
+
routes 'config/routes'
|
17
|
+
|
18
|
+
layout :application # It will load Web::Views::ApplicationLayout
|
19
|
+
|
20
|
+
templates 'templates'
|
21
|
+
|
22
|
+
##
|
23
|
+
# ASSETS
|
24
|
+
#
|
25
|
+
assets do
|
26
|
+
# In order to skip JavaScript compression comment the following line
|
27
|
+
javascript_compressor :builtin
|
28
|
+
|
29
|
+
# In order to skip stylesheet compression comment the following line
|
30
|
+
stylesheet_compressor :builtin
|
31
|
+
end
|
32
|
+
|
33
|
+
##
|
34
|
+
# SECURITY
|
35
|
+
#
|
36
|
+
|
37
|
+
# X-Frame-Options is a HTTP header supported by modern browsers.
|
38
|
+
# It determines if a web page can or cannot be included via <frame> and
|
39
|
+
# <iframe> tags by untrusted domains.
|
40
|
+
#
|
41
|
+
# Web applications can send this header to prevent Clickjacking attacks.
|
42
|
+
#
|
43
|
+
# Read more at:
|
44
|
+
#
|
45
|
+
# * https://developer.mozilla.org/en-US/docs/Web/HTTP/X-Frame-Options
|
46
|
+
# * https://www.owasp.org/index.php/Clickjacking
|
47
|
+
#
|
48
|
+
security.x_frame_options 'DENY'
|
49
|
+
|
50
|
+
# X-Content-Type-Options prevents browsers from interpreting files as
|
51
|
+
# something else than declared by the content type in the HTTP headers.
|
52
|
+
#
|
53
|
+
# Read more at:
|
54
|
+
#
|
55
|
+
# * https://www.owasp.org/index.php/OWASP_Secure_Headers_Project#X-Content-Type-Options
|
56
|
+
# * https://msdn.microsoft.com/en-us/library/gg622941%28v=vs.85%29.aspx
|
57
|
+
# * https://blogs.msdn.microsoft.com/ie/2008/09/02/ie8-security-part-vi-beta-2-update
|
58
|
+
#
|
59
|
+
security.x_content_type_options 'nosniff'
|
60
|
+
|
61
|
+
# X-XSS-Protection is a HTTP header to determine the behavior of the
|
62
|
+
# browser in case an XSS attack is detected.
|
63
|
+
#
|
64
|
+
# Read more at:
|
65
|
+
#
|
66
|
+
# * https://www.owasp.org/index.php/Cross-site_Scripting_(XSS)
|
67
|
+
# * https://www.owasp.org/index.php/OWASP_Secure_Headers_Project#X-XSS-Protection
|
68
|
+
#
|
69
|
+
security.x_xss_protection '1; mode=block'
|
70
|
+
|
71
|
+
# Content-Security-Policy (CSP) is a HTTP header supported by modern
|
72
|
+
# browsers. It determines trusted sources of execution for dynamic
|
73
|
+
# contents (JavaScript) or other web related assets: stylesheets, images,
|
74
|
+
# fonts, plugins, etc.
|
75
|
+
#
|
76
|
+
# Web applications can send this header to mitigate Cross Site Scripting
|
77
|
+
# (XSS) attacks.
|
78
|
+
#
|
79
|
+
# The default value allows images, scripts, AJAX, fonts and CSS from the
|
80
|
+
# same origin, and does not allow any other resources to load (eg object,
|
81
|
+
# frame, media, etc).
|
82
|
+
#
|
83
|
+
# Inline JavaScript is NOT allowed. To enable it, please use:
|
84
|
+
# "script-src 'unsafe-inline'".
|
85
|
+
#
|
86
|
+
# Content Security Policy introduction:
|
87
|
+
#
|
88
|
+
# * http://www.html5rocks.com/en/tutorials/security/content-security-policy/
|
89
|
+
# * https://www.owasp.org/index.php/Content_Security_Policy
|
90
|
+
# * https://www.owasp.org/index.php/Cross-site_Scripting_%28XSS%29
|
91
|
+
#
|
92
|
+
# Inline and eval JavaScript risks:
|
93
|
+
#
|
94
|
+
# * http://www.html5rocks.com/en/tutorials/security/content-security-policy/#inline-code-considered-harmful
|
95
|
+
# * http://www.html5rocks.com/en/tutorials/security/content-security-policy/#eval-too
|
96
|
+
#
|
97
|
+
# Content Security Policy usage:
|
98
|
+
#
|
99
|
+
# * http://content-security-policy.com/
|
100
|
+
# * https://developer.mozilla.org/en-US/docs/Web/Security/CSP/Using_Content_Security_Policy
|
101
|
+
#
|
102
|
+
# Content Security Policy references:
|
103
|
+
#
|
104
|
+
# * https://developer.mozilla.org/en-US/docs/Web/Security/CSP/CSP_policy_directives
|
105
|
+
#
|
106
|
+
security.content_security_policy %(
|
107
|
+
form-action 'self';
|
108
|
+
frame-ancestors 'self';
|
109
|
+
base-uri 'self';
|
110
|
+
default-src 'none';
|
111
|
+
script-src 'self';
|
112
|
+
connect-src 'self';
|
113
|
+
img-src 'self' https: data:;
|
114
|
+
style-src 'self' 'unsafe-inline' https:;
|
115
|
+
font-src 'self';
|
116
|
+
object-src 'none';
|
117
|
+
plugin-types application/pdf;
|
118
|
+
child-src 'self';
|
119
|
+
frame-src 'self';
|
120
|
+
media-src 'self'
|
121
|
+
)
|
122
|
+
|
123
|
+
##
|
124
|
+
# FRAMEWORKS
|
125
|
+
#
|
126
|
+
|
127
|
+
# Configure the code that will yield each time Web::Action is included
|
128
|
+
# This is useful for sharing common functionality
|
129
|
+
#
|
130
|
+
# See: http://www.rubydoc.info/gems/hanami-controller#Configuration
|
131
|
+
controller.prepare do
|
132
|
+
# include MyAuthentication # included in all the actions
|
133
|
+
# before :authenticate! # run an authentication before callback
|
134
|
+
end
|
135
|
+
|
136
|
+
# Configure the code that will yield each time Web::View is included
|
137
|
+
# This is useful for sharing common functionality
|
138
|
+
#
|
139
|
+
# See: http://www.rubydoc.info/gems/hanami-view#Configuration
|
140
|
+
view.prepare do
|
141
|
+
include Hanami::Helpers
|
142
|
+
include Web::Assets::Helpers
|
143
|
+
end
|
144
|
+
end
|
145
|
+
|
146
|
+
##
|
147
|
+
# DEVELOPMENT
|
148
|
+
#
|
149
|
+
configure :development do
|
150
|
+
# Don't handle exceptions, render the stack trace
|
151
|
+
handle_exceptions false
|
152
|
+
end
|
153
|
+
|
154
|
+
##
|
155
|
+
# TEST
|
156
|
+
#
|
157
|
+
configure :test do
|
158
|
+
# Don't handle exceptions, render the stack trace
|
159
|
+
handle_exceptions false
|
160
|
+
end
|
161
|
+
end
|
162
|
+
end
|
@@ -0,0 +1 @@
|
|
1
|
+
get '/', to: 'home#index'
|
File without changes
|
@@ -0,0 +1,32 @@
|
|
1
|
+
module Web
|
2
|
+
module Controllers
|
3
|
+
module Home
|
4
|
+
class Index
|
5
|
+
include Web::Action
|
6
|
+
|
7
|
+
expose :identifier, :show_button, :button_color
|
8
|
+
|
9
|
+
def call(params)
|
10
|
+
@identifier = params.get(:flagsmith, :identifier)
|
11
|
+
|
12
|
+
if @identifier.nil? || @identifier.blank?
|
13
|
+
# Get the default flags for the current environment
|
14
|
+
flags = $flagsmith.get_environment_flags
|
15
|
+
@show_button = flags.is_feature_enabled("secret_button")
|
16
|
+
@button_data = JSON.parse(flags.get_feature_value("secret_button"))["colour"]
|
17
|
+
else
|
18
|
+
trait_key = params.get(:flagsmith, :trait_key)
|
19
|
+
trait_value = params.get(:flagsmith, :trait_value)
|
20
|
+
traits = trait_key.nil? ? nil : { trait_key: trait_value }
|
21
|
+
|
22
|
+
# Get the flags for an identity, including the provided trait which will be
|
23
|
+
# persisted to the API for future requests.
|
24
|
+
identity_flags = $flagsmith.get_identity_flags(identifier, traits)
|
25
|
+
@show_button = identity_flags.is_feature_enabled('secret_button')
|
26
|
+
@button_color = JSON.parse(identity_flags.get_feature_value('secret_button'))['colour']
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
end
|
31
|
+
end
|
32
|
+
end
|
@@ -0,0 +1,29 @@
|
|
1
|
+
module Web
|
2
|
+
module Views
|
3
|
+
module Home
|
4
|
+
class Index
|
5
|
+
include Web::View
|
6
|
+
|
7
|
+
def form
|
8
|
+
form_for :flagsmith, '/', method: :get do
|
9
|
+
h3 'Identify as a User'
|
10
|
+
div style: 'margin-bottom: 1em;' do
|
11
|
+
label :identifier
|
12
|
+
text_field :identifier
|
13
|
+
end
|
14
|
+
p '... with an optional user trait'
|
15
|
+
div style: 'margin-bottom: 1em;' do
|
16
|
+
label :trait_key
|
17
|
+
text_field :trait_key
|
18
|
+
end
|
19
|
+
div style: 'margin-bottom: 1em;' do
|
20
|
+
label :trait_value
|
21
|
+
text_field :trait_value
|
22
|
+
end
|
23
|
+
div { submit 'Identify!' }
|
24
|
+
end
|
25
|
+
end
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
@@ -0,0 +1,17 @@
|
|
1
|
+
require 'bundler/setup'
|
2
|
+
require 'hanami/setup'
|
3
|
+
require 'hanami/model'
|
4
|
+
require_relative '../apps/web/application'
|
5
|
+
|
6
|
+
Hanami.configure do
|
7
|
+
mount Web::Application, at: '/'
|
8
|
+
|
9
|
+
model do
|
10
|
+
adapter :sql, ENV.fetch('DATABASE_URL')
|
11
|
+
end
|
12
|
+
|
13
|
+
environment :development do
|
14
|
+
# See: https://guides.hanamirb.org/projects/logging
|
15
|
+
logger level: :debug
|
16
|
+
end
|
17
|
+
end
|
File without changes
|
@@ -0,0 +1,9 @@
|
|
1
|
+
$flagsmith = Flagsmith::Client.new(
|
2
|
+
enable_local_evaluation: true,
|
3
|
+
environment_refresh_interval_seconds: 60,
|
4
|
+
default_flag_handler: lambda { |feature_name|
|
5
|
+
Flagsmith::Flag.new(
|
6
|
+
feature_name: feature_name, enabled: false, value: {}.to_json, feature_id: nil
|
7
|
+
)
|
8
|
+
}
|
9
|
+
)
|
data/example/config.ru
ADDED
File without changes
|
File without changes
|
File without changes
|
@@ -0,0 +1,12 @@
|
|
1
|
+
# Require this file for feature tests
|
2
|
+
require_relative './spec_helper'
|
3
|
+
|
4
|
+
require 'capybara'
|
5
|
+
require 'capybara/rspec'
|
6
|
+
|
7
|
+
RSpec.configure do |config|
|
8
|
+
config.include RSpec::FeatureExampleGroup
|
9
|
+
|
10
|
+
config.include Capybara::DSL, feature: true
|
11
|
+
config.include Capybara::RSpecMatchers, feature: true
|
12
|
+
end
|
@@ -0,0 +1,103 @@
|
|
1
|
+
# Require this file for unit tests
|
2
|
+
ENV['HANAMI_ENV'] ||= 'test'
|
3
|
+
|
4
|
+
require_relative '../config/environment'
|
5
|
+
Hanami.boot
|
6
|
+
Hanami::Utils.require!("#{__dir__}/support")
|
7
|
+
|
8
|
+
# This file was generated by the `rspec --init` command. Conventionally, all
|
9
|
+
# specs live under a `spec` directory, which RSpec adds to the `$LOAD_PATH`.
|
10
|
+
# The generated `.rspec` file contains `--require spec_helper` which will cause
|
11
|
+
# this file to always be loaded, without a need to explicitly require it in any
|
12
|
+
# files.
|
13
|
+
#
|
14
|
+
# Given that it is always loaded, you are encouraged to keep this file as
|
15
|
+
# light-weight as possible. Requiring heavyweight dependencies from this file
|
16
|
+
# will add to the boot time of your test suite on EVERY test run, even for an
|
17
|
+
# individual file that may not need all of that loaded. Instead, consider making
|
18
|
+
# a separate helper file that requires the additional dependencies and performs
|
19
|
+
# the additional setup, and require it from the spec files that actually need
|
20
|
+
# it.
|
21
|
+
#
|
22
|
+
# The `.rspec` file also contains a few flags that are not defaults but that
|
23
|
+
# users commonly want.
|
24
|
+
#
|
25
|
+
# See http://rubydoc.info/gems/rspec-core/RSpec/Core/Configuration
|
26
|
+
RSpec.configure do |config|
|
27
|
+
# rspec-expectations config goes here. You can use an alternate
|
28
|
+
# assertion/expectation library such as wrong or the stdlib/minitest
|
29
|
+
# assertions if you prefer.
|
30
|
+
config.expect_with :rspec do |expectations|
|
31
|
+
# This option will default to `true` in RSpec 4. It makes the `description`
|
32
|
+
# and `failure_message` of custom matchers include text for helper methods
|
33
|
+
# defined using `chain`, e.g.:
|
34
|
+
# be_bigger_than(2).and_smaller_than(4).description
|
35
|
+
# # => "be bigger than 2 and smaller than 4"
|
36
|
+
# ...rather than:
|
37
|
+
# # => "be bigger than 2"
|
38
|
+
expectations.include_chain_clauses_in_custom_matcher_descriptions = true
|
39
|
+
end
|
40
|
+
|
41
|
+
# rspec-mocks config goes here. You can use an alternate test double
|
42
|
+
# library (such as bogus or mocha) by changing the `mock_with` option here.
|
43
|
+
config.mock_with :rspec do |mocks|
|
44
|
+
# Prevents you from mocking or stubbing a method that does not exist on
|
45
|
+
# a real object. This is generally recommended, and will default to
|
46
|
+
# `true` in RSpec 4.
|
47
|
+
mocks.verify_partial_doubles = true
|
48
|
+
end
|
49
|
+
|
50
|
+
# The settings below are suggested to provide a good initial experience
|
51
|
+
# with RSpec, but feel free to customize to your heart's content.
|
52
|
+
=begin
|
53
|
+
# These two settings work together to allow you to limit a spec run
|
54
|
+
# to individual examples or groups you care about by tagging them with
|
55
|
+
# `:focus` metadata. When nothing is tagged with `:focus`, all examples
|
56
|
+
# get run.
|
57
|
+
config.filter_run :focus
|
58
|
+
config.run_all_when_everything_filtered = true
|
59
|
+
|
60
|
+
# Allows RSpec to persist some state between runs in order to support
|
61
|
+
# the `--only-failures` and `--next-failure` CLI options. We recommend
|
62
|
+
# you configure your source control system to ignore this file.
|
63
|
+
config.example_status_persistence_file_path = "spec/examples.txt"
|
64
|
+
|
65
|
+
# Limits the available syntax to the non-monkey patched syntax that is
|
66
|
+
# recommended. For more details, see:
|
67
|
+
# - http://myronmars.to/n/dev-blog/2012/06/rspecs-new-expectation-syntax
|
68
|
+
# - http://teaisaweso.me/blog/2013/05/27/rspecs-new-message-expectation-syntax/
|
69
|
+
# - http://myronmars.to/n/dev-blog/2014/05/notable-changes-in-rspec-3#new__config_option_to_disable_rspeccore_monkey_patching
|
70
|
+
config.disable_monkey_patching!
|
71
|
+
|
72
|
+
# This setting enables warnings. It's recommended, but in many cases may
|
73
|
+
# be too noisy due to issues in dependencies.
|
74
|
+
config.warnings = false
|
75
|
+
|
76
|
+
# Many RSpec users commonly either run the entire suite or an individual
|
77
|
+
# file, and it's useful to allow more verbose output when running an
|
78
|
+
# individual spec file.
|
79
|
+
if config.files_to_run.one?
|
80
|
+
# Use the documentation formatter for detailed output,
|
81
|
+
# unless a formatter has already been configured
|
82
|
+
# (e.g. via a command-line flag).
|
83
|
+
config.default_formatter = 'doc'
|
84
|
+
end
|
85
|
+
|
86
|
+
# Print the 10 slowest examples and example groups at the
|
87
|
+
# end of the spec run, to help surface which specs are running
|
88
|
+
# particularly slow.
|
89
|
+
config.profile_examples = 10
|
90
|
+
|
91
|
+
# Run specs in random order to surface order dependencies. If you find an
|
92
|
+
# order dependency and want to debug it, you can fix the order by providing
|
93
|
+
# the seed, which is printed after each run.
|
94
|
+
# --seed 1234
|
95
|
+
config.order = :random
|
96
|
+
|
97
|
+
# Seed global randomization in this process using the `--seed` CLI option.
|
98
|
+
# Setting this allows you to use `--seed` to deterministically reproduce
|
99
|
+
# test failures related to randomization by passing the same `--seed` value
|
100
|
+
# as the one that triggered the failure.
|
101
|
+
Kernel.srand config.seed
|
102
|
+
=end
|
103
|
+
end
|
File without changes
|
File without changes
|
File without changes
|
@@ -0,0 +1,10 @@
|
|
1
|
+
require "spec_helper"
|
2
|
+
|
3
|
+
RSpec.describe Web::Views::ApplicationLayout, type: :view do
|
4
|
+
let(:layout) { Web::Views::ApplicationLayout.new({ format: :html }, "contents") }
|
5
|
+
let(:rendered) { layout.render }
|
6
|
+
|
7
|
+
it 'contains application name' do
|
8
|
+
expect(rendered).to include('Web')
|
9
|
+
end
|
10
|
+
end
|
@@ -0,0 +1,10 @@
|
|
1
|
+
RSpec.describe Web::Views::Home::Index, type: :view do
|
2
|
+
let(:exposures) { Hash[format: :html] }
|
3
|
+
let(:template) { Hanami::View::Template.new('apps/web/templates/home/index.html.slim') }
|
4
|
+
let(:view) { described_class.new(template, exposures) }
|
5
|
+
let(:rendered) { view.render }
|
6
|
+
|
7
|
+
it 'exposes #format' do
|
8
|
+
expect(view.format).to eq exposures.fetch(:format)
|
9
|
+
end
|
10
|
+
end
|
@@ -0,0 +1,88 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
require 'semantic'
|
4
|
+
require 'securerandom'
|
5
|
+
|
6
|
+
require_relative 'environments/models'
|
7
|
+
require_relative 'features/models'
|
8
|
+
require_relative 'identities/models'
|
9
|
+
require_relative 'organisations/models'
|
10
|
+
require_relative 'projects/models'
|
11
|
+
require_relative 'segments/evaluator'
|
12
|
+
require_relative 'segments/models'
|
13
|
+
require_relative 'utils/hash_func'
|
14
|
+
|
15
|
+
module Flagsmith
|
16
|
+
module Engine
|
17
|
+
# Flags engine methods
|
18
|
+
module Core
|
19
|
+
include Flagsmith::Engine::Segments::Evaluator
|
20
|
+
|
21
|
+
def get_identity_feature_state(environment, identity, feature_name, override_traits = nil)
|
22
|
+
feature_states = get_identity_feature_states_dict(environment, identity, override_traits).values
|
23
|
+
|
24
|
+
feature_state = feature_states.find { |f| f.feature.name == feature_name }
|
25
|
+
|
26
|
+
raise Flagsmith::FeatureStateNotFound, 'Feature State Not Found' if feature_state.nil?
|
27
|
+
|
28
|
+
feature_state
|
29
|
+
end
|
30
|
+
|
31
|
+
def get_identity_feature_states(environment, identity, override_traits = nil)
|
32
|
+
feature_states = get_identity_feature_states_dict(environment, identity, override_traits).values
|
33
|
+
|
34
|
+
return feature_states.select(&:enabled?) if environment.project.hide_disabled_flags
|
35
|
+
|
36
|
+
feature_states
|
37
|
+
end
|
38
|
+
|
39
|
+
def get_environment_feature_state(environment, feature_name)
|
40
|
+
features_state = environment.feature_states.find { |f| f.feature.name == feature_name }
|
41
|
+
|
42
|
+
raise Flagsmith::FeatureStateNotFound, 'Feature State Not Found' if features_state.nil?
|
43
|
+
|
44
|
+
features_state
|
45
|
+
end
|
46
|
+
|
47
|
+
def get_environment_feature_states(environment)
|
48
|
+
return environment.feature_states.select(&:enabled?) if environment.project.hide_disabled_flags
|
49
|
+
|
50
|
+
environment.feature_states
|
51
|
+
end
|
52
|
+
|
53
|
+
private
|
54
|
+
|
55
|
+
def get_identity_feature_states_dict(environment, identity, override_traits = nil)
|
56
|
+
# Get feature states from the environment
|
57
|
+
feature_states = {}
|
58
|
+
override = ->(fs) { feature_states[fs.feature.id] = fs }
|
59
|
+
environment.feature_states.each(&override)
|
60
|
+
|
61
|
+
override_by_matching_segments(environment, identity, override_traits) do |fs|
|
62
|
+
override.call(fs) unless higher_segment_priority?(feature_states, fs)
|
63
|
+
end
|
64
|
+
|
65
|
+
# Override with any feature states defined directly the identity
|
66
|
+
identity.identity_features.each(&override)
|
67
|
+
feature_states
|
68
|
+
end
|
69
|
+
|
70
|
+
# Override with any feature states defined by matching segments
|
71
|
+
def override_by_matching_segments(environment, identity, override_traits)
|
72
|
+
identity_segments = get_identity_segments(environment, identity, override_traits)
|
73
|
+
identity_segments.each do |matching_segment|
|
74
|
+
matching_segment.feature_states.each do |feature_state|
|
75
|
+
yield feature_state if block_given?
|
76
|
+
end
|
77
|
+
end
|
78
|
+
end
|
79
|
+
|
80
|
+
def higher_segment_priority?(collection, feature_state)
|
81
|
+
collection.key?(feature_state.feature.id) &&
|
82
|
+
collection[feature_state.feature.id].higher_segment_priority?(
|
83
|
+
feature_state
|
84
|
+
)
|
85
|
+
end
|
86
|
+
end
|
87
|
+
end
|
88
|
+
end
|
@@ -0,0 +1,61 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
module Flagsmith
|
4
|
+
module Engine
|
5
|
+
# EnvironmentModel
|
6
|
+
class Environment
|
7
|
+
attr_reader :id, :api_key
|
8
|
+
attr_accessor :project, :feature_states, :amplitude_config, :segment_config,
|
9
|
+
:mixpanel_config, :heap_config
|
10
|
+
|
11
|
+
def initialize(id:, api_key:, project:, feature_states: [])
|
12
|
+
@id = id
|
13
|
+
@api_key = api_key
|
14
|
+
@project = project
|
15
|
+
@feature_states = feature_states
|
16
|
+
end
|
17
|
+
|
18
|
+
class << self
|
19
|
+
def build(json)
|
20
|
+
project = Flagsmith::Engine::Project.build(json[:project])
|
21
|
+
feature_states = json[:feature_states].map do |fs|
|
22
|
+
Flagsmith::Engine::FeatureState.build(fs)
|
23
|
+
end
|
24
|
+
|
25
|
+
new(**json.slice(:id, :api_key).merge(project: project, feature_states: feature_states))
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
module Environments
|
31
|
+
# EnvironmentApiKeyModel
|
32
|
+
class ApiKey
|
33
|
+
attr_reader :id, :key, :created_at, :name, :client_api_key
|
34
|
+
attr_accessor :expires_at, :active
|
35
|
+
|
36
|
+
def initialize(params = {})
|
37
|
+
@id = params.fetch(:id)
|
38
|
+
@key = params.fetch(:key)
|
39
|
+
@name = params.fetch(:name)
|
40
|
+
@client_api_key = params.fetch(:client_api_key)
|
41
|
+
@created_at = params.fetch(:created_at, Time.now)
|
42
|
+
@expires_at = params.fetch(:expires_at, nil)
|
43
|
+
@active = params.fetch(:active, true)
|
44
|
+
end
|
45
|
+
|
46
|
+
def valid?
|
47
|
+
active && (!expires_at || expires_at > Time.now)
|
48
|
+
end
|
49
|
+
|
50
|
+
class << self
|
51
|
+
def build(json)
|
52
|
+
attributes = json.slice(:id, :key, :name, :client_api_key, :active)
|
53
|
+
attributes = attributes.merge(expires_at: Date.parse(json[:created_at])) unless json[:created_at].nil?
|
54
|
+
attributes = attributes.merge(expires_at: Date.parse(json[:expires_at])) unless json[:expires_at].nil?
|
55
|
+
new(**attributes)
|
56
|
+
end
|
57
|
+
end
|
58
|
+
end
|
59
|
+
end
|
60
|
+
end
|
61
|
+
end
|