ahoy_captain 0.8 → 0.10.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (167) hide show
  1. checksums.yaml +4 -4
  2. data/README.md +25 -13
  3. data/app/assets/javascript/ahoy_captain/application.js +4 -4
  4. data/app/assets/javascript/ahoy_captain/controllers/active_links_controller.js +30 -0
  5. data/app/assets/javascript/ahoy_captain/controllers/application.js +5 -5
  6. data/app/assets/javascript/ahoy_captain/controllers/application_controller.js +16 -9
  7. data/app/assets/javascript/ahoy_captain/controllers/combobox_controller.js +341 -0
  8. data/app/assets/javascript/ahoy_captain/controllers/details_modal_controller.js +5 -5
  9. data/app/assets/javascript/ahoy_captain/controllers/dropdown_label_controller.js +2 -2
  10. data/app/assets/javascript/ahoy_captain/controllers/filter/item_controller.js +12 -0
  11. data/app/assets/javascript/ahoy_captain/controllers/filter_form_controller.js +13 -0
  12. data/app/assets/javascript/ahoy_captain/controllers/filter_modal_controller.js +45 -0
  13. data/app/assets/javascript/ahoy_captain/controllers/funnel_chart_controller.js +116 -104
  14. data/app/assets/javascript/ahoy_captain/controllers/index.js +4 -3
  15. data/app/assets/javascript/ahoy_captain/controllers/interval_controller.js +8 -3
  16. data/app/assets/javascript/ahoy_captain/controllers/line_chart_controller.js +188 -0
  17. data/app/assets/javascript/ahoy_captain/controllers/predicate_select_controller.js +9 -0
  18. data/app/assets/javascript/ahoy_captain/controllers/properties_controller.js +8 -0
  19. data/app/assets/javascript/ahoy_captain/controllers/property_filter_controller.js +46 -0
  20. data/app/assets/javascript/ahoy_captain/controllers/realtime_controller.js +12 -9
  21. data/app/assets/javascript/ahoy_captain/controllers/tile_controller.js +9 -0
  22. data/app/assets/javascript/ahoy_captain/controllers/toggle_controller.js +17 -0
  23. data/app/assets/javascript/ahoy_captain/helpers/chart_utils.js +156 -0
  24. data/app/assets/javascript/ahoy_captain/helpers/number_formatters.js +55 -0
  25. data/app/components/ahoy_captain/combobox_component.html.erb +33 -0
  26. data/app/components/ahoy_captain/combobox_component.rb +13 -0
  27. data/app/components/ahoy_captain/comparison_link_component.rb +40 -0
  28. data/app/components/ahoy_captain/dropdown_button_component.html.erb +5 -5
  29. data/app/components/ahoy_captain/dropdown_link_component.html.erb +5 -5
  30. data/app/components/ahoy_captain/filter/dropdown_component.html.erb +50 -0
  31. data/app/components/ahoy_captain/filter/dropdown_component.rb +51 -0
  32. data/app/components/ahoy_captain/filter/modal_component.html.erb +12 -9
  33. data/app/components/ahoy_captain/filter/select_component.html.erb +23 -19
  34. data/app/components/ahoy_captain/filter/select_component.rb +41 -9
  35. data/app/components/ahoy_captain/filter/tag_component.html.erb +8 -4
  36. data/app/components/ahoy_captain/filter/tag_component.rb +6 -30
  37. data/app/components/ahoy_captain/filter/tag_container_component.html.erb +2 -3
  38. data/app/components/ahoy_captain/filter/tag_container_component.rb +1 -8
  39. data/app/components/ahoy_captain/stats/comparable_container_component.html.erb +25 -0
  40. data/app/components/ahoy_captain/stats/comparable_container_component.rb +86 -0
  41. data/app/components/ahoy_captain/stats/container_component.html.erb +15 -0
  42. data/app/components/ahoy_captain/stats/container_component.rb +26 -0
  43. data/app/components/ahoy_captain/sticky_nav_component.html.erb +13 -22
  44. data/app/components/ahoy_captain/sticky_nav_component.rb +11 -0
  45. data/app/components/ahoy_captain/table_component.html.erb +4 -37
  46. data/app/components/ahoy_captain/table_component.rb +25 -5
  47. data/app/components/ahoy_captain/tables/devices_table_component.rb +11 -0
  48. data/app/components/ahoy_captain/tables/dynamic_table.rb +13 -0
  49. data/app/components/ahoy_captain/tables/dynamic_table_component.rb +204 -0
  50. data/app/components/ahoy_captain/tables/goals_table_component.rb +17 -0
  51. data/app/components/ahoy_captain/tables/header_component.html.erb +5 -0
  52. data/app/components/ahoy_captain/tables/header_component.rb +18 -0
  53. data/app/components/ahoy_captain/tables/headers/header_component.html.erb +5 -0
  54. data/app/components/ahoy_captain/tables/headers/header_component.rb +16 -0
  55. data/app/components/ahoy_captain/tables/properties_table_component.rb +27 -0
  56. data/app/components/ahoy_captain/tables/row_component.html.erb +4 -0
  57. data/app/components/ahoy_captain/tables/rows/row_component.html.erb +6 -0
  58. data/app/components/ahoy_captain/tables/rows/row_component.rb +40 -0
  59. data/app/components/ahoy_captain/tile_component.html.erb +21 -10
  60. data/app/components/ahoy_captain/tile_component.rb +3 -2
  61. data/app/components/ahoy_captain/tooltip_component.html.erb +2 -2
  62. data/app/controllers/ahoy_captain/application_controller.rb +19 -29
  63. data/app/controllers/ahoy_captain/campaigns_controller.rb +2 -10
  64. data/app/controllers/ahoy_captain/cities_controller.rb +2 -6
  65. data/app/controllers/ahoy_captain/countries_controller.rb +2 -6
  66. data/app/controllers/ahoy_captain/devices_controller.rb +3 -6
  67. data/app/controllers/ahoy_captain/entry_pages_controller.rb +2 -4
  68. data/app/controllers/ahoy_captain/exit_pages_controller.rb +3 -4
  69. data/app/controllers/ahoy_captain/exports_controller.rb +14 -0
  70. data/app/controllers/ahoy_captain/filters/base_controller.rb +1 -3
  71. data/app/controllers/ahoy_captain/filters/goals_controller.rb +9 -0
  72. data/app/controllers/ahoy_captain/filters/pages/actions_controller.rb +1 -1
  73. data/app/controllers/ahoy_captain/filters/pages/entry_pages_controller.rb +3 -3
  74. data/app/controllers/ahoy_captain/filters/pages/exit_pages_controller.rb +2 -3
  75. data/app/controllers/ahoy_captain/filters/properties/names_controller.rb +11 -0
  76. data/app/controllers/ahoy_captain/filters/properties/values_controller.rb +15 -0
  77. data/app/controllers/ahoy_captain/filters/sources_controller.rb +1 -1
  78. data/app/controllers/ahoy_captain/filters/utms_controller.rb +1 -1
  79. data/app/controllers/ahoy_captain/properties_controller.rb +41 -0
  80. data/app/controllers/ahoy_captain/regions_controller.rb +3 -7
  81. data/app/controllers/ahoy_captain/sources_controller.rb +2 -5
  82. data/app/controllers/ahoy_captain/stats/base_controller.rb +86 -5
  83. data/app/controllers/ahoy_captain/stats/bounce_rates_controller.rb +2 -1
  84. data/app/controllers/ahoy_captain/stats/total_pageviews_controller.rb +2 -1
  85. data/app/controllers/ahoy_captain/stats/total_visits_controller.rb +2 -1
  86. data/app/controllers/ahoy_captain/stats/unique_visitors_controller.rb +3 -1
  87. data/app/controllers/ahoy_captain/stats/views_per_visits_controller.rb +3 -11
  88. data/app/controllers/ahoy_captain/stats/visit_durations_controller.rb +2 -1
  89. data/app/controllers/ahoy_captain/top_pages_controller.rb +2 -8
  90. data/app/decorators/ahoy_captain/application_decorator.rb +27 -3
  91. data/app/decorators/ahoy_captain/campaign_decorator.rb +8 -0
  92. data/app/decorators/ahoy_captain/city_decorator.rb +12 -0
  93. data/app/decorators/ahoy_captain/country_decorator.rb +10 -0
  94. data/app/decorators/ahoy_captain/device_decorator.rb +13 -2
  95. data/app/decorators/ahoy_captain/page_decorator.rb +11 -0
  96. data/app/decorators/ahoy_captain/region_decorator.rb +16 -0
  97. data/app/decorators/ahoy_captain/source_decorator.rb +7 -0
  98. data/app/helpers/ahoy_captain/application_helper.rb +62 -3
  99. data/app/models/ahoy_captain/comparison_mode.rb +72 -0
  100. data/app/models/ahoy_captain/export.rb +48 -0
  101. data/app/models/ahoy_captain/filter_parser.rb +82 -0
  102. data/app/models/ahoy_captain/range_from_params.rb +75 -0
  103. data/app/models/ahoy_captain/rangeable.rb +0 -3
  104. data/app/models/concerns/ahoy_captain/compare_mode.rb +19 -0
  105. data/app/models/concerns/ahoy_captain/limitable.rb +17 -0
  106. data/app/models/concerns/ahoy_captain/range_options.rb +1 -14
  107. data/app/presenters/ahoy_captain/dashboard_presenter.rb +18 -49
  108. data/app/presenters/ahoy_captain/goals_presenter.rb +3 -2
  109. data/app/queries/ahoy_captain/application_query.rb +78 -13
  110. data/app/queries/ahoy_captain/campaign_query.rb +14 -0
  111. data/app/queries/ahoy_captain/city_query.rb +11 -0
  112. data/app/queries/ahoy_captain/country_query.rb +10 -0
  113. data/app/queries/ahoy_captain/device_query.rb +10 -0
  114. data/app/queries/ahoy_captain/entry_pages_query.rb +3 -2
  115. data/app/queries/ahoy_captain/event_query.rb +20 -13
  116. data/app/queries/ahoy_captain/exit_pages_query.rb +6 -4
  117. data/app/queries/ahoy_captain/region_query.rb +11 -0
  118. data/app/queries/ahoy_captain/source_query.rb +10 -0
  119. data/app/queries/ahoy_captain/stats/average_views_per_visit_query.rb +11 -4
  120. data/app/queries/ahoy_captain/stats/average_visit_duration_query.rb +14 -2
  121. data/app/queries/ahoy_captain/stats/base_query.rb +18 -0
  122. data/app/queries/ahoy_captain/stats/bounce_rates_query.rb +15 -1
  123. data/app/queries/ahoy_captain/stats/total_pageviews_query.rb +2 -2
  124. data/app/queries/ahoy_captain/stats/total_visitors_query.rb +1 -1
  125. data/app/queries/ahoy_captain/stats/unique_visitors_query.rb +1 -1
  126. data/app/queries/ahoy_captain/stats/views_per_visit_query.rb +1 -1
  127. data/app/queries/ahoy_captain/stats/visit_duration_query.rb +3 -3
  128. data/app/queries/ahoy_captain/top_page_query.rb +13 -0
  129. data/app/queries/ahoy_captain/visit_query.rb +2 -3
  130. data/app/queries/concerns/ahoy_captain/comparable_queries.rb +25 -0
  131. data/app/queries/concerns/ahoy_captain/comparable_query.rb +138 -0
  132. data/app/queries/concerns/ahoy_captain/lazy_comparable_query.rb +42 -0
  133. data/app/views/ahoy_captain/devices/_table.html.erb +2 -0
  134. data/app/views/ahoy_captain/devices/index.html+details.erb +1 -1
  135. data/app/views/ahoy_captain/devices/index.html.erb +2 -2
  136. data/app/views/ahoy_captain/funnels/show.html.erb +5 -2
  137. data/app/views/ahoy_captain/goals/index.html.erb +2 -37
  138. data/app/views/ahoy_captain/layouts/application.html.erb +3 -4
  139. data/app/views/ahoy_captain/properties/_form.html.erb +6 -0
  140. data/app/views/ahoy_captain/properties/index.html.erb +3 -0
  141. data/app/views/ahoy_captain/properties/show.html.erb +6 -0
  142. data/app/views/ahoy_captain/realtimes/show.html.erb +1 -1
  143. data/app/views/ahoy_captain/roots/_filters.html.erb +80 -0
  144. data/app/views/ahoy_captain/roots/show.html.erb +76 -109
  145. data/app/views/ahoy_captain/stats/base/index.html.erb +34 -9
  146. data/app/views/ahoy_captain/stats/show.html.erb +8 -55
  147. data/config/routes.rb +9 -0
  148. data/lib/ahoy_captain/ahoy/event_methods.rb +35 -74
  149. data/lib/ahoy_captain/ahoy/visit_methods.rb +1 -1
  150. data/lib/ahoy_captain/configuration.rb +18 -7
  151. data/lib/ahoy_captain/engine.rb +22 -0
  152. data/lib/ahoy_captain/filter_configuration/filter.rb +16 -0
  153. data/lib/ahoy_captain/filter_configuration/filter_collection.rb +48 -0
  154. data/lib/ahoy_captain/filters_configuration.rb +77 -0
  155. data/lib/ahoy_captain/goals.rb +10 -2
  156. data/lib/ahoy_captain/period_collection.rb +1 -1
  157. data/lib/ahoy_captain/predicate_label.rb +7 -0
  158. data/lib/ahoy_captain/version.rb +1 -1
  159. data/lib/ahoy_captain.rb +7 -1
  160. data/lib/generators/ahoy_captain/templates/config.rb.tt +32 -0
  161. metadata +80 -21
  162. data/app/assets/javascript/ahoy_captain/controllers/filter_controller.js +0 -145
  163. data/app/assets/javascript/ahoy_captain/controllers/filter_tag_controller.js +0 -17
  164. data/app/assets/javascript/ahoy_captain/controllers/link_controller.js +0 -43
  165. data/app/assets/javascript/ahoy_captain/controllers/navigation_controller.js +0 -25
  166. data/app/models/ahoy_captain/current.rb +0 -9
  167. data/app/models/ahoy_captain/url_helpers.rb +0 -6
@@ -0,0 +1,40 @@
1
+ module AhoyCaptain
2
+ module Tables
3
+ module Rows
4
+ class RowComponent < ViewComponent::Base
5
+ def initialize(table:, item:)
6
+ @table = table
7
+ @item = item
8
+ end
9
+
10
+ def progress_bar(value, max, label)
11
+ items = []
12
+ items << view_context.content_tag(:progress, "", class: "progress-primary bg-base-100 h-8 grow", value: value, max: max)
13
+ items << view_context.content_tag(:span, class: "grow text-elipsis overflow-hidden absolute left-4 bottom-3 h-8 text-primary-content") do
14
+ label
15
+ end
16
+
17
+ items.join.html_safe
18
+ end
19
+
20
+ def item(value = nil, &block)
21
+ view_context.content_tag(:span, class: "w-8 ml-8 text-right ") do
22
+ if value
23
+ value
24
+ else
25
+ capture(&block)
26
+ end
27
+ end
28
+ end
29
+
30
+ def percent_total(item)
31
+ '%.1f' % ((item.unit_amount.to_i * 1.0 / total)*100.0)
32
+ end
33
+
34
+ def tooltip(value)
35
+ AhoyCaptain::TooltipComponent.new(amount: value).render_in(self)
36
+ end
37
+ end
38
+ end
39
+ end
40
+ end
@@ -1,12 +1,23 @@
1
- <div class='<%= 'lg:col-span-2' if wide %> col-span-1 p-4 bg-base-200 rounded-md p-8 mx-4 lg:mx-0'>
2
- <div class="flex justify-between">
3
- <h5 class='text-lg text-accent-content'><%= title %></h5>
4
- <div class="flex self-center gap-3">
5
- <%= display_links %>
1
+ <div
2
+ data-controller="tile"
3
+ class="card card-compact <%= 'lg:col-span-2' if wide %> col-span-1 shadow-xl rounded-md lg:mx-0 bg-base-200 <%= @classes %>">
4
+ <% if title.present? || display_links.present? %>
5
+ <div class="flex justify-between">
6
+ <% if title.present? %>
7
+ <h2 class="card-title" data-tile-target="title"><%= title %></h2>
8
+ <% end %>
9
+ <% if display_links.present? %>
10
+ <div class="flex self-center lg:gap-3">
11
+ <%= display_links %>
12
+ </div>
13
+ <% end %>
6
14
  </div>
7
- </div>
15
+ <% end %>
16
+
8
17
  <%= statistic_display %>
9
- <div class="flex justify-center">
10
- <%= details_cta %>
11
- </div>
12
- </div>
18
+ <% if details_cta.present? %>
19
+ <div class="flex justify-center">
20
+ <%= details_cta %>
21
+ </div>
22
+ <% end %>
23
+ </div>
@@ -5,7 +5,8 @@ class AhoyCaptain::TileComponent < ViewComponent::Base
5
5
  renders_one :display_links
6
6
  renders_one :details_cta
7
7
 
8
- def initialize(title: nil, wide: false)
8
+ def initialize(title: nil, wide: false, classes: "p-8 mx-4")
9
+ @classes = classes
9
10
  @title = title
10
11
  @wide = wide
11
12
  end
@@ -13,4 +14,4 @@ class AhoyCaptain::TileComponent < ViewComponent::Base
13
14
  private
14
15
 
15
16
  attr_reader :title, :wide
16
- end
17
+ end
@@ -1,3 +1,3 @@
1
- <div class="tooltip" data-tip=<%= amount %>>
1
+ <div class="tooltip " data-tip=<%= amount %>>
2
2
  <p><%= abbreviate %></p>
3
- </div>
3
+ </div>
@@ -1,40 +1,35 @@
1
-
2
-
3
1
  module AhoyCaptain
4
- module Limitable
5
- private
6
-
7
- def limit
8
- if request.variant.include?(:details)
9
- nil
10
- else
11
- if params[:limit]
12
- params[:limit].to_i
13
- else
14
- 10
15
- end
16
- end
17
- end
18
- end
19
-
20
2
  class ApplicationController < ActionController::Base
21
3
  include Pagy::Backend
4
+ include CompareMode
5
+ include RangeOptions
6
+ include Rangeable
22
7
 
23
8
  layout 'ahoy_captain/layouts/application'
24
9
 
25
- before_action do
26
- Current.request = self
10
+ def period
11
+ params[:period] || AhoyCaptain.config.ranges.default
27
12
  end
28
13
 
29
14
  # show the details frame
30
- before_action do
15
+ before_action :use_details_frame
16
+
17
+ # act like an spa without being an spa
18
+ before_action :act_like_an_spa
19
+
20
+ rescue_from Widget::WidgetDisabled do |e|
21
+ render template: 'ahoy_captain/shared/widget_disabled', locals: { frame: e.frame }
22
+ end
23
+
24
+ private
25
+
26
+ def use_details_frame
31
27
  if request.headers['Turbo-Frame'] == 'details'
32
28
  request.variant = :details
33
29
  end
34
30
  end
35
31
 
36
- # act like an spa without being an spa
37
- before_action do
32
+ def act_like_an_spa
38
33
  if request.format.html? && request.headers['Turbo-Frame'].blank?
39
34
  if request.path != root_path
40
35
  requested_params = Rails.application.routes.recognize_path(request.path).except(:controller, :action)
@@ -46,12 +41,6 @@ module AhoyCaptain
46
41
  end
47
42
  end
48
43
 
49
- rescue_from Widget::WidgetDisabled do |e|
50
- render template: 'ahoy_captain/shared/widget_disabled', locals: { frame: e.frame }
51
- end
52
-
53
- private
54
-
55
44
  def visit_query
56
45
  VisitQuery.call(params)
57
46
  end
@@ -60,6 +49,7 @@ module AhoyCaptain
60
49
  EventQuery.call(params)
61
50
  end
62
51
 
52
+ # Only paginate details requests requests
63
53
  def paginate(collection)
64
54
  if paginate?
65
55
  pagy, results = pagy(collection, page: params[:page])
@@ -10,17 +10,9 @@ module AhoyCaptain
10
10
 
11
11
  def index
12
12
  results = cached(:campaigns, params[:campaigns_type]) do
13
- visit_query
14
- .select(
15
- "COALESCE(#{params[:campaigns_type]}, 'Direct/None') as label",
16
- "count(COALESCE(#{params[:campaigns_type]}, 'Direct/None')) as count",
17
- "sum(count(COALESCE(#{params[:campaigns_type]}, 'Direct/None'))) OVER() as total_count"
18
- )
19
- .group("COALESCE(#{params[:campaigns_type]}, 'Direct/None')")
20
- .order(Arel.sql("count(COALESCE(#{params[:campaigns_type]}, 'Direct/None')) desc"))
21
- .limit(limit)
13
+ CampaignQuery.call(params).limit(limit)
22
14
  end
23
- @campaigns = paginate(results).map { |campaign| CampaignDecorator.new(campaign) }
15
+ @campaigns = paginate(results).map { |campaign| CampaignDecorator.new(campaign, self) }
24
16
  @campaign_type = params[:campaigns_type]&.titleize&.gsub("Utm", "UTM")
25
17
  end
26
18
  end
@@ -10,15 +10,11 @@ module AhoyCaptain
10
10
 
11
11
  def index
12
12
  results = cached(:cities) do
13
- visit_query
14
- .select("city, country, count(concat(city, region, country)) as count, sum(count(concat(city, region, country))) over() as total_count")
15
- .where.not(city: nil)
16
- .group("city, region, country")
17
- .order(Arel.sql "count(concat(city, region, country)) desc")
13
+ CityQuery.call(params)
18
14
  .limit(limit)
19
15
  end
20
16
 
21
- @cities = paginate(results).map { |city| CityDecorator.new(city) }
17
+ @cities = paginate(results).map { |city| CityDecorator.new(city, self) }
22
18
  end
23
19
  end
24
20
  end
@@ -1,7 +1,6 @@
1
1
  module AhoyCaptain
2
2
  class CountriesController < ApplicationController
3
3
  include Limitable
4
- include Rangeable
5
4
 
6
5
  before_action do
7
6
  if Widget.disabled?(:locations, :countries)
@@ -11,14 +10,11 @@ module AhoyCaptain
11
10
 
12
11
  def index
13
12
  results = cached(:countries) do
14
- visit_query
15
- .reselect("country as label, count(country) as count, sum(count(country)) OVER() as total_count")
16
- .group("country")
17
- .order("count(country) desc")
13
+ CountryQuery.call(params)
18
14
  .limit(limit)
19
15
  end
20
16
 
21
- @countries = paginate(results).map { |country| CountryDecorator.new(country) }
17
+ @countries = paginate(results).map { |country| CountryDecorator.new(country, self) }
22
18
  end
23
19
  end
24
20
  end
@@ -10,14 +10,11 @@ module AhoyCaptain
10
10
 
11
11
  def index
12
12
  results = cached(:devices, params[:devices_type]) do
13
- visit_query
14
- .select("#{params[:devices_type]} as label", "count(#{params[:devices_type]}) as count", "sum(count(#{params[:devices_type]})) over() as total_count")
15
- .group(params[:devices_type])
16
- .order("count(#{params[:devices_type]}) desc")
17
- .limit(limit)
13
+ DeviceQuery.call(params)
14
+ .limit(limit)
18
15
  end
19
16
 
20
- @devices = results.map { |device| DeviceDecorator.new(device) }
17
+ @devices = results.map { |device| DeviceDecorator.new(device, self) }
21
18
  end
22
19
  end
23
20
  end
@@ -10,12 +10,10 @@ module AhoyCaptain
10
10
 
11
11
  def index
12
12
  results = cached(:entry_pages) do
13
- EntryPagesQuery.call(params, event_query)
14
- .order(Arel.sql "count(#{AhoyCaptain.config.event[:url_column]}) desc")
15
- .limit(limit)
13
+ EntryPagesQuery.call(params).limit(limit)
16
14
  end
17
15
 
18
- @pages = paginate(results).map { |page| EntryPageDecorator.new(page) }
16
+ @pages = paginate(results).map { |page| EntryPageDecorator.new(page, self) }
19
17
  end
20
18
  end
21
19
  end
@@ -10,11 +10,10 @@ module AhoyCaptain
10
10
 
11
11
  def index
12
12
  results = cached(:exit_pages) do
13
- ExitPagesQuery.call(params, event_query)
14
- .order(Arel.sql "count(#{AhoyCaptain.config.event[:url_column]}) desc")
15
- .limit(limit)
13
+ ExitPagesQuery.call(params)
14
+ .limit(limit)
16
15
  end
17
- @pages = paginate(results).map { |page| ExitPageDecorator.new(page) }
16
+ @pages = paginate(results).map { |page| ExitPageDecorator.new(page, self) }
18
17
  end
19
18
  end
20
19
  end
@@ -0,0 +1,14 @@
1
+ module AhoyCaptain
2
+ class ExportsController < ApplicationController
3
+ skip_before_action :act_like_an_spa
4
+
5
+ def show
6
+ export = Export.new(params, self).build
7
+ file = export.to_zip
8
+ send_data file.read,
9
+ type: 'application/zip',
10
+ disposition: 'attachment',
11
+ filename: "AhoyCaptain export #{request.host} #{range[0].to_date} to #{(range[1] || Time.current).to_date}.zip"
12
+ end
13
+ end
14
+ end
@@ -1,12 +1,10 @@
1
1
  module AhoyCaptain
2
2
  module Filters
3
3
  class BaseController < ApplicationController
4
- include Rangeable
5
-
6
4
  private
7
5
 
8
6
  def serialize(value)
9
- { text: value }
7
+ { text: (value.presence || AhoyCaptain.none.text), value: (value.presence || AhoyCaptain.none.value) }
10
8
  end
11
9
 
12
10
  def visit_query
@@ -0,0 +1,9 @@
1
+ module AhoyCaptain
2
+ module Filters
3
+ class GoalsController < BaseController
4
+ def index
5
+ render json: AhoyCaptain.configuration.goals.map { |goal| { text: goal.title, value: goal.id } }
6
+ end
7
+ end
8
+ end
9
+ end
@@ -5,7 +5,7 @@ module AhoyCaptain
5
5
  def index
6
6
  query = event_query.all.with_url.distinct_url
7
7
 
8
- render json: query.map { |row| { text: row.url } }
8
+ render json: query.map { |row| serialize(row.url) }
9
9
  end
10
10
  end
11
11
  end
@@ -1,11 +1,11 @@
1
1
  module AhoyCaptain
2
2
  module Filters
3
3
  module Pages
4
- # TODO: ACCOMODATE EXIT_PAGES
5
4
  class EntryPagesController < BaseController
6
5
  def index
7
- query = event_query.all.with_url.distinct_url
8
- render json: query.map { |row| { text: row.url } }
6
+ query = event_query.all.distinct("entry_pages.url").select("entry_pages.url as url")
7
+
8
+ render json: query.map { |row| serialize(row.url) }
9
9
  end
10
10
 
11
11
  end
@@ -1,12 +1,11 @@
1
1
  module AhoyCaptain
2
2
  module Filters
3
3
  module Pages
4
- # TODO: ACCOMODATE ENTRY_PAGES
5
4
  class ExitPagesController < BaseController
6
5
  def index
7
- query = event_query.with_url.distinct_url
6
+ query = event_query.distinct("exit_pages.url").select("exit_pages.url as url")
8
7
 
9
- render json: query.map { |row| { text: row.url } }
8
+ render json: query.map { |row| serialize(row.url) }
10
9
  end
11
10
 
12
11
  end
@@ -0,0 +1,11 @@
1
+ module AhoyCaptain
2
+ module Filters
3
+ module Properties
4
+ class NamesController < BaseController
5
+ def index
6
+ render json: ::Ahoy::Event.select("jsonb_object_keys(properties) as keys").distinct("jsonb_object_keys(properties)").map(&:keys).map { |key| serialize(key) }
7
+ end
8
+ end
9
+ end
10
+ end
11
+ end
@@ -0,0 +1,15 @@
1
+ module AhoyCaptain
2
+ module Filters
3
+ module Properties
4
+ class ValuesController < BaseController
5
+ def index
6
+ param_key = params[:q].to_unsafe_h.detect { |k,v| k.ends_with?("_i_cont") && k.starts_with?("properties.") }[0]
7
+ key = param_key.delete_prefix("properties.").delete_suffix("_i_cont")
8
+ query = event_query.all.distinct.select("properties->>'#{key}'").pluck(Arel.sql "properties->>'#{key}'")
9
+
10
+ render json: query.map { |element| serialize(element) }
11
+ end
12
+ end
13
+ end
14
+ end
15
+ end
@@ -4,7 +4,7 @@ module AhoyCaptain
4
4
  def index
5
5
  query = visit_query.all
6
6
 
7
- render json: query.result.select("distinct referring_domain").where.not(referring_domain: nil).group(:referring_domain).order(Arel.sql "count(*) desc").pluck(:referring_domain).map { |city| serialize(city) }
7
+ render json: query.select("distinct referring_domain").where.not(referring_domain: nil).group(:referring_domain).order(Arel.sql "count(*) desc").pluck(:referring_domain).map { |city| serialize(city) }
8
8
  end
9
9
  end
10
10
  end
@@ -2,7 +2,7 @@ module AhoyCaptain
2
2
  module Filters
3
3
  class UtmsController < BaseController
4
4
  def index
5
- query = visit_query.select("#{params[:type]}", "count(#{params[:type]}) as total").group(params[:type]).order(Arel.sql "count(#{params[:type]}) desc").pluck(params[:type]).map { |city| serialize(city || "Direct/none") }
5
+ query = visit_query.select("#{params[:type]}", "count(#{params[:type]}) as total").group(params[:type]).order(Arel.sql "count(#{params[:type]}) desc").pluck(params[:type]).map { |city| serialize(city) }
6
6
  render json: query
7
7
  end
8
8
  end
@@ -0,0 +1,41 @@
1
+ module AhoyCaptain
2
+ class PropertiesController < ApplicationController
3
+ before_action do
4
+ @options = ::Ahoy::Event.select("jsonb_object_keys(properties) as keys").distinct("jsonb_object_keys(properties)").map(&:keys).map { |key| [Base64.urlsafe_encode64(key), key]}.to_h
5
+ end
6
+
7
+ def index
8
+ end
9
+
10
+ def show
11
+ value = Base64.urlsafe_decode64(params[:id])
12
+
13
+ @properties = event_query
14
+ .select(
15
+ "COALESCE(properties->>'#{value}', '(none)') AS label",
16
+ "COUNT(*) AS events_count",
17
+ "COUNT(DISTINCT visit_id) AS unique_visitors_count",
18
+ "(COUNT(DISTINCT visit_id)/COUNT(*)::numeric) * 100 as percentage"
19
+ )
20
+ .group("COALESCE(properties->>'#{value}', '(none)')")
21
+ .order(Arel.sql "COUNT(*) desc")
22
+ end
23
+
24
+ private
25
+
26
+ helper_method :has_property?
27
+ def has_property?(value)
28
+ searching_properties[value]
29
+ end
30
+
31
+ helper_method :selected_property?
32
+ def selected_property?(value)
33
+ encoded = Base64.urlsafe_encode64(value, padding: false)
34
+ encoded == params[:id]
35
+ end
36
+
37
+ def searching_properties
38
+ JSON.parse(params.dig("q", "properties_json_cont") || '{}')
39
+ end
40
+ end
41
+ end
@@ -10,15 +10,11 @@ module AhoyCaptain
10
10
 
11
11
  def index
12
12
  results = cached(:regions) do
13
- visit_query
14
- .reselect("region, country, count(concat(region, country)) as count, sum(count(region)) over() as total_count")
15
- .where.not(region: nil)
16
- .group("region, country")
17
- .order(Arel.sql "count(concat(region, country)) desc")
18
- .limit(limit)
13
+ RegionQuery.call(params)
14
+ .limit(limit)
19
15
  end
20
16
 
21
- @regions = paginate(results).map { |region| RegionDecorator.new(region) }
17
+ @regions = paginate(results).map { |region| RegionDecorator.new(region, self) }
22
18
  end
23
19
  end
24
20
  end
@@ -10,14 +10,11 @@ module AhoyCaptain
10
10
 
11
11
  def index
12
12
  results = cached(:sources) do
13
- visit_query.select("substring(referring_domain from '(?:.*://)?(?:www\.)?([^/?]*)') as referring_domain, count(substring(referring_domain from '(?:.*://)?(?:www\.)?([^/?]*)')) as count, sum(count(substring(referring_domain from '(?:.*://)?(?:www\.)?([^/?]*)'))) OVER() as total_count")
14
- .where.not(referring_domain: nil)
15
- .group("substring(referring_domain from '(?:.*://)?(?:www\.)?([^/?]*)')")
16
- .order(Arel.sql "count(substring(referring_domain from '(?:.*://)?(?:www\.)?([^/?]*)')) desc")
13
+ SourceQuery.call(params)
17
14
  .limit(limit)
18
15
  end
19
16
 
20
- @sources = paginate(results).map { |source| AhoyCaptain::SourceDecorator.new(source) }
17
+ @sources = paginate(results).map { |source| AhoyCaptain::SourceDecorator.new(source, self) }
21
18
  end
22
19
  end
23
20
 
@@ -1,7 +1,6 @@
1
1
  module AhoyCaptain
2
2
  module Stats
3
3
  class BaseController < ApplicationController
4
- include Rangeable
5
4
 
6
5
  INTERVAL_PERIOD = {
7
6
  "realtime" => ["minute"],
@@ -12,8 +11,18 @@ module AhoyCaptain
12
11
  }
13
12
 
14
13
  INTERVALS = ["minute", "hour", "day", "week", "month"]
14
+
15
15
  private
16
16
 
17
+ helper_method :metric_type
18
+ def metric_type(stats)
19
+ if compare_mode?
20
+ stats.current.values.first.try(:class) || stats.compared_to.values.first.try(:class)
21
+ else
22
+ stats.values.first.class
23
+ end
24
+ end
25
+
17
26
  helper_method :selected_interval
18
27
  def selected_interval
19
28
  if params[:interval].in?(INTERVALS)
@@ -32,11 +41,13 @@ module AhoyCaptain
32
41
  # assume we're in a realtime
33
42
  return INTERVAL_PERIOD["realtime"][0]
34
43
  end
35
- diff = (range[1] - range[0]).seconds
36
- if diff.in_months > 1
44
+ diff = (range[1] - range[0]).seconds.in_days
45
+ if diff >= 31
37
46
  "month"
38
- elsif diff.in_days > 0
47
+ elsif diff > 1
39
48
  "day"
49
+ elsif diff == 1
50
+ "hour"
40
51
  else
41
52
  "hour"
42
53
  end
@@ -49,7 +60,7 @@ module AhoyCaptain
49
60
 
50
61
  diff = (range[1] - range[0]).seconds.in_days
51
62
 
52
- if diff == 0
63
+ if diff < 1
53
64
  INTERVAL_PERIOD["day"]
54
65
  elsif diff <= 7
55
66
  INTERVAL_PERIOD["7d"]
@@ -62,6 +73,76 @@ module AhoyCaptain
62
73
  INTERVAL_PERIOD["month"]
63
74
  end
64
75
  end
76
+
77
+ def lazy_window(result, value = 0, base = nil)
78
+ if result.is_a?(AhoyCaptain::LazyComparableQuery::LazyComparison)
79
+ result.result.current = lazy_window(result.result.current, value, range)
80
+ result.result.compared_to = lazy_window(result.result.compared_to, value, result.compare_range)
81
+ return result.result
82
+ end
83
+
84
+ base ||= range
85
+ window = window_for(selected_interval, result.keys[0].class, base.numeric)
86
+
87
+ window.each do |item|
88
+ if result.key?(item)
89
+ next
90
+ end
91
+
92
+ result[item] ||= value
93
+ end
94
+
95
+ transform = interval_label_transformation(selected_interval)
96
+
97
+ if transform
98
+ result.transform_keys! { |key| key.strftime(transform) }
99
+ end
100
+
101
+ result
102
+ end
103
+
104
+ def interval_label_transformation(interval)
105
+ return nil
106
+ if interval == 'hour'
107
+ return '%H:%M %p'
108
+ end
109
+
110
+ nil
111
+ end
112
+
113
+ # base should be a range
114
+ def window_for(interval, type, base = nil)
115
+ function = case type.to_s
116
+ when 'Date', 'NilClass'
117
+ ->(value) {
118
+ date = Time.at(value).utc
119
+ if interval == 'month'
120
+ date.change(day: 1)
121
+ elsif interval == 'week'
122
+ date.beginning_of_week
123
+ elsif interval == 'day'
124
+ date.beginning_of_day
125
+ elsif interval == 'hour'
126
+ date.beginning_of_hour
127
+ elsif interval == 'minute'
128
+ date.beginning_of_minute
129
+ else
130
+ abort
131
+ end.to_date
132
+ }
133
+ when 'DateTime'
134
+ ->(value) { Time.at(value).utc.change(sec: 0) }
135
+ when 'ActiveSupport::TimeWithZone'
136
+ ->(value) { Time.at(value).utc }
137
+ else
138
+ raise ArgumentError
139
+ end
140
+
141
+ base
142
+ .step(1.send(interval))
143
+ .to_a
144
+ .map { |value| function.call(value) }
145
+ end
65
146
  end
66
147
  end
67
148
  end
@@ -4,7 +4,8 @@ module AhoyCaptain
4
4
  # @todo: this is lazy
5
5
  def index
6
6
  @stats = AhoyCaptain::Stats::BounceRatesQuery.call(params)
7
- @stats = @stats.group_by_period(selected_interval, "daily_bounce_rate.date").average("bounce_rate")
7
+ @stats = lazy_window(@stats.with_lazy_comparison(compare_mode?).group_by_period(selected_interval, "daily_bounce_rate.date").average("bounce_rate"))
8
+ @label = "Bounce Rate"
8
9
  end
9
10
  end
10
11
  end
@@ -2,7 +2,8 @@ module AhoyCaptain
2
2
  module Stats
3
3
  class TotalPageviewsController < BaseController
4
4
  def index
5
- @stats = AhoyCaptain::Stats::TotalPageviewsQuery.call(params).group_by_period(selected_interval, :time).count
5
+ @stats = lazy_window(AhoyCaptain::Stats::TotalPageviewsQuery.call(params).with_lazy_comparison(compare_mode?).group_by_period(selected_interval, :time).count, 0)
6
+ @label = "Visitors"
6
7
  end
7
8
  end
8
9
  end