karafka-web 0.6.3 → 0.7.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (214) hide show
  1. checksums.yaml +4 -4
  2. checksums.yaml.gz.sig +0 -0
  3. data/.github/workflows/ci.yml +13 -4
  4. data/CHANGELOG.md +119 -5
  5. data/Gemfile +1 -0
  6. data/Gemfile.lock +27 -24
  7. data/README.md +2 -0
  8. data/bin/rspecs +6 -0
  9. data/certs/cert_chain.pem +21 -21
  10. data/docker-compose.yml +22 -0
  11. data/karafka-web.gemspec +3 -3
  12. data/lib/karafka/web/app.rb +6 -2
  13. data/lib/karafka/web/cli.rb +51 -47
  14. data/lib/karafka/web/config.rb +33 -9
  15. data/lib/karafka/web/contracts/base.rb +32 -0
  16. data/lib/karafka/web/contracts/config.rb +63 -0
  17. data/lib/karafka/web/deserializer.rb +10 -1
  18. data/lib/karafka/web/errors.rb +29 -7
  19. data/lib/karafka/web/installer.rb +58 -148
  20. data/lib/karafka/web/management/base.rb +34 -0
  21. data/lib/karafka/web/management/clean_boot_file.rb +31 -0
  22. data/lib/karafka/web/management/create_initial_states.rb +101 -0
  23. data/lib/karafka/web/management/create_topics.rb +127 -0
  24. data/lib/karafka/web/management/delete_topics.rb +28 -0
  25. data/lib/karafka/web/management/enable.rb +82 -0
  26. data/lib/karafka/web/management/extend_boot_file.rb +37 -0
  27. data/lib/karafka/web/processing/consumer.rb +73 -17
  28. data/lib/karafka/web/processing/consumers/aggregators/base.rb +56 -0
  29. data/lib/karafka/web/processing/consumers/aggregators/metrics.rb +154 -0
  30. data/lib/karafka/web/processing/consumers/aggregators/state.rb +180 -0
  31. data/lib/karafka/web/processing/consumers/contracts/aggregated_stats.rb +32 -0
  32. data/lib/karafka/web/processing/consumers/contracts/metrics.rb +53 -0
  33. data/lib/karafka/web/processing/consumers/contracts/process.rb +19 -0
  34. data/lib/karafka/web/processing/consumers/contracts/state.rb +49 -0
  35. data/lib/karafka/web/processing/consumers/contracts/topic_stats.rb +21 -0
  36. data/lib/karafka/web/processing/consumers/metrics.rb +29 -0
  37. data/lib/karafka/web/processing/consumers/schema_manager.rb +56 -0
  38. data/lib/karafka/web/processing/consumers/state.rb +6 -9
  39. data/lib/karafka/web/processing/time_series_tracker.rb +130 -0
  40. data/lib/karafka/web/tracking/consumers/contracts/consumer_group.rb +2 -2
  41. data/lib/karafka/web/tracking/consumers/contracts/job.rb +2 -1
  42. data/lib/karafka/web/tracking/consumers/contracts/partition.rb +14 -1
  43. data/lib/karafka/web/tracking/consumers/contracts/report.rb +10 -8
  44. data/lib/karafka/web/tracking/consumers/contracts/subscription_group.rb +2 -2
  45. data/lib/karafka/web/tracking/consumers/contracts/topic.rb +2 -2
  46. data/lib/karafka/web/tracking/consumers/listeners/processing.rb +6 -2
  47. data/lib/karafka/web/tracking/consumers/listeners/statistics.rb +15 -1
  48. data/lib/karafka/web/tracking/consumers/reporter.rb +14 -6
  49. data/lib/karafka/web/tracking/consumers/sampler.rb +80 -39
  50. data/lib/karafka/web/tracking/contracts/error.rb +2 -1
  51. data/lib/karafka/web/ui/app.rb +20 -10
  52. data/lib/karafka/web/ui/base.rb +56 -6
  53. data/lib/karafka/web/ui/controllers/base.rb +28 -0
  54. data/lib/karafka/web/ui/controllers/become_pro.rb +1 -1
  55. data/lib/karafka/web/ui/controllers/cluster.rb +12 -6
  56. data/lib/karafka/web/ui/controllers/consumers.rb +4 -2
  57. data/lib/karafka/web/ui/controllers/dashboard.rb +32 -0
  58. data/lib/karafka/web/ui/controllers/errors.rb +19 -6
  59. data/lib/karafka/web/ui/controllers/jobs.rb +4 -2
  60. data/lib/karafka/web/ui/controllers/requests/params.rb +28 -0
  61. data/lib/karafka/web/ui/controllers/responses/redirect.rb +29 -0
  62. data/lib/karafka/web/ui/helpers/application_helper.rb +57 -14
  63. data/lib/karafka/web/ui/helpers/paths_helper.rb +48 -0
  64. data/lib/karafka/web/ui/lib/hash_proxy.rb +18 -6
  65. data/lib/karafka/web/ui/lib/paginations/base.rb +61 -0
  66. data/lib/karafka/web/ui/lib/paginations/offset_based.rb +96 -0
  67. data/lib/karafka/web/ui/lib/paginations/page_based.rb +70 -0
  68. data/lib/karafka/web/ui/lib/paginations/paginators/arrays.rb +33 -0
  69. data/lib/karafka/web/ui/lib/paginations/paginators/base.rb +23 -0
  70. data/lib/karafka/web/ui/lib/paginations/paginators/partitions.rb +52 -0
  71. data/lib/karafka/web/ui/lib/paginations/paginators/sets.rb +85 -0
  72. data/lib/karafka/web/ui/lib/paginations/watermark_offsets_based.rb +75 -0
  73. data/lib/karafka/web/ui/lib/ttl_cache.rb +82 -0
  74. data/lib/karafka/web/ui/models/cluster_info.rb +59 -0
  75. data/lib/karafka/web/ui/models/consumers_metrics.rb +46 -0
  76. data/lib/karafka/web/ui/models/{state.rb → consumers_state.rb} +6 -2
  77. data/lib/karafka/web/ui/models/health.rb +37 -7
  78. data/lib/karafka/web/ui/models/message.rb +123 -39
  79. data/lib/karafka/web/ui/models/metrics/aggregated.rb +196 -0
  80. data/lib/karafka/web/ui/models/metrics/charts/aggregated.rb +50 -0
  81. data/lib/karafka/web/ui/models/metrics/charts/topics.rb +109 -0
  82. data/lib/karafka/web/ui/models/metrics/topics.rb +101 -0
  83. data/lib/karafka/web/ui/models/partition.rb +27 -0
  84. data/lib/karafka/web/ui/models/process.rb +12 -1
  85. data/lib/karafka/web/ui/models/status.rb +110 -22
  86. data/lib/karafka/web/ui/models/visibility_filter.rb +33 -0
  87. data/lib/karafka/web/ui/pro/app.rb +87 -19
  88. data/lib/karafka/web/ui/pro/controllers/cluster.rb +11 -0
  89. data/lib/karafka/web/ui/pro/controllers/consumers.rb +13 -7
  90. data/lib/karafka/web/ui/pro/controllers/dashboard.rb +54 -0
  91. data/lib/karafka/web/ui/pro/controllers/dlq.rb +1 -2
  92. data/lib/karafka/web/ui/pro/controllers/errors.rb +46 -10
  93. data/lib/karafka/web/ui/pro/controllers/explorer.rb +145 -15
  94. data/lib/karafka/web/ui/pro/controllers/health.rb +10 -2
  95. data/lib/karafka/web/ui/pro/controllers/messages.rb +62 -0
  96. data/lib/karafka/web/ui/pro/controllers/routing.rb +44 -0
  97. data/lib/karafka/web/ui/pro/views/consumers/_breadcrumbs.erb +7 -1
  98. data/lib/karafka/web/ui/pro/views/consumers/_consumer.erb +1 -1
  99. data/lib/karafka/web/ui/pro/views/consumers/_counters.erb +7 -5
  100. data/lib/karafka/web/ui/pro/views/consumers/consumer/_job.erb +3 -3
  101. data/lib/karafka/web/ui/pro/views/consumers/consumer/_metrics.erb +5 -4
  102. data/lib/karafka/web/ui/pro/views/consumers/consumer/_partition.erb +13 -4
  103. data/lib/karafka/web/ui/pro/views/consumers/consumer/_subscription_group.erb +3 -2
  104. data/lib/karafka/web/ui/pro/views/consumers/consumer/_tabs.erb +7 -0
  105. data/lib/karafka/web/ui/pro/views/consumers/details.erb +21 -0
  106. data/lib/karafka/web/ui/pro/views/consumers/index.erb +4 -2
  107. data/lib/karafka/web/ui/pro/views/dashboard/_ranges_selector.erb +39 -0
  108. data/lib/karafka/web/ui/pro/views/dashboard/index.erb +82 -0
  109. data/lib/karafka/web/ui/pro/views/dlq/_topic.erb +1 -1
  110. data/lib/karafka/web/ui/pro/views/errors/_breadcrumbs.erb +8 -6
  111. data/lib/karafka/web/ui/pro/views/errors/_error.erb +2 -2
  112. data/lib/karafka/web/ui/pro/views/errors/_partition_option.erb +1 -1
  113. data/lib/karafka/web/ui/pro/views/errors/_table.erb +21 -0
  114. data/lib/karafka/web/ui/pro/views/errors/_title_with_select.erb +31 -0
  115. data/lib/karafka/web/ui/pro/views/errors/index.erb +9 -56
  116. data/lib/karafka/web/ui/pro/views/errors/partition.erb +17 -0
  117. data/lib/karafka/web/ui/pro/views/errors/show.erb +1 -1
  118. data/lib/karafka/web/ui/pro/views/explorer/_breadcrumbs.erb +6 -4
  119. data/lib/karafka/web/ui/pro/views/explorer/_filtered.erb +16 -0
  120. data/lib/karafka/web/ui/pro/views/explorer/_message.erb +14 -4
  121. data/lib/karafka/web/ui/pro/views/explorer/_no_topics.erb +7 -0
  122. data/lib/karafka/web/ui/pro/views/explorer/_partition_option.erb +3 -3
  123. data/lib/karafka/web/ui/pro/views/explorer/_topic.erb +1 -1
  124. data/lib/karafka/web/ui/pro/views/explorer/index.erb +12 -8
  125. data/lib/karafka/web/ui/pro/views/explorer/messages/_headers.erb +15 -0
  126. data/lib/karafka/web/ui/pro/views/explorer/messages/_key.erb +12 -0
  127. data/lib/karafka/web/ui/pro/views/explorer/partition/_details.erb +35 -0
  128. data/lib/karafka/web/ui/pro/views/explorer/partition/_messages.erb +1 -0
  129. data/lib/karafka/web/ui/pro/views/explorer/partition.erb +6 -4
  130. data/lib/karafka/web/ui/pro/views/explorer/show.erb +48 -5
  131. data/lib/karafka/web/ui/pro/views/explorer/topic/_details.erb +23 -0
  132. data/lib/karafka/web/ui/pro/views/explorer/topic/_empty.erb +3 -0
  133. data/lib/karafka/web/ui/pro/views/explorer/topic/_limited.erb +4 -0
  134. data/lib/karafka/web/ui/pro/views/explorer/topic.erb +51 -0
  135. data/lib/karafka/web/ui/pro/views/health/_breadcrumbs.erb +16 -0
  136. data/lib/karafka/web/ui/pro/views/health/_no_data.erb +9 -0
  137. data/lib/karafka/web/ui/pro/views/health/_partition.erb +17 -15
  138. data/lib/karafka/web/ui/pro/views/health/_partition_offset.erb +40 -0
  139. data/lib/karafka/web/ui/pro/views/health/_tabs.erb +27 -0
  140. data/lib/karafka/web/ui/pro/views/health/offsets.erb +71 -0
  141. data/lib/karafka/web/ui/pro/views/health/overview.erb +68 -0
  142. data/lib/karafka/web/ui/pro/views/jobs/_job.erb +6 -3
  143. data/lib/karafka/web/ui/pro/views/jobs/index.erb +4 -1
  144. data/lib/karafka/web/ui/pro/views/routing/_consumer_group.erb +37 -0
  145. data/lib/karafka/web/ui/pro/views/routing/_detail.erb +25 -0
  146. data/lib/karafka/web/ui/pro/views/routing/_topic.erb +23 -0
  147. data/lib/karafka/web/ui/pro/views/routing/index.erb +10 -0
  148. data/lib/karafka/web/ui/pro/views/routing/show.erb +26 -0
  149. data/lib/karafka/web/ui/pro/views/shared/_navigation.erb +7 -10
  150. data/lib/karafka/web/ui/public/images/logo-gray.svg +28 -0
  151. data/lib/karafka/web/ui/public/javascripts/application.js +30 -0
  152. data/lib/karafka/web/ui/public/javascripts/chart.min.js +14 -0
  153. data/lib/karafka/web/ui/public/javascripts/charts.js +330 -0
  154. data/lib/karafka/web/ui/public/javascripts/datepicker.js +6 -0
  155. data/lib/karafka/web/ui/public/javascripts/live_poll.js +39 -12
  156. data/lib/karafka/web/ui/public/javascripts/offset_datetime.js +74 -0
  157. data/lib/karafka/web/ui/public/javascripts/tabs.js +59 -0
  158. data/lib/karafka/web/ui/public/stylesheets/application.css +11 -0
  159. data/lib/karafka/web/ui/public/stylesheets/datepicker.min.css +12 -0
  160. data/lib/karafka/web/ui/views/cluster/_no_partitions.erb +3 -0
  161. data/lib/karafka/web/ui/views/cluster/_partition.erb +20 -22
  162. data/lib/karafka/web/ui/views/cluster/index.erb +6 -1
  163. data/lib/karafka/web/ui/views/consumers/_consumer.erb +1 -1
  164. data/lib/karafka/web/ui/views/consumers/_counters.erb +6 -4
  165. data/lib/karafka/web/ui/views/consumers/_summary.erb +3 -3
  166. data/lib/karafka/web/ui/views/consumers/index.erb +3 -1
  167. data/lib/karafka/web/ui/views/dashboard/_feature_pro.erb +3 -0
  168. data/lib/karafka/web/ui/views/dashboard/_not_enough_data.erb +15 -0
  169. data/lib/karafka/web/ui/views/dashboard/_ranges_selector.erb +23 -0
  170. data/lib/karafka/web/ui/views/dashboard/index.erb +95 -0
  171. data/lib/karafka/web/ui/views/errors/_detail.erb +12 -0
  172. data/lib/karafka/web/ui/views/errors/_error.erb +2 -2
  173. data/lib/karafka/web/ui/views/errors/show.erb +1 -1
  174. data/lib/karafka/web/ui/views/jobs/index.erb +3 -1
  175. data/lib/karafka/web/ui/views/layout.erb +10 -3
  176. data/lib/karafka/web/ui/views/routing/_consumer_group.erb +8 -6
  177. data/lib/karafka/web/ui/views/routing/_detail.erb +2 -2
  178. data/lib/karafka/web/ui/views/routing/_topic.erb +1 -1
  179. data/lib/karafka/web/ui/views/routing/show.erb +1 -1
  180. data/lib/karafka/web/ui/views/shared/_brand.erb +2 -2
  181. data/lib/karafka/web/ui/views/shared/_chart.erb +14 -0
  182. data/lib/karafka/web/ui/views/shared/_content.erb +2 -2
  183. data/lib/karafka/web/ui/views/shared/_feature_pro.erb +1 -1
  184. data/lib/karafka/web/ui/views/shared/_flashes.erb +9 -0
  185. data/lib/karafka/web/ui/views/shared/_footer.erb +22 -0
  186. data/lib/karafka/web/ui/views/shared/_header.erb +15 -9
  187. data/lib/karafka/web/ui/views/shared/_live_poll.erb +7 -0
  188. data/lib/karafka/web/ui/views/shared/_navigation.erb +5 -8
  189. data/lib/karafka/web/ui/views/shared/_no_paginated_data.erb +9 -0
  190. data/lib/karafka/web/ui/views/shared/_pagination.erb +17 -13
  191. data/lib/karafka/web/ui/views/shared/_tab_nav.erb +7 -0
  192. data/lib/karafka/web/ui/views/shared/exceptions/not_found.erb +34 -32
  193. data/lib/karafka/web/ui/views/shared/exceptions/pro_only.erb +45 -43
  194. data/lib/karafka/web/ui/views/status/failures/_consumers_reports_schema_state.erb +15 -0
  195. data/lib/karafka/web/ui/views/status/failures/_enabled.erb +8 -0
  196. data/lib/karafka/web/ui/views/status/failures/_initial_consumers_metrics.erb +11 -0
  197. data/lib/karafka/web/ui/views/status/failures/{_initial_state.erb → _initial_consumers_state.erb} +3 -3
  198. data/lib/karafka/web/ui/views/status/failures/_partitions.erb +14 -6
  199. data/lib/karafka/web/ui/views/status/info/_components.erb +21 -1
  200. data/lib/karafka/web/ui/views/status/show.erb +62 -5
  201. data/lib/karafka/web/ui/views/status/successes/_enabled.erb +1 -0
  202. data/lib/karafka/web/ui/views/status/warnings/_replication.erb +19 -0
  203. data/lib/karafka/web/version.rb +1 -1
  204. data/lib/karafka/web.rb +11 -0
  205. data.tar.gz.sig +0 -0
  206. metadata +124 -39
  207. metadata.gz.sig +0 -0
  208. data/lib/karafka/web/processing/consumers/aggregator.rb +0 -130
  209. data/lib/karafka/web/tracking/contracts/base.rb +0 -34
  210. data/lib/karafka/web/ui/lib/paginate_array.rb +0 -38
  211. data/lib/karafka/web/ui/pro/views/explorer/_encryption_enabled.erb +0 -18
  212. data/lib/karafka/web/ui/pro/views/explorer/partition/_watermark_offsets.erb +0 -10
  213. data/lib/karafka/web/ui/pro/views/health/index.erb +0 -60
  214. /data/lib/karafka/web/ui/pro/views/explorer/{_detail.erb → messages/_detail.erb} +0 -0
@@ -1,4 +1,23 @@
1
+ <%
2
+ republish_path = root_path('messages', @message.topic, @message.partition, @message.offset, 'republish')
3
+ surrounding_path = explorer_path(@message.topic, @message.partition, @message.offset, 'surrounding')
4
+ %>
5
+
1
6
  <div class="container">
7
+ <div class="row mb-0">
8
+ <div class="col-sm-12 text-end">
9
+ <a href="<%= surrounding_path %>" class="btn btn-secondary btn-sm float-end ms-1">
10
+ &#8651;
11
+ Surrounding
12
+ </a>
13
+
14
+ <form action="<%= republish_path %>" method="post" class="confirm-action float-end">
15
+ <%== csrf_tag(republish_path) %>
16
+ <input type="submit" value="&#10227; Republish" class="btn btn-primary btn-sm"/>
17
+ </form>
18
+ </div>
19
+ </div>
20
+
2
21
  <div class="row mb-4">
3
22
  <div class="col-sm-12">
4
23
  <h5 class="mb-2">
@@ -13,10 +32,10 @@
13
32
  <div class="col-sm-12">
14
33
  <table class="processes bg-white table table-hover table-bordered table-striped mb-0 align-middle">
15
34
  <tbody>
16
- <% @message.metadata.to_h.except(:received_at).each do |k, v| %>
35
+ <% @message.metadata.to_h.except(:received_at, :key, :headers).each do |k, v| %>
17
36
  <%==
18
37
  partial(
19
- 'explorer/detail',
38
+ 'explorer/messages/detail',
20
39
  locals: {
21
40
  k: k,
22
41
  v: v
@@ -24,6 +43,30 @@
24
43
  )
25
44
  %>
26
45
  <% end %>
46
+
47
+ <%==
48
+ partial(
49
+ 'explorer/messages/detail',
50
+ locals: {
51
+ k: 'bytesize',
52
+ v: format_memory((@message.raw_payload&.bytesize || 0 / 1024.to_f).round(2))
53
+ }
54
+ )
55
+ %>
56
+
57
+ <%==
58
+ partial(
59
+ 'explorer/messages/key',
60
+ locals: { message: @message }
61
+ )
62
+ %>
63
+
64
+ <%==
65
+ partial(
66
+ 'explorer/messages/headers',
67
+ locals: { message: @message }
68
+ )
69
+ %>
27
70
  </tbody>
28
71
  </table>
29
72
  </div>
@@ -41,9 +84,9 @@
41
84
  </div>
42
85
  </div>
43
86
 
44
- <% if @decrypt %>
87
+ <% if @visibility_filter.payload?(@message) %>
45
88
  <div class="row">
46
- <div class="col-sm-12">
89
+ <div class="col-sm-12 mb-4">
47
90
  <% if @payload_error %>
48
91
  <%== partial 'explorer/failed_deserialization' %>
49
92
  <% end %>
@@ -60,6 +103,6 @@
60
103
  </div>
61
104
  </div>
62
105
  <% else %>
63
- <%== partial 'explorer/encryption_enabled' %>
106
+ <%== partial 'explorer/filtered' %>
64
107
  <% end %>
65
108
  </div>
@@ -0,0 +1,23 @@
1
+ <div class="container mb-4">
2
+ <div class="row">
3
+ <div class="col-sm-6 ms-0 ps-0">
4
+ <% if @messages && !@messages.empty? %>
5
+ <a href="<%= explorer_path(@topic_id, 'recent') %>" class="btn btn-secondary btn-sm" title="Display the most recent message for this partition">
6
+ &#x2192; Recent
7
+ </a>
8
+ <% end %>
9
+ </div>
10
+
11
+ <div class="col-sm-6 text-end me-0 pe-0">
12
+ Partitions:
13
+ <span class="badge bg-secondary mt-1 mb-1">
14
+ total: <%= @partitions_count %>
15
+ </span>
16
+
17
+ <span class="badge bg-secondary mt-1 mb-1">
18
+ visible:
19
+ <%= [@active_partitions.first, @active_partitions.last].uniq.join(' to ') %>
20
+ </span>
21
+ </div>
22
+ </div>
23
+ </div>
@@ -0,0 +1,3 @@
1
+ <div class="alert alert-info mt-4" role="alert">
2
+ This topic is empty and does not contain any data in any of the partitions.
3
+ </div>
@@ -0,0 +1,4 @@
1
+ <div class="alert alert-info mt-4 mb-4" role="alert">
2
+ Due to many partitions, only part of the data is visible on each page.
3
+ Please navigate through the pages to view all data.
4
+ </div>
@@ -0,0 +1,51 @@
1
+ <div class="container mb-4">
2
+ <div class="row">
3
+ <div class="col">
4
+ <h3>
5
+ <%= @topic_id %>
6
+ </h3>
7
+ </div>
8
+
9
+ <div class="col">
10
+ <div class="col-auto text-end">
11
+ <label class="col-form-label">Partition</label>
12
+ </div>
13
+ </div>
14
+
15
+ <div class="col pt-1 mb-0 pb-0">
16
+ <div class="col-auto">
17
+ <select class="form-select form-select-sm mb-0 form-control" id="current-partition">
18
+ <%==
19
+ each_partial(
20
+ [nil] + @partitions_count.times.to_a,
21
+ 'explorer/partition_option',
22
+ local: :partition
23
+ )
24
+ %>
25
+ </select>
26
+ </div>
27
+ </div>
28
+ </div>
29
+
30
+ <hr>
31
+ </div>
32
+
33
+ <div class="container">
34
+ <div class="row mb-5">
35
+ <div class="col-sm-12">
36
+ <%== partial('explorer/topic/details') %>
37
+
38
+ <% if @limited %>
39
+ <%== partial('explorer/topic/limited') %>
40
+ <% end %>
41
+
42
+ <% if @messages.empty? && params.current_page == 1 %>
43
+ <%== partial 'explorer/topic/empty' %>
44
+ <% elsif @messages.empty? %>
45
+ <%== partial 'shared/no_paginated_data' %>
46
+ <% else %>
47
+ <%== partial('explorer/partition/messages') %>
48
+ <% end %>
49
+ </div>
50
+ </div>
51
+ </div>
@@ -3,3 +3,19 @@
3
3
  Consumers groups health
4
4
  </a>
5
5
  </li>
6
+
7
+ <% if current_path.include?('/overview') %>
8
+ <li class="breadcrumb-item">
9
+ <a href="<%= root_path('health', 'overview') %>">
10
+ Overview
11
+ </a>
12
+ </li>
13
+ <% end %>
14
+
15
+ <% if current_path.include?('/offsets') %>
16
+ <li class="breadcrumb-item">
17
+ <a href="<%= root_path('health', 'offsets') %>">
18
+ Offsets
19
+ </a>
20
+ </li>
21
+ <% end %>
@@ -0,0 +1,9 @@
1
+ <div class="container mb-4">
2
+ <div class="row">
3
+ <div class="col-lg-12">
4
+ <div class="alert alert-info" role="alert">
5
+ No health data is available. It may mean no processes are running.
6
+ </div>
7
+ </div>
8
+ </div>
9
+ </div>
@@ -1,37 +1,39 @@
1
- <tr class="align-middle status-row-<%= details[:process].status %>">
2
- <td>
3
- <%= topic_name %>
4
- </td>
1
+ <tr class="align-middle <%= lso_risk_state_bg(details) %> status-row-<%= details.process.status %>">
5
2
  <td>
6
3
  <%= partition_id %>
7
4
  </td>
8
5
  <td>
9
- <%== offset_with_label details[:lag_stored].to_i %>
6
+ <%== lag_with_label details.lag_stored %>
10
7
  </td>
11
8
  <td>
12
- <span class="badge <%= lag_trend_bg(details[:lag_stored_d]) %>">
13
- <%= details[:lag_stored_d] %>
9
+ <span class="badge <%= lag_trend_bg(details.lag_stored_d) %>">
10
+ <%= details.lag_stored_d %>
14
11
  </span>
15
12
  </td>
16
13
  <td>
17
- <%= details[:committed_offset] %>
14
+ <%== offset_with_label topic_name, partition_id, details.committed_offset %>
18
15
  </td>
19
16
  <td>
20
- <%== offset_with_label details[:stored_offset].to_i %>
17
+ <%== offset_with_label topic_name, partition_id, details.stored_offset %>
18
+ </td>
19
+ <td>
20
+ <span class="badge <%= kafka_state_bg(details.fetch_state) %> mt-1 mb-1">
21
+ <%= details.fetch_state %>
22
+ </span>
21
23
  </td>
22
24
  <td>
23
- <span class="badge <%= kafka_state_bg(details[:fetch_state]) %> mt-1 mb-1">
24
- <%= details[:fetch_state] %>
25
+ <span class="badge <%= kafka_state_bg(details.poll_state) %> mt-1 mb-1">
26
+ <%= details.poll_state %>
25
27
  </span>
26
28
  </td>
27
29
  <td>
28
- <span class="badge <%= kafka_state_bg(details[:poll_state]) %> mt-1 mb-1">
29
- <%= details[:poll_state] %>
30
+ <span class="badge bg-success <%= lso_risk_state_bg(details) %> bg-opacity-100">
31
+ <%= details.lso_risk_state %>
30
32
  </span>
31
33
  </td>
32
34
  <td>
33
- <a href="<%= root_path('consumers', details[:process].id, 'subscriptions') %>">
34
- <%= details[:process].name %>
35
+ <a href="<%= root_path('consumers', details.process.id, 'subscriptions') %>">
36
+ <%= details.process.name %>
35
37
  </a>
36
38
  </td>
37
39
  </tr>
@@ -0,0 +1,40 @@
1
+ <tr class="align-middle <%= lso_risk_state_bg(details) %> status-row-<%= details.process.status %>">
2
+ <td>
3
+ <%= partition_id %>
4
+ </td>
5
+ <td>
6
+ <%== lag_with_label details.lag_stored %>
7
+ </td>
8
+ <td>
9
+ <%== offset_with_label topic_name, partition_id, details.committed_offset %>
10
+ </td>
11
+ <td>
12
+ <%== relative_time(Time.now - details.committed_offset_fd / 1_000) %>
13
+ </td>
14
+ <td>
15
+ <%== offset_with_label topic_name, partition_id, details.stored_offset %>
16
+ </td>
17
+ <td>
18
+ <%== relative_time(Time.now - details.stored_offset_fd / 1_000) %>
19
+ </td>
20
+ <td>
21
+ <%== offset_with_label topic_name, partition_id, details.lo_offset %>
22
+ </td>
23
+ <td>
24
+ <%== offset_with_label topic_name, partition_id, details.hi_offset %>
25
+ </td>
26
+ <td>
27
+ <%== relative_time(Time.now - details.hi_offset_fd / 1_000) %>
28
+ </td>
29
+ <td>
30
+ <%== offset_with_label topic_name, partition_id, details.ls_offset %>
31
+ </td>
32
+ <td>
33
+ <%== relative_time(Time.now - details.ls_offset_fd / 1_000) %>
34
+ </td>
35
+ <td>
36
+ <span class="badge bg-success <%= lso_risk_state_bg(details) %> bg-opacity-100">
37
+ <%= details.lso_risk_state %>
38
+ </span>
39
+ </td>
40
+ </tr>
@@ -0,0 +1,27 @@
1
+ <div class="container">
2
+ <div class="row mb-5">
3
+ <div class="col-sm-12">
4
+
5
+ <ul class="nav nav-tabs">
6
+ <li class="nav-item">
7
+ <a
8
+ class="nav-link <%= nav_class(include: 'overview') %>"
9
+ href="<%= root_path('health', 'overview') %>"
10
+ >
11
+ Overview
12
+ </a>
13
+ </li>
14
+
15
+ <li class="nav-item">
16
+ <a
17
+ class="nav-link <%= nav_class(include: 'offsets') %>"
18
+ href="<%= root_path('health', 'offsets') %>"
19
+ >
20
+ Offsets
21
+ </a>
22
+ </li>
23
+ </ul>
24
+
25
+ </div>
26
+ </div>
27
+ </div>
@@ -0,0 +1,71 @@
1
+ <%== view_title('Consumers groups offsets details') %>
2
+
3
+ <% if @stats.empty? %>
4
+ <%== partial 'health/no_data' %>
5
+ <% else %>
6
+ <%== partial 'health/tabs' %>
7
+ <% end %>
8
+
9
+ <% @stats.each_with_index do |(cg_name, details), index| %>
10
+ <div class="container mb-5">
11
+ <div class="row mb-4">
12
+ <div class="col-sm-8">
13
+ <h4 class="mb-4"><%= cg_name %></h4>
14
+ </div>
15
+
16
+ <div class="col-sm-4">
17
+ <span class="float-end">
18
+ Last rebalance:
19
+ <span class="badge bg-secondary">
20
+ <%== relative_time(details[:rebalanced_at]) %>
21
+ </span>
22
+ </span>
23
+ </div>
24
+ </div>
25
+
26
+ <div class="row mb-3">
27
+ <div class="col-sm-12">
28
+ <% topics = details[:topics].sort_by(&:first) %>
29
+ <% topics.each_with_index do |(topic_name, partitions), index| %>
30
+ <table class="processes bg-white table table-hover table-bordered table-striped align-middle <%= (index+1 < topics.size) ? 'mb-5' : 'mb-3' %>">
31
+ <thead>
32
+ <tr class="align-middle">
33
+ <th colspan="12">
34
+ <h5 class="mb-0"><%= topic_name %></h4>
35
+ </th>
36
+ </tr>
37
+ <tr class="align-middle">
38
+ <th>Partition</th>
39
+ <th>Lag stored</th>
40
+ <th>Committed offset</th>
41
+ <th>Committed offset change</th>
42
+ <th>Stored offset</th>
43
+ <th>Stored offset change</th>
44
+ <th>Low offset</th>
45
+ <th>High offset</th>
46
+ <th>High offset change</th>
47
+ <th>Last stable offset</th>
48
+ <th>Last stable offset change</th>
49
+ <th>Last stable offset state</th>
50
+ </tr>
51
+ </thead>
52
+ <tbody>
53
+ <% partitions.sort_by(&:first).each do |partition_id, details| %>
54
+ <%==
55
+ partial(
56
+ 'health/partition_offset',
57
+ locals: {
58
+ topic_name: topic_name,
59
+ partition_id: partition_id,
60
+ details: details
61
+ }
62
+ )
63
+ %>
64
+ <% end %>
65
+ </tbody>
66
+ </table>
67
+ <% end %>
68
+ </div>
69
+ </div>
70
+ </div>
71
+ <% end %>
@@ -0,0 +1,68 @@
1
+ <%== view_title('Consumers groups overview') %>
2
+
3
+ <% if @stats.empty? %>
4
+ <%== partial 'health/no_data' %>
5
+ <% else %>
6
+ <%== partial 'health/tabs' %>
7
+ <% end %>
8
+
9
+ <% @stats.each_with_index do |(cg_name, details), index| %>
10
+ <div class="container mb-5">
11
+ <div class="row mb-4">
12
+ <div class="col-sm-8">
13
+ <h4 class="mb-4"><%= cg_name %></h4>
14
+ </div>
15
+
16
+ <div class="col-sm-4">
17
+ <span class="float-end">
18
+ Last rebalance:
19
+ <span class="badge bg-secondary">
20
+ <%== relative_time(details[:rebalanced_at]) %>
21
+ </span>
22
+ </span>
23
+ </div>
24
+ </div>
25
+
26
+ <div class="row mb-3">
27
+ <div class="col-sm-12">
28
+ <% topics = details[:topics].sort_by(&:first) %>
29
+ <% topics.each_with_index do |(topic_name, partitions), index| %>
30
+ <table class="processes bg-white table table-hover table-bordered table-striped align-middle <%= (index+1 < topics.size) ? 'mb-5' : 'mb-3' %>">
31
+ <thead>
32
+ <tr class="align-middle">
33
+ <th colspan="100">
34
+ <h5 class="mb-0"><%= topic_name %></h4>
35
+ </th>
36
+ </tr>
37
+ <tr class="align-middle">
38
+ <th>Partition</th>
39
+ <th>Lag stored</th>
40
+ <th>Lag stored trend</th>
41
+ <th>Committed offset</th>
42
+ <th>Stored offset</th>
43
+ <th>Fetch state</th>
44
+ <th>Poll state</th>
45
+ <th>LSO state</th>
46
+ <th>Process name</th>
47
+ </tr>
48
+ </thead>
49
+ <tbody>
50
+ <% partitions.sort_by(&:first).each do |partition_id, details| %>
51
+ <%==
52
+ partial(
53
+ 'health/partition',
54
+ locals: {
55
+ topic_name: topic_name,
56
+ partition_id: partition_id,
57
+ details: details
58
+ }
59
+ )
60
+ %>
61
+ <% end %>
62
+ </tbody>
63
+ </table>
64
+ <% end %>
65
+ </div>
66
+ </div>
67
+ </div>
68
+ <% end %>
@@ -22,13 +22,16 @@
22
22
  <code>#<%= job.type %></code>
23
23
  </td>
24
24
  <td>
25
- <%== offset_with_label job.first_offset %>
25
+ <%= job.messages %>
26
26
  </td>
27
27
  <td>
28
- <%== offset_with_label job.last_offset %>
28
+ <%== offset_with_label job.topic, job.partition, job.first_offset, explore: true %>
29
29
  </td>
30
30
  <td>
31
- <%== offset_with_label job.committed_offset %>
31
+ <%== offset_with_label job.topic, job.partition, job.last_offset, explore: true %>
32
+ </td>
33
+ <td>
34
+ <%== offset_with_label job.topic, job.partition, job.committed_offset %>
32
35
  </td>
33
36
  <td>
34
37
  <%== relative_time job.started_at %>
@@ -1,7 +1,9 @@
1
1
  <%== view_title('Running jobs', hr: true) %>
2
2
 
3
- <% if @jobs.empty? %>
3
+ <% if @jobs.empty? && params.current_page <= 1 %>
4
4
  <%== partial 'jobs/no_jobs' %>
5
+ <% elsif @jobs.empty? %>
6
+ <%== partial 'shared/no_paginated_data' %>
5
7
  <% else %>
6
8
  <div class="container">
7
9
  <div class="row mb-5">
@@ -13,6 +15,7 @@
13
15
  <th>Topic</th>
14
16
  <th>Consumer</th>
15
17
  <th>Type</th>
18
+ <th>Messages</th>
16
19
  <th>First offset</th>
17
20
  <th>Last offset</th>
18
21
  <th>Committed offset</th>
@@ -0,0 +1,37 @@
1
+ <div class="row mb-4">
2
+ <div class="col-sm-12">
3
+ <h4 class="mb-2">
4
+ <%= consumer_group.id %>
5
+ </h4>
6
+ <hr/>
7
+ </div>
8
+ </div>
9
+
10
+ <div class="row mb-5">
11
+ <div class="col-sm-12">
12
+ <table class="processes bg-white table table-hover table-bordered table-striped mb-0 align-middle">
13
+ <thead>
14
+ <tr class="align-middle">
15
+ <th>Subscription group</th>
16
+ <th>Topic</th>
17
+ <th>Type</th>
18
+ <th>Active</th>
19
+ <th></th>
20
+ </tr>
21
+ </thead>
22
+ <tbody>
23
+ <% consumer_group.topics.each do |topic| %>
24
+ <%==
25
+ partial(
26
+ 'routing/topic',
27
+ locals: {
28
+ subscription_group: topic.subscription_group,
29
+ topic: topic
30
+ }
31
+ )
32
+ %>
33
+ <% end %>
34
+ </tbody>
35
+ </table>
36
+ </div>
37
+ </div>
@@ -0,0 +1,25 @@
1
+ <% if v.is_a?(Hash) || v.respond_to?(:to_h) %>
2
+ <% v.to_h.each do |k2, v2| %>
3
+ <tr>
4
+ <td>
5
+ <%= "#{k}.#{k2}" %>
6
+ </td>
7
+ <td>
8
+ <% if %w[sasl ssl].any? { |scope| k2.to_s.include?(scope) } %>
9
+ ***
10
+ <% else %>
11
+ <%= v2 %>
12
+ <% end %>
13
+ </td>
14
+ </tr>
15
+ <% end %>
16
+ <% else %>
17
+ <tr>
18
+ <td>
19
+ <%= k %>
20
+ </td>
21
+ <td>
22
+ <%= object_value_to_s(v) %>
23
+ </td>
24
+ </tr>
25
+ <% end %>
@@ -0,0 +1,23 @@
1
+ <tr>
2
+ <td>
3
+ <%= subscription_group %>
4
+ </td>
5
+ <td>
6
+ <%= topic.name %>
7
+ </td>
8
+ <td>
9
+ <span class="badge bg-secondary">
10
+ <%= topic.patterns.type %>
11
+ </span>
12
+ </td>
13
+ <td>
14
+ <span class="badge bg-<%= topic.active? ? 'success' : 'warning text-dark' %>">
15
+ <%= topic.active? %>
16
+ </span>
17
+ </td>
18
+ <td class="text-center">
19
+ <a href="<%= root_path('routing', topic.id) %>" class="btn btn-sm btn-secondary text-white">
20
+ Details
21
+ </a>
22
+ </td>
23
+ </tr>
@@ -0,0 +1,10 @@
1
+ <%== view_title('Routing details') %>
2
+
3
+ <div class="container mb-5">
4
+ <%==
5
+ each_partial(
6
+ @routes,
7
+ 'routing/consumer_group'
8
+ )
9
+ %>
10
+ </div>
@@ -0,0 +1,26 @@
1
+ <%==
2
+ title = [@topic.consumer_group.name, @topic.subscription_group, @topic.name].join(': ')
3
+ view_title(title, hr: true)
4
+ %>
5
+
6
+ <div class="container">
7
+ <div class="row mb-5">
8
+ <div class="col-sm-12">
9
+ <table class="processes bg-white table table-hover table-bordered table-striped mb-0 align-middle">
10
+ <tbody>
11
+ <% flat_hash(@topic.to_h).each do |k, v| %>
12
+ <%==
13
+ partial(
14
+ 'routing/detail',
15
+ locals: {
16
+ k: k,
17
+ v: v
18
+ }
19
+ )
20
+ %>
21
+ <% end %>
22
+ </tbody>
23
+ </table>
24
+ </div>
25
+ </div>
26
+ </div>