flipper-ui 0.16.2 → 0.20.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (155) hide show
  1. checksums.yaml +5 -5
  2. data/docs/ui/README.md +34 -24
  3. data/docs/ui/images/banner.png +0 -0
  4. data/docs/ui/images/description.png +0 -0
  5. data/docs/ui/images/feature.png +0 -0
  6. data/docs/ui/images/features.png +0 -0
  7. data/examples/ui/basic.ru +20 -0
  8. data/flipper-ui.gemspec +2 -3
  9. data/lib/flipper/ui/action.rb +3 -6
  10. data/lib/flipper/ui/actions/feature.rb +5 -2
  11. data/lib/flipper/ui/actions/features.rb +14 -1
  12. data/lib/flipper/ui/actions/file.rb +1 -1
  13. data/lib/flipper/ui/assets/javascripts/application.coffee +5 -3
  14. data/lib/flipper/ui/configuration.rb +34 -10
  15. data/lib/flipper/ui/decorators/feature.rb +39 -13
  16. data/lib/flipper/ui/public/css/application.css +20 -6493
  17. data/lib/flipper/ui/public/js/application.js +5 -5
  18. data/lib/flipper/ui/util.rb +40 -0
  19. data/lib/flipper/ui/views/add_actor.erb +2 -2
  20. data/lib/flipper/ui/views/add_feature.erb +2 -2
  21. data/lib/flipper/ui/views/add_group.erb +1 -1
  22. data/lib/flipper/ui/views/feature.erb +199 -180
  23. data/lib/flipper/ui/views/features.erb +55 -36
  24. data/lib/flipper/ui/views/layout.erb +4 -14
  25. data/lib/flipper/ui.rb +4 -7
  26. data/lib/flipper/version.rb +1 -1
  27. data/spec/flipper/ui/actions/actors_gate_spec.rb +9 -13
  28. data/spec/flipper/ui/actions/feature_spec.rb +14 -16
  29. data/spec/flipper/ui/actions/features_spec.rb +49 -14
  30. data/spec/flipper/ui/actions/file_spec.rb +0 -10
  31. data/spec/flipper/ui/actions/groups_gate_spec.rb +0 -6
  32. data/spec/flipper/ui/actions/percentage_of_actors_gate_spec.rb +0 -2
  33. data/spec/flipper/ui/actions/percentage_of_time_gate_spec.rb +0 -2
  34. data/spec/flipper/ui/configuration_spec.rb +69 -34
  35. data/spec/flipper/ui/decorators/feature_spec.rb +2 -32
  36. data/spec/flipper/ui_spec.rb +1 -1
  37. metadata +21 -133
  38. data/docs/ui/images/configured-ui.png +0 -0
  39. data/docs/ui/images/environment-banner.png +0 -0
  40. data/lib/flipper/ui/assets/stylesheets/.DS_Store +0 -0
  41. data/lib/flipper/ui/assets/stylesheets/application.scss +0 -19
  42. data/lib/flipper/ui/assets/stylesheets/bootstrap/_alert.scss +0 -51
  43. data/lib/flipper/ui/assets/stylesheets/bootstrap/_badge.scss +0 -47
  44. data/lib/flipper/ui/assets/stylesheets/bootstrap/_breadcrumb.scss +0 -38
  45. data/lib/flipper/ui/assets/stylesheets/bootstrap/_button-group.scss +0 -166
  46. data/lib/flipper/ui/assets/stylesheets/bootstrap/_buttons.scss +0 -143
  47. data/lib/flipper/ui/assets/stylesheets/bootstrap/_card.scss +0 -270
  48. data/lib/flipper/ui/assets/stylesheets/bootstrap/_carousel.scss +0 -191
  49. data/lib/flipper/ui/assets/stylesheets/bootstrap/_close.scss +0 -34
  50. data/lib/flipper/ui/assets/stylesheets/bootstrap/_code.scss +0 -56
  51. data/lib/flipper/ui/assets/stylesheets/bootstrap/_custom-forms.scss +0 -297
  52. data/lib/flipper/ui/assets/stylesheets/bootstrap/_dropdown.scss +0 -131
  53. data/lib/flipper/ui/assets/stylesheets/bootstrap/_forms.scss +0 -333
  54. data/lib/flipper/ui/assets/stylesheets/bootstrap/_functions.scss +0 -86
  55. data/lib/flipper/ui/assets/stylesheets/bootstrap/_grid.scss +0 -52
  56. data/lib/flipper/ui/assets/stylesheets/bootstrap/_images.scss +0 -42
  57. data/lib/flipper/ui/assets/stylesheets/bootstrap/_input-group.scss +0 -159
  58. data/lib/flipper/ui/assets/stylesheets/bootstrap/_jumbotron.scss +0 -16
  59. data/lib/flipper/ui/assets/stylesheets/bootstrap/_list-group.scss +0 -115
  60. data/lib/flipper/ui/assets/stylesheets/bootstrap/_media.scss +0 -8
  61. data/lib/flipper/ui/assets/stylesheets/bootstrap/_mixins.scss +0 -42
  62. data/lib/flipper/ui/assets/stylesheets/bootstrap/_modal.scss +0 -168
  63. data/lib/flipper/ui/assets/stylesheets/bootstrap/_nav.scss +0 -118
  64. data/lib/flipper/ui/assets/stylesheets/bootstrap/_navbar.scss +0 -311
  65. data/lib/flipper/ui/assets/stylesheets/bootstrap/_pagination.scss +0 -77
  66. data/lib/flipper/ui/assets/stylesheets/bootstrap/_popover.scss +0 -183
  67. data/lib/flipper/ui/assets/stylesheets/bootstrap/_print.scss +0 -124
  68. data/lib/flipper/ui/assets/stylesheets/bootstrap/_progress.scss +0 -33
  69. data/lib/flipper/ui/assets/stylesheets/bootstrap/_reboot.scss +0 -482
  70. data/lib/flipper/ui/assets/stylesheets/bootstrap/_root.scss +0 -19
  71. data/lib/flipper/ui/assets/stylesheets/bootstrap/_tables.scss +0 -180
  72. data/lib/flipper/ui/assets/stylesheets/bootstrap/_tooltip.scss +0 -115
  73. data/lib/flipper/ui/assets/stylesheets/bootstrap/_transitions.scss +0 -36
  74. data/lib/flipper/ui/assets/stylesheets/bootstrap/_type.scss +0 -125
  75. data/lib/flipper/ui/assets/stylesheets/bootstrap/_utilities.scss +0 -14
  76. data/lib/flipper/ui/assets/stylesheets/bootstrap/_variables.scss +0 -894
  77. data/lib/flipper/ui/assets/stylesheets/bootstrap/bootstrap-grid.scss +0 -32
  78. data/lib/flipper/ui/assets/stylesheets/bootstrap/bootstrap-reboot.scss +0 -12
  79. data/lib/flipper/ui/assets/stylesheets/bootstrap/bootstrap.scss +0 -42
  80. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_alert.scss +0 -13
  81. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_background-variant.scss +0 -21
  82. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_badge.scss +0 -12
  83. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_border-radius.scss +0 -35
  84. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_box-shadow.scss +0 -5
  85. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_breakpoints.scss +0 -123
  86. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_buttons.scss +0 -109
  87. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_caret.scss +0 -65
  88. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_clearfix.scss +0 -7
  89. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_float.scss +0 -11
  90. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_forms.scss +0 -137
  91. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_gradients.scss +0 -45
  92. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_grid-framework.scss +0 -67
  93. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_grid.scss +0 -52
  94. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_hover.scss +0 -39
  95. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_image.scss +0 -36
  96. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_list-group.scss +0 -21
  97. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_lists.scss +0 -7
  98. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_nav-divider.scss +0 -10
  99. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_navbar-align.scss +0 -10
  100. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_pagination.scss +0 -22
  101. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_reset-text.scss +0 -17
  102. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_resize.scss +0 -6
  103. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_screen-reader.scss +0 -35
  104. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_size.scss +0 -6
  105. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_table-row.scss +0 -30
  106. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_text-emphasis.scss +0 -14
  107. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_text-hide.scss +0 -9
  108. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_text-truncate.scss +0 -8
  109. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_transition.scss +0 -9
  110. data/lib/flipper/ui/assets/stylesheets/bootstrap/mixins/_visibility.scss +0 -7
  111. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_align.scss +0 -8
  112. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_background.scss +0 -19
  113. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_borders.scss +0 -59
  114. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_clearfix.scss +0 -3
  115. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_display.scss +0 -38
  116. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_embed.scss +0 -52
  117. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_flex.scss +0 -46
  118. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_float.scss +0 -9
  119. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_position.scss +0 -36
  120. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_screenreaders.scss +0 -11
  121. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_sizing.scss +0 -12
  122. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_spacing.scss +0 -51
  123. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_text.scss +0 -52
  124. data/lib/flipper/ui/assets/stylesheets/bootstrap/utilities/_visibility.scss +0 -11
  125. data/lib/flipper/ui/assets/stylesheets/primer/.scss-lint.yml +0 -446
  126. data/lib/flipper/ui/assets/stylesheets/primer/_alerts.scss +0 -106
  127. data/lib/flipper/ui/assets/stylesheets/primer/_avatars.scss +0 -36
  128. data/lib/flipper/ui/assets/stylesheets/primer/_base.scss +0 -40
  129. data/lib/flipper/ui/assets/stylesheets/primer/_blankslate.scss +0 -96
  130. data/lib/flipper/ui/assets/stylesheets/primer/_buttons.scss +0 -404
  131. data/lib/flipper/ui/assets/stylesheets/primer/_counter.scss +0 -10
  132. data/lib/flipper/ui/assets/stylesheets/primer/_filter-list.scss +0 -68
  133. data/lib/flipper/ui/assets/stylesheets/primer/_flex-table.scss +0 -20
  134. data/lib/flipper/ui/assets/stylesheets/primer/_forms.scss +0 -756
  135. data/lib/flipper/ui/assets/stylesheets/primer/_layout.scss +0 -69
  136. data/lib/flipper/ui/assets/stylesheets/primer/_menu.scss +0 -113
  137. data/lib/flipper/ui/assets/stylesheets/primer/_mixins.scss +0 -53
  138. data/lib/flipper/ui/assets/stylesheets/primer/_normalize.scss +0 -425
  139. data/lib/flipper/ui/assets/stylesheets/primer/_states.scss +0 -32
  140. data/lib/flipper/ui/assets/stylesheets/primer/_tabnav.scss +0 -65
  141. data/lib/flipper/ui/assets/stylesheets/primer/_tooltips.scss +0 -255
  142. data/lib/flipper/ui/assets/stylesheets/primer/_truncate.scss +0 -27
  143. data/lib/flipper/ui/assets/stylesheets/primer/_type.scss +0 -92
  144. data/lib/flipper/ui/assets/stylesheets/primer/_utility.scss +0 -73
  145. data/lib/flipper/ui/assets/stylesheets/primer/_variables.scss +0 -34
  146. data/lib/flipper/ui/assets/stylesheets/primer/primer.scss +0 -39
  147. data/lib/flipper/ui/eruby.rb +0 -11
  148. data/lib/flipper/ui/public/fonts/bootstrap/glyphicons-halflings-regular.eot +0 -0
  149. data/lib/flipper/ui/public/fonts/bootstrap/glyphicons-halflings-regular.svg +0 -288
  150. data/lib/flipper/ui/public/fonts/bootstrap/glyphicons-halflings-regular.ttf +0 -0
  151. data/lib/flipper/ui/public/fonts/bootstrap/glyphicons-halflings-regular.woff +0 -0
  152. data/lib/flipper/ui/public/fonts/bootstrap/glyphicons-halflings-regular.woff2 +0 -0
  153. data/lib/flipper/ui/public/images/remove.png +0 -0
  154. data/lib/flipper/ui/public/octicons/octicons.less +0 -235
  155. data/lib/flipper/ui/public/octicons/sprockets-octicons.scss +0 -232
@@ -1,43 +1,62 @@
1
1
  <% if @show_blank_slate %>
2
2
  <div class="jumbotron text-center">
3
- <span class="mega-octicon octicon-plus"></span>
4
- <span class="mega-octicon octicon-list-unordered"></span>
5
- <span class="mega-octicon octicon-zap"></span>
6
- <h4>But I've got a blank space baby...</h4>
7
- <p>And I'll flip your features.</p>
8
- <div class="embed-responsive embed-responsive-16by9">
9
- <iframe class="embed-responsive-item" width="560" height="315" src="https://www.youtube.com/embed/e-ORhEE9VVg" frameborder="0" allowfullscreen></iframe>
10
- </div>
3
+ <% if Flipper::UI.configuration.fun %>
4
+ <h4>But I've got a blank space baby...</h4>
5
+ <p>And I'll flip your features.</p>
6
+ <%- if Flipper::UI.configuration.feature_creation_enabled -%>
7
+ <p>
8
+ <a class="btn btn-primary btn-sm" href="<%= script_name %>/features/new">Add Feature</a>
9
+ </p>
10
+ <%- end -%>
11
+ <div class="embed-responsive embed-responsive-16by9">
12
+ <iframe class="embed-responsive-item" width="560" height="315" src="https://www.youtube.com/embed/e-ORhEE9VVg" frameborder="0" allowfullscreen></iframe>
13
+ </div>
14
+ <% else %>
15
+ <h4>Getting Started</h4>
16
+ <p class="mb-1">You have not added any features to configure yet.</p>
17
+ <%- if Flipper::UI.configuration.feature_creation_enabled -%>
18
+ <p class="mt-2">
19
+ <a class="btn btn-primary btn-sm" href="<%= script_name %>/features/new">Add Feature</a>
20
+ </p>
21
+ <% else %>
22
+ <p>
23
+ Check the <a href="https://github.com/jnunemaker/flipper#examples">examples</a> to
24
+ learn how to add one.
25
+ </p>
26
+ <%- end -%>
27
+ <% end %>
11
28
  </div>
12
29
  <% else %>
13
30
  <div class="card">
14
- <h4 class="card-header">Features</h4>
15
- <table class="table">
16
- <thead>
17
- <tr class="d-flex">
18
- <th class="col-1">
19
- <span class="octicon octicon-squirrel"></span>
20
- </th>
21
- <th class="col">Feature</th>
22
- <th class="col">Enabled Gates</th>
23
- </tr>
24
- </thead>
25
- <tbody>
26
- <% @features.each do |feature| %>
27
- <tr class="d-flex">
28
- <td class="col-1">
29
- <span class="octicon octicon-squirrel <%= feature.color_class %>"></span>
30
- </td>
31
- <td class="col">
32
- <a href="<%= "#{script_name}/features/#{feature.key}" %>">
33
- <%= feature.key %></a>
34
- </td>
35
- <td class="col">
36
- <%= feature.pretty_enabled_gate_names %>
37
- </td>
38
- </tr>
39
- <% end %>
40
- </tbody>
41
- </table>
31
+ <div class="card-header">
32
+ <%- if Flipper::UI.configuration.feature_creation_enabled -%>
33
+ <div class="float-right">
34
+ <a class="btn btn-primary btn-sm" href="<%= script_name %>/features/new">Add Feature</a>
35
+ </div>
36
+ <%- end -%>
37
+ <h4 class="m-0">Features</h4>
38
+ </div>
39
+ <div class="card-body py-0">
40
+ <% @features.each do |feature| %>
41
+ <div class="feature row align-items-center mt-0 px-3 border-bottom">
42
+ <div class="col-1 col-md-auto">
43
+ <span class="octicon octicon-squirrel <%= feature.color_class %>" data-toggle="tooltip" title=<%= feature.state.to_s.capitalize %>></span>
44
+ </div>
45
+ <div class="col-10">
46
+ <a href="<%= "#{script_name}/features/#{feature.key}" %>" class="d-block px-0 py-3 btn text-left text-dark">
47
+ <div class="text-truncate" style="font-weight: 500"><%= feature.key %></div>
48
+ <% if Flipper::UI.configuration.show_feature_description_in_list? && Flipper::UI::Util.present?(feature.description) %>
49
+ <div class="text-muted font-weight-light" style="line-height: 1.4; white-space: initial; padding: 8px 0">
50
+ <%= feature.description %>
51
+ </div>
52
+ <% end %>
53
+ <div class="text-muted text-truncate">
54
+ <%== feature.gates_in_words %>
55
+ </div>
56
+ </a>
57
+ </div>
58
+ </div>
59
+ <% end %>
60
+ </div>
42
61
  </div>
43
62
  <% end %>
@@ -6,17 +6,12 @@
6
6
  <meta http-equiv="X-UA-Compatible" content="IE=edge">
7
7
  <meta name="viewport" content="width=device-width, initial-scale=1">
8
8
 
9
+ <link rel="stylesheet" href="https://maxcdn.bootstrapcdn.com/bootstrap/4.0.0/css/bootstrap.min.css" integrity="sha384-Gn5384xqQ1aoWXA+058RXPxPg6fy4IWvTNh0E263XmFcJlSAwiGgFAW/dAiS6JXm" crossorigin="anonymous">
9
10
  <link rel="stylesheet" href="<%= script_name %>/octicons/octicons.css">
10
11
  <link rel="stylesheet" href="<%= script_name %>/css/application.css">
11
-
12
- <!-- HTML5 shim and Respond.js for IE8 support of HTML5 elements and media queries -->
13
- <!--[if lt IE 9]>
14
- <script src="https://oss.maxcdn.com/html5shiv/3.7.2/html5shiv.min.js"></script>
15
- <script src="https://oss.maxcdn.com/respond/1.4.2/respond.min.js"></script>
16
- <![endif]-->
17
12
  </head>
18
- <body>
19
- <div class="container">
13
+ <body class="py-4">
14
+ <div class="container mw-600">
20
15
  <%- unless Flipper::UI.configuration.banner_text.nil? -%>
21
16
  <div class="alert alert-<%= Flipper::UI.configuration.banner_class %> text-center font-weight-bold">
22
17
  <%= Flipper::UI.configuration.banner_text %>
@@ -24,7 +19,7 @@
24
19
  <%- end -%>
25
20
 
26
21
  <nav aria-label="breadcrumb">
27
- <ol class="breadcrumb bg-white border border-light align-items-center">
22
+ <ol class="breadcrumb bg-white border align-items-center mb-4">
28
23
  <% @breadcrumbs.each do |breadcrumb| %>
29
24
  <li class="breadcrumb-item <% if breadcrumb.active? %>active<% end %>">
30
25
  <% if breadcrumb.active? %>
@@ -34,11 +29,6 @@
34
29
  <% end %>
35
30
  </li>
36
31
  <% end %>
37
- <li class="ml-auto">
38
- <%- if Flipper::UI.configuration.feature_creation_enabled -%>
39
- <a class="btn btn-sm btn-light" href="<%= script_name %>/features/new">Add Feature</a>
40
- <%- end -%>
41
- </li>
42
32
  </ol>
43
33
  </nav>
44
34
 
data/lib/flipper/ui.rb CHANGED
@@ -30,9 +30,6 @@ module Flipper
30
30
  "deprecated. This configuration option has moved to Flipper::UI::Configuration"
31
31
  end
32
32
  end
33
-
34
- # Public: Set attributes on this instance to customize UI text
35
- attr_reader :configuration
36
33
  end
37
34
 
38
35
  def self.root
@@ -41,15 +38,15 @@ module Flipper
41
38
 
42
39
  def self.app(flipper = nil, options = {})
43
40
  env_key = options.fetch(:env_key, 'flipper')
44
- app = ->() { [200, { 'Content-Type' => 'text/html' }, ['']] }
41
+ rack_protection_options = options.fetch(:rack_protection, use: :authenticity_token)
42
+ app = ->(_) { [200, { 'Content-Type' => 'text/html' }, ['']] }
45
43
  builder = Rack::Builder.new
46
44
  yield builder if block_given?
47
- builder.use Rack::Protection
48
- builder.use Rack::Protection::AuthenticityToken
45
+ builder.use Rack::Protection, rack_protection_options
49
46
  builder.use Rack::MethodOverride
50
47
  builder.use Flipper::Middleware::SetupEnv, flipper, env_key: env_key
51
48
  builder.use Flipper::Middleware::Memoizer, env_key: env_key
52
- builder.use Middleware, env_key: env_key
49
+ builder.use Flipper::UI::Middleware, env_key: env_key
53
50
  builder.run app
54
51
  klass = self
55
52
  builder.define_singleton_method(:inspect) { klass.inspect } # pretty rake routes output
@@ -1,3 +1,3 @@
1
1
  module Flipper
2
- VERSION = '0.16.2'.freeze
2
+ VERSION = '0.20.0'.freeze
3
3
  end
@@ -44,7 +44,7 @@ RSpec.describe Flipper::UI::Actions::ActorsGate do
44
44
 
45
45
  describe 'POST /features/:feature/actors' do
46
46
  context 'enabling an actor' do
47
- let(:value) { 'User:6' }
47
+ let(:value) { 'User;6' }
48
48
 
49
49
  before do
50
50
  post 'features/search/actors',
@@ -53,7 +53,7 @@ RSpec.describe Flipper::UI::Actions::ActorsGate do
53
53
  end
54
54
 
55
55
  it 'adds item to members' do
56
- expect(flipper[:search].actors_value).to include('User:6')
56
+ expect(flipper[:search].actors_value).to include('User;6')
57
57
  end
58
58
 
59
59
  it 'redirects back to feature' do
@@ -62,10 +62,10 @@ RSpec.describe Flipper::UI::Actions::ActorsGate do
62
62
  end
63
63
 
64
64
  context 'value contains whitespace' do
65
- let(:value) { ' User:6 ' }
65
+ let(:value) { ' User;6 ' }
66
66
 
67
67
  it 'adds item without whitespace' do
68
- expect(flipper[:search].actors_value).to include('User:6')
68
+ expect(flipper[:search].actors_value).to include('User;6')
69
69
  end
70
70
  end
71
71
 
@@ -73,39 +73,35 @@ RSpec.describe Flipper::UI::Actions::ActorsGate do
73
73
  context 'empty value' do
74
74
  let(:value) { '' }
75
75
 
76
- # rubocop:disable Metrics/LineLength
77
76
  it 'redirects back to feature' do
78
77
  expect(last_response.status).to be(302)
79
78
  expect(last_response.headers['Location']).to eq('/features/search/actors?error=%22%22+is+not+a+valid+actor+value.')
80
79
  end
81
- # rubocop:enable Metrics/LineLength
82
80
  end
83
81
 
84
82
  context 'nil value' do
85
83
  let(:value) { nil }
86
84
 
87
- # rubocop:disable Metrics/LineLength
88
85
  it 'redirects back to feature' do
89
86
  expect(last_response.status).to be(302)
90
87
  expect(last_response.headers['Location']).to eq('/features/search/actors?error=%22%22+is+not+a+valid+actor+value.')
91
88
  end
92
- # rubocop:enable Metrics/LineLength
93
89
  end
94
90
  end
95
91
  end
96
92
 
97
93
  context 'disabling an actor' do
98
- let(:value) { 'User:6' }
94
+ let(:value) { 'User;6' }
99
95
 
100
96
  before do
101
- flipper[:search].enable_actor Flipper::Actor.new('User:6')
97
+ flipper[:search].enable_actor Flipper::Actor.new('User;6')
102
98
  post 'features/search/actors',
103
99
  { 'value' => value, 'operation' => 'disable', 'authenticity_token' => token },
104
100
  'rack.session' => session
105
101
  end
106
102
 
107
103
  it 'removes item from members' do
108
- expect(flipper[:search].actors_value).not_to include('User:6')
104
+ expect(flipper[:search].actors_value).not_to include('User;6')
109
105
  end
110
106
 
111
107
  it 'redirects back to feature' do
@@ -114,10 +110,10 @@ RSpec.describe Flipper::UI::Actions::ActorsGate do
114
110
  end
115
111
 
116
112
  context 'value contains whitespace' do
117
- let(:value) { ' User:6 ' }
113
+ let(:value) { ' User;6 ' }
118
114
 
119
115
  it 'removes item without whitespace' do
120
- expect(flipper[:search].actors_value).not_to include('User:6')
116
+ expect(flipper[:search].actors_value).not_to include('User;6')
121
117
  end
122
118
  end
123
119
  end
@@ -70,6 +70,15 @@ RSpec.describe Flipper::UI::Actions::Feature do
70
70
 
71
71
  describe 'GET /features/:feature' do
72
72
  before do
73
+ Flipper::UI.configure do |config|
74
+ config.descriptions_source = lambda { |_keys|
75
+ {
76
+ "stats" => "Most awesome stats",
77
+ "search" => "Most in-depth search",
78
+ }
79
+ }
80
+ end
81
+
73
82
  get '/features/search'
74
83
  end
75
84
 
@@ -81,10 +90,11 @@ RSpec.describe Flipper::UI::Actions::Feature do
81
90
  expect(last_response.body).to include('search')
82
91
  expect(last_response.body).to include('Enable')
83
92
  expect(last_response.body).to include('Disable')
84
- expect(last_response.body).to include('Actors')
85
- expect(last_response.body).to include('Groups')
86
- expect(last_response.body).to include('Percentage of Time')
87
- expect(last_response.body).to include('Percentage of Actors')
93
+ expect(last_response.body).to include('No actors enabled')
94
+ expect(last_response.body).to include('No groups enabled')
95
+ expect(last_response.body).to include('Enabled for 0% of time')
96
+ expect(last_response.body).to include('Enabled for 0% of actors')
97
+ expect(last_response.body).to include('Most in-depth search')
88
98
  end
89
99
  end
90
100
 
@@ -99,12 +109,6 @@ RSpec.describe Flipper::UI::Actions::Feature do
99
109
 
100
110
  it 'renders template' do
101
111
  expect(last_response.body).to include('search_features')
102
- expect(last_response.body).to include('Enable')
103
- expect(last_response.body).to include('Disable')
104
- expect(last_response.body).to include('Actors')
105
- expect(last_response.body).to include('Groups')
106
- expect(last_response.body).to include('Percentage of Time')
107
- expect(last_response.body).to include('Percentage of Actors')
108
112
  end
109
113
  end
110
114
 
@@ -119,12 +123,6 @@ RSpec.describe Flipper::UI::Actions::Feature do
119
123
 
120
124
  it 'renders template' do
121
125
  expect(last_response.body).to include('a/b')
122
- expect(last_response.body).to include('Enable')
123
- expect(last_response.body).to include('Disable')
124
- expect(last_response.body).to include('Actors')
125
- expect(last_response.body).to include('Groups')
126
- expect(last_response.body).to include('Percentage of Time')
127
- expect(last_response.body).to include('Percentage of Actors')
128
126
  end
129
127
  end
130
128
  end
@@ -13,19 +13,58 @@ RSpec.describe Flipper::UI::Actions::Features do
13
13
  end
14
14
 
15
15
  describe 'GET /features' do
16
- before do
17
- flipper[:stats].enable
18
- flipper[:search].enable
19
- get '/features'
20
- end
16
+ context "when there are some features" do
17
+ before do
18
+ flipper[:stats].enable
19
+ flipper[:search].enable
20
+ get '/features'
21
+ end
21
22
 
22
- it 'responds with success' do
23
- expect(last_response.status).to be(200)
23
+ it 'responds with success' do
24
+ expect(last_response.status).to be(200)
25
+ end
26
+
27
+ it 'renders template' do
28
+ expect(last_response.body).to include('stats')
29
+ expect(last_response.body).to include('search')
30
+ end
24
31
  end
25
32
 
26
- it 'renders template' do
27
- expect(last_response.body).to include('stats')
28
- expect(last_response.body).to include('search')
33
+ context "when there are no features to list" do
34
+ before do
35
+ @original_fun_enabled = Flipper::UI.configuration.fun
36
+ Flipper::UI.configuration.fun = fun_mode
37
+ end
38
+
39
+ after do
40
+ Flipper::UI.configuration.fun = @original_fun_enabled
41
+ end
42
+
43
+ context "when fun mode is enabled" do
44
+ let(:fun_mode) { true }
45
+ before { get '/features' }
46
+
47
+ it 'responds with success' do
48
+ expect(last_response.status).to be(200)
49
+ end
50
+
51
+ it 'renders template' do
52
+ expect(last_response.body).to include('And I\'ll flip your features.')
53
+ end
54
+ end
55
+
56
+ context "when fun mode is disabled" do
57
+ let(:fun_mode) { false }
58
+ before { get '/features' }
59
+
60
+ it 'responds with success' do
61
+ expect(last_response.status).to be(200)
62
+ end
63
+
64
+ it 'renders template' do
65
+ expect(last_response.body).to include('You have not added any features to configure yet.')
66
+ end
67
+ end
29
68
  end
30
69
  end
31
70
 
@@ -72,12 +111,10 @@ RSpec.describe Flipper::UI::Actions::Features do
72
111
  expect(flipper.features.map(&:key)).to eq([])
73
112
  end
74
113
 
75
- # rubocop:disable Metrics/LineLength
76
114
  it 'redirects back to feature' do
77
115
  expect(last_response.status).to be(302)
78
116
  expect(last_response.headers['Location']).to eq('/features/new?error=%22%22+is+not+a+valid+feature+name.')
79
117
  end
80
- # rubocop:enable Metrics/LineLength
81
118
  end
82
119
 
83
120
  context 'nil feature name' do
@@ -87,12 +124,10 @@ RSpec.describe Flipper::UI::Actions::Features do
87
124
  expect(flipper.features.map(&:key)).to eq([])
88
125
  end
89
126
 
90
- # rubocop:disable Metrics/LineLength
91
127
  it 'redirects back to feature' do
92
128
  expect(last_response.status).to be(302)
93
129
  expect(last_response.headers['Location']).to eq('/features/new?error=%22%22+is+not+a+valid+feature+name.')
94
130
  end
95
- # rubocop:enable Metrics/LineLength
96
131
  end
97
132
  end
98
133
  end
@@ -21,16 +21,6 @@ RSpec.describe Flipper::UI::Actions::File do
21
21
  end
22
22
  end
23
23
 
24
- describe 'GET /fonts/bootstrap/glyphicons-halflings-regular.eot' do
25
- before do
26
- get '/fonts/bootstrap/glyphicons-halflings-regular.eot'
27
- end
28
-
29
- it 'responds with 200' do
30
- expect(last_response.status).to be(200)
31
- end
32
- end
33
-
34
24
  describe 'GET /octicons/octicons.eot' do
35
25
  before do
36
26
  get '/octicons/octicons.eot'
@@ -72,34 +72,28 @@ RSpec.describe Flipper::UI::Actions::GroupsGate do
72
72
  context 'unknown group name' do
73
73
  let(:group_name) { 'not_here' }
74
74
 
75
- # rubocop:disable Metrics/LineLength
76
75
  it 'redirects back to feature' do
77
76
  expect(last_response.status).to be(302)
78
77
  expect(last_response.headers['Location']).to eq('/features/search/groups?error=The+group+named+%22not_here%22+has+not+been+registered.')
79
78
  end
80
- # rubocop:enable Metrics/LineLength
81
79
  end
82
80
 
83
81
  context 'empty group name' do
84
82
  let(:group_name) { '' }
85
83
 
86
- # rubocop:disable Metrics/LineLength
87
84
  it 'redirects back to feature' do
88
85
  expect(last_response.status).to be(302)
89
86
  expect(last_response.headers['Location']).to eq('/features/search/groups?error=The+group+named+%22%22+has+not+been+registered.')
90
87
  end
91
- # rubocop:enable Metrics/LineLength
92
88
  end
93
89
 
94
90
  context 'nil group name' do
95
91
  let(:group_name) { nil }
96
92
 
97
- # rubocop:disable Metrics/LineLength
98
93
  it 'redirects back to feature' do
99
94
  expect(last_response.status).to be(302)
100
95
  expect(last_response.headers['Location']).to eq('/features/search/groups?error=The+group+named+%22%22+has+not+been+registered.')
101
96
  end
102
- # rubocop:enable Metrics/LineLength
103
97
  end
104
98
  end
105
99
  end
@@ -41,12 +41,10 @@ RSpec.describe Flipper::UI::Actions::PercentageOfActorsGate do
41
41
  expect(flipper[:search].percentage_of_actors_value).to be(0)
42
42
  end
43
43
 
44
- # rubocop:disable Metrics/LineLength
45
44
  it 'redirects back to feature' do
46
45
  expect(last_response.status).to be(302)
47
46
  expect(last_response.headers['Location']).to eq('/features/search?error=Invalid+percentage+of+actors+value%3A+value+must+be+a+positive+number+less+than+or+equal+to+100%2C+but+was+555')
48
47
  end
49
- # rubocop:enable Metrics/LineLength
50
48
  end
51
49
  end
52
50
  end
@@ -41,12 +41,10 @@ RSpec.describe Flipper::UI::Actions::PercentageOfTimeGate do
41
41
  expect(flipper[:search].percentage_of_time_value).to be(0)
42
42
  end
43
43
 
44
- # rubocop:disable Metrics/LineLength
45
44
  it 'redirects back to feature' do
46
45
  expect(last_response.status).to be(302)
47
46
  expect(last_response.headers['Location']).to eq('/features/search?error=Invalid+percentage+of+time+value%3A+value+must+be+a+positive+number+less+than+or+equal+to+100%2C+but+was+555')
48
47
  end
49
- # rubocop:enable Metrics/LineLength
50
48
  end
51
49
  end
52
50
  end
@@ -3,43 +3,10 @@ require 'helper'
3
3
  RSpec.describe Flipper::UI::Configuration do
4
4
  let(:configuration) { described_class.new }
5
5
 
6
- describe "#actors" do
7
- it "has default text" do
8
- expect(configuration.actors.title).to eq("Actors")
9
- expect(configuration.actors.description).to eq("Enable actors using the form above.")
10
- end
11
-
12
- it "can be updated" do
13
- configuration.actors.title = "Actors Section"
14
- expect(configuration.actors.title).to eq("Actors Section")
15
- end
16
- end
17
-
18
- describe "#groups" do
19
- it "has default text" do
20
- expect(configuration.groups.title).to eq("Groups")
21
- expect(configuration.groups.description).to eq("Enable groups using the form above.")
22
- end
23
- end
24
-
25
- describe "#percentage_of_actors" do
26
- it "has default text" do
27
- expect(configuration.percentage_of_actors.title).to eq("Percentage of Actors")
28
- expect(configuration.percentage_of_actors.description).to eq("Percentage of actors functions independently of percentage of time. If you enable 50% of Actors and 25% of Time then the feature will always be enabled for 50% of users and occasionally enabled 25% of the time for everyone.") # rubocop:disable Metrics/LineLength
29
- end
30
- end
31
-
32
- describe "#percentage_of_time" do
33
- it "has default text" do
34
- expect(configuration.percentage_of_time.title).to eq("Percentage of Time")
35
- expect(configuration.percentage_of_time.description).to eq("Percentage of actors functions independently of percentage of time. If you enable 50% of Actors and 25% of Time then the feature will always be enabled for 50% of users and occasionally enabled 25% of the time for everyone.") # rubocop:disable Metrics/LineLength
36
- end
37
- end
38
-
39
6
  describe "#delete" do
40
7
  it "has default text" do
41
8
  expect(configuration.delete.title).to eq("Danger Zone")
42
- expect(configuration.delete.description).to eq("Deleting a feature removes it from the list of features and disables it for everyone.") # rubocop:disable Metrics/LineLength
9
+ expect(configuration.delete.description).to eq("Deleting a feature removes it from the list of features and disables it for everyone.")
43
10
  end
44
11
  end
45
12
 
@@ -102,4 +69,72 @@ RSpec.describe Flipper::UI::Configuration do
102
69
  expect(configuration.feature_removal_enabled).to eq(false)
103
70
  end
104
71
  end
72
+
73
+ describe "#fun" do
74
+ it "has default value" do
75
+ expect(configuration.fun).to eq(true)
76
+ end
77
+
78
+ it "can be updated" do
79
+ configuration.fun = false
80
+ expect(configuration.fun).to eq(false)
81
+ end
82
+ end
83
+
84
+ describe "#descriptions_source" do
85
+ it "has default value" do
86
+ expect(configuration.descriptions_source.call(%w[foo bar])).to eq({})
87
+ end
88
+
89
+ context "descriptions source is provided" do
90
+ it "can be updated" do
91
+ configuration.descriptions_source = lambda do |_keys|
92
+ YAML.load_file(FlipperRoot.join('spec/support/descriptions.yml'))
93
+ end
94
+ keys = %w[some_awesome_feature foo]
95
+ result = configuration.descriptions_source.call(keys)
96
+ expected = {
97
+ "some_awesome_feature" => "Awesome feature description",
98
+ }
99
+ expect(result).to eq(expected)
100
+ end
101
+ end
102
+ end
103
+
104
+ describe "#show_feature_description_in_list" do
105
+ it "has default value" do
106
+ expect(configuration.show_feature_description_in_list).to eq(false)
107
+ end
108
+
109
+ it "can be updated" do
110
+ configuration.show_feature_description_in_list = true
111
+ expect(configuration.show_feature_description_in_list).to eq(true)
112
+ end
113
+ end
114
+
115
+ describe "#show_feature_description_in_list?" do
116
+ subject { configuration.show_feature_description_in_list? }
117
+
118
+ context 'when using_descriptions? is false and show_feature_description_in_list is false' do
119
+ it { is_expected.to eq(false) }
120
+ end
121
+
122
+ context 'when using_descriptions? is false and show_feature_description_in_list is true' do
123
+ before { configuration.show_feature_description_in_list = true }
124
+ it { is_expected.to eq(false) }
125
+ end
126
+
127
+ context 'when using_descriptions? is true and show_feature_description_in_list is false' do
128
+ before { allow(configuration).to receive(:using_descriptions?).and_return(true) }
129
+ it { is_expected.to eq(false) }
130
+ end
131
+
132
+ context 'when using_descriptions? is true and show_feature_description_in_list is true' do
133
+ before do
134
+ allow(configuration).to receive(:using_descriptions?).and_return(true)
135
+ configuration.show_feature_description_in_list = true
136
+ end
137
+ it { is_expected.to eq(true) }
138
+ end
139
+ end
105
140
  end
@@ -22,36 +22,6 @@ RSpec.describe Flipper::UI::Decorators::Feature do
22
22
  end
23
23
  end
24
24
 
25
- describe '#as_json' do
26
- before do
27
- @result = subject.as_json
28
- end
29
-
30
- it 'returns Hash' do
31
- expect(@result).to be_instance_of(Hash)
32
- end
33
-
34
- it 'includes id' do
35
- expect(@result['id']).to eq('some_awesome_feature')
36
- end
37
-
38
- it 'includes pretty name' do
39
- expect(@result['name']).to eq('Some Awesome Feature')
40
- end
41
-
42
- it 'includes state' do
43
- expect(@result['state']).to eq('off')
44
- end
45
-
46
- it 'includes gates' do
47
- gates = subject.gates.map do |gate|
48
- value = subject.gate_values[gate.key]
49
- Flipper::UI::Decorators::Gate.new(gate, value).as_json
50
- end
51
- expect(@result['gates']).to eq(gates)
52
- end
53
- end
54
-
55
25
  describe '#<=>' do
56
26
  let(:on) do
57
27
  flipper.enable(:on_a)
@@ -77,11 +47,11 @@ RSpec.describe Flipper::UI::Decorators::Feature do
77
47
  end
78
48
 
79
49
  it 'sorts :on before :off' do
80
- expect((on <=> conditional)).to be(-1)
50
+ expect((on <=> off)).to be(-1)
81
51
  end
82
52
 
83
53
  it 'sorts :conditional before :off' do
84
- expect((on <=> conditional)).to be(-1)
54
+ expect((conditional <=> off)).to be(-1)
85
55
  end
86
56
 
87
57
  it 'sorts on key for identical states' do