ahoy_captain 0.8 → 0.10.0
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +4 -4
- data/README.md +25 -13
- data/app/assets/javascript/ahoy_captain/application.js +4 -4
- data/app/assets/javascript/ahoy_captain/controllers/active_links_controller.js +30 -0
- data/app/assets/javascript/ahoy_captain/controllers/application.js +5 -5
- data/app/assets/javascript/ahoy_captain/controllers/application_controller.js +16 -9
- data/app/assets/javascript/ahoy_captain/controllers/combobox_controller.js +341 -0
- data/app/assets/javascript/ahoy_captain/controllers/details_modal_controller.js +5 -5
- data/app/assets/javascript/ahoy_captain/controllers/dropdown_label_controller.js +2 -2
- data/app/assets/javascript/ahoy_captain/controllers/filter/item_controller.js +12 -0
- data/app/assets/javascript/ahoy_captain/controllers/filter_form_controller.js +13 -0
- data/app/assets/javascript/ahoy_captain/controllers/filter_modal_controller.js +45 -0
- data/app/assets/javascript/ahoy_captain/controllers/funnel_chart_controller.js +116 -104
- data/app/assets/javascript/ahoy_captain/controllers/index.js +4 -3
- data/app/assets/javascript/ahoy_captain/controllers/interval_controller.js +8 -3
- data/app/assets/javascript/ahoy_captain/controllers/line_chart_controller.js +188 -0
- data/app/assets/javascript/ahoy_captain/controllers/predicate_select_controller.js +9 -0
- data/app/assets/javascript/ahoy_captain/controllers/properties_controller.js +8 -0
- data/app/assets/javascript/ahoy_captain/controllers/property_filter_controller.js +46 -0
- data/app/assets/javascript/ahoy_captain/controllers/realtime_controller.js +12 -9
- data/app/assets/javascript/ahoy_captain/controllers/tile_controller.js +9 -0
- data/app/assets/javascript/ahoy_captain/controllers/toggle_controller.js +17 -0
- data/app/assets/javascript/ahoy_captain/helpers/chart_utils.js +156 -0
- data/app/assets/javascript/ahoy_captain/helpers/number_formatters.js +55 -0
- data/app/components/ahoy_captain/combobox_component.html.erb +33 -0
- data/app/components/ahoy_captain/combobox_component.rb +13 -0
- data/app/components/ahoy_captain/comparison_link_component.rb +40 -0
- data/app/components/ahoy_captain/dropdown_button_component.html.erb +5 -5
- data/app/components/ahoy_captain/dropdown_link_component.html.erb +5 -5
- data/app/components/ahoy_captain/filter/dropdown_component.html.erb +50 -0
- data/app/components/ahoy_captain/filter/dropdown_component.rb +51 -0
- data/app/components/ahoy_captain/filter/modal_component.html.erb +12 -9
- data/app/components/ahoy_captain/filter/select_component.html.erb +23 -19
- data/app/components/ahoy_captain/filter/select_component.rb +41 -9
- data/app/components/ahoy_captain/filter/tag_component.html.erb +8 -4
- data/app/components/ahoy_captain/filter/tag_component.rb +6 -30
- data/app/components/ahoy_captain/filter/tag_container_component.html.erb +2 -3
- data/app/components/ahoy_captain/filter/tag_container_component.rb +1 -8
- data/app/components/ahoy_captain/stats/comparable_container_component.html.erb +25 -0
- data/app/components/ahoy_captain/stats/comparable_container_component.rb +86 -0
- data/app/components/ahoy_captain/stats/container_component.html.erb +15 -0
- data/app/components/ahoy_captain/stats/container_component.rb +26 -0
- data/app/components/ahoy_captain/sticky_nav_component.html.erb +13 -22
- data/app/components/ahoy_captain/sticky_nav_component.rb +11 -0
- data/app/components/ahoy_captain/table_component.html.erb +4 -37
- data/app/components/ahoy_captain/table_component.rb +25 -5
- data/app/components/ahoy_captain/tables/devices_table_component.rb +11 -0
- data/app/components/ahoy_captain/tables/dynamic_table.rb +13 -0
- data/app/components/ahoy_captain/tables/dynamic_table_component.rb +204 -0
- data/app/components/ahoy_captain/tables/goals_table_component.rb +17 -0
- data/app/components/ahoy_captain/tables/header_component.html.erb +5 -0
- data/app/components/ahoy_captain/tables/header_component.rb +18 -0
- data/app/components/ahoy_captain/tables/headers/header_component.html.erb +5 -0
- data/app/components/ahoy_captain/tables/headers/header_component.rb +16 -0
- data/app/components/ahoy_captain/tables/properties_table_component.rb +27 -0
- data/app/components/ahoy_captain/tables/row_component.html.erb +4 -0
- data/app/components/ahoy_captain/tables/rows/row_component.html.erb +6 -0
- data/app/components/ahoy_captain/tables/rows/row_component.rb +40 -0
- data/app/components/ahoy_captain/tile_component.html.erb +21 -10
- data/app/components/ahoy_captain/tile_component.rb +3 -2
- data/app/components/ahoy_captain/tooltip_component.html.erb +2 -2
- data/app/controllers/ahoy_captain/application_controller.rb +19 -29
- data/app/controllers/ahoy_captain/campaigns_controller.rb +2 -10
- data/app/controllers/ahoy_captain/cities_controller.rb +2 -6
- data/app/controllers/ahoy_captain/countries_controller.rb +2 -6
- data/app/controllers/ahoy_captain/devices_controller.rb +3 -6
- data/app/controllers/ahoy_captain/entry_pages_controller.rb +2 -4
- data/app/controllers/ahoy_captain/exit_pages_controller.rb +3 -4
- data/app/controllers/ahoy_captain/exports_controller.rb +14 -0
- data/app/controllers/ahoy_captain/filters/base_controller.rb +1 -3
- data/app/controllers/ahoy_captain/filters/goals_controller.rb +9 -0
- data/app/controllers/ahoy_captain/filters/pages/actions_controller.rb +1 -1
- data/app/controllers/ahoy_captain/filters/pages/entry_pages_controller.rb +3 -3
- data/app/controllers/ahoy_captain/filters/pages/exit_pages_controller.rb +2 -3
- data/app/controllers/ahoy_captain/filters/properties/names_controller.rb +11 -0
- data/app/controllers/ahoy_captain/filters/properties/values_controller.rb +15 -0
- data/app/controllers/ahoy_captain/filters/sources_controller.rb +1 -1
- data/app/controllers/ahoy_captain/filters/utms_controller.rb +1 -1
- data/app/controllers/ahoy_captain/properties_controller.rb +41 -0
- data/app/controllers/ahoy_captain/regions_controller.rb +3 -7
- data/app/controllers/ahoy_captain/sources_controller.rb +2 -5
- data/app/controllers/ahoy_captain/stats/base_controller.rb +86 -5
- data/app/controllers/ahoy_captain/stats/bounce_rates_controller.rb +2 -1
- data/app/controllers/ahoy_captain/stats/total_pageviews_controller.rb +2 -1
- data/app/controllers/ahoy_captain/stats/total_visits_controller.rb +2 -1
- data/app/controllers/ahoy_captain/stats/unique_visitors_controller.rb +3 -1
- data/app/controllers/ahoy_captain/stats/views_per_visits_controller.rb +3 -11
- data/app/controllers/ahoy_captain/stats/visit_durations_controller.rb +2 -1
- data/app/controllers/ahoy_captain/top_pages_controller.rb +2 -8
- data/app/decorators/ahoy_captain/application_decorator.rb +27 -3
- data/app/decorators/ahoy_captain/campaign_decorator.rb +8 -0
- data/app/decorators/ahoy_captain/city_decorator.rb +12 -0
- data/app/decorators/ahoy_captain/country_decorator.rb +10 -0
- data/app/decorators/ahoy_captain/device_decorator.rb +13 -2
- data/app/decorators/ahoy_captain/page_decorator.rb +11 -0
- data/app/decorators/ahoy_captain/region_decorator.rb +16 -0
- data/app/decorators/ahoy_captain/source_decorator.rb +7 -0
- data/app/helpers/ahoy_captain/application_helper.rb +62 -3
- data/app/models/ahoy_captain/comparison_mode.rb +72 -0
- data/app/models/ahoy_captain/export.rb +48 -0
- data/app/models/ahoy_captain/filter_parser.rb +82 -0
- data/app/models/ahoy_captain/range_from_params.rb +75 -0
- data/app/models/ahoy_captain/rangeable.rb +0 -3
- data/app/models/concerns/ahoy_captain/compare_mode.rb +19 -0
- data/app/models/concerns/ahoy_captain/limitable.rb +17 -0
- data/app/models/concerns/ahoy_captain/range_options.rb +1 -14
- data/app/presenters/ahoy_captain/dashboard_presenter.rb +18 -49
- data/app/presenters/ahoy_captain/goals_presenter.rb +3 -2
- data/app/queries/ahoy_captain/application_query.rb +78 -13
- data/app/queries/ahoy_captain/campaign_query.rb +14 -0
- data/app/queries/ahoy_captain/city_query.rb +11 -0
- data/app/queries/ahoy_captain/country_query.rb +10 -0
- data/app/queries/ahoy_captain/device_query.rb +10 -0
- data/app/queries/ahoy_captain/entry_pages_query.rb +3 -2
- data/app/queries/ahoy_captain/event_query.rb +20 -13
- data/app/queries/ahoy_captain/exit_pages_query.rb +6 -4
- data/app/queries/ahoy_captain/region_query.rb +11 -0
- data/app/queries/ahoy_captain/source_query.rb +10 -0
- data/app/queries/ahoy_captain/stats/average_views_per_visit_query.rb +11 -4
- data/app/queries/ahoy_captain/stats/average_visit_duration_query.rb +14 -2
- data/app/queries/ahoy_captain/stats/base_query.rb +18 -0
- data/app/queries/ahoy_captain/stats/bounce_rates_query.rb +15 -1
- data/app/queries/ahoy_captain/stats/total_pageviews_query.rb +2 -2
- data/app/queries/ahoy_captain/stats/total_visitors_query.rb +1 -1
- data/app/queries/ahoy_captain/stats/unique_visitors_query.rb +1 -1
- data/app/queries/ahoy_captain/stats/views_per_visit_query.rb +1 -1
- data/app/queries/ahoy_captain/stats/visit_duration_query.rb +3 -3
- data/app/queries/ahoy_captain/top_page_query.rb +13 -0
- data/app/queries/ahoy_captain/visit_query.rb +2 -3
- data/app/queries/concerns/ahoy_captain/comparable_queries.rb +25 -0
- data/app/queries/concerns/ahoy_captain/comparable_query.rb +138 -0
- data/app/queries/concerns/ahoy_captain/lazy_comparable_query.rb +42 -0
- data/app/views/ahoy_captain/devices/_table.html.erb +2 -0
- data/app/views/ahoy_captain/devices/index.html+details.erb +1 -1
- data/app/views/ahoy_captain/devices/index.html.erb +2 -2
- data/app/views/ahoy_captain/funnels/show.html.erb +5 -2
- data/app/views/ahoy_captain/goals/index.html.erb +2 -37
- data/app/views/ahoy_captain/layouts/application.html.erb +3 -4
- data/app/views/ahoy_captain/properties/_form.html.erb +6 -0
- data/app/views/ahoy_captain/properties/index.html.erb +3 -0
- data/app/views/ahoy_captain/properties/show.html.erb +6 -0
- data/app/views/ahoy_captain/realtimes/show.html.erb +1 -1
- data/app/views/ahoy_captain/roots/_filters.html.erb +80 -0
- data/app/views/ahoy_captain/roots/show.html.erb +76 -109
- data/app/views/ahoy_captain/stats/base/index.html.erb +34 -9
- data/app/views/ahoy_captain/stats/show.html.erb +8 -55
- data/config/routes.rb +9 -0
- data/lib/ahoy_captain/ahoy/event_methods.rb +35 -74
- data/lib/ahoy_captain/ahoy/visit_methods.rb +1 -1
- data/lib/ahoy_captain/configuration.rb +18 -7
- data/lib/ahoy_captain/engine.rb +22 -0
- data/lib/ahoy_captain/filter_configuration/filter.rb +16 -0
- data/lib/ahoy_captain/filter_configuration/filter_collection.rb +48 -0
- data/lib/ahoy_captain/filters_configuration.rb +77 -0
- data/lib/ahoy_captain/goals.rb +10 -2
- data/lib/ahoy_captain/period_collection.rb +1 -1
- data/lib/ahoy_captain/predicate_label.rb +7 -0
- data/lib/ahoy_captain/version.rb +1 -1
- data/lib/ahoy_captain.rb +7 -1
- data/lib/generators/ahoy_captain/templates/config.rb.tt +32 -0
- metadata +80 -21
- data/app/assets/javascript/ahoy_captain/controllers/filter_controller.js +0 -145
- data/app/assets/javascript/ahoy_captain/controllers/filter_tag_controller.js +0 -17
- data/app/assets/javascript/ahoy_captain/controllers/link_controller.js +0 -43
- data/app/assets/javascript/ahoy_captain/controllers/navigation_controller.js +0 -25
- data/app/models/ahoy_captain/current.rb +0 -9
- 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
|
2
|
-
|
3
|
-
|
4
|
-
|
5
|
-
|
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
|
-
|
15
|
+
<% end %>
|
16
|
+
|
8
17
|
<%= statistic_display %>
|
9
|
-
|
10
|
-
|
11
|
-
|
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
|
-
|
26
|
-
|
10
|
+
def period
|
11
|
+
params[:period] || AhoyCaptain.config.ranges.default
|
27
12
|
end
|
28
13
|
|
29
14
|
# show the details frame
|
30
|
-
before_action
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
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
|
-
|
14
|
-
|
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
|
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
|
14
|
-
.
|
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
|
@@ -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.
|
8
|
-
|
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.
|
6
|
+
query = event_query.distinct("exit_pages.url").select("exit_pages.url as url")
|
8
7
|
|
9
|
-
render json: query.map { |row|
|
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.
|
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
|
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
|
-
|
14
|
-
|
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
|
-
|
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
|
44
|
+
diff = (range[1] - range[0]).seconds.in_days
|
45
|
+
if diff >= 31
|
37
46
|
"month"
|
38
|
-
elsif diff
|
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
|
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
|