ahoy_matey 1.5.3 → 3.0.1

Sign up to get free protection for your applications and to get access to all the features.
Files changed (90) hide show
  1. checksums.yaml +5 -5
  2. data/CHANGELOG.md +88 -0
  3. data/CONTRIBUTING.md +42 -0
  4. data/LICENSE.txt +1 -1
  5. data/README.md +369 -380
  6. data/app/controllers/ahoy/base_controller.rb +15 -15
  7. data/app/controllers/ahoy/events_controller.rb +8 -2
  8. data/app/controllers/ahoy/visits_controller.rb +8 -1
  9. data/app/jobs/ahoy/geocode_job.rb +10 -0
  10. data/app/jobs/ahoy/geocode_v2_job.rb +28 -0
  11. data/config/routes.rb +1 -1
  12. data/lib/ahoy.rb +68 -86
  13. data/lib/ahoy/base_store.rb +97 -0
  14. data/lib/ahoy/controller.rb +26 -17
  15. data/lib/ahoy/database_store.rb +89 -0
  16. data/lib/ahoy/engine.rb +7 -7
  17. data/lib/ahoy/helper.rb +40 -0
  18. data/lib/ahoy/model.rb +5 -27
  19. data/lib/ahoy/{properties.rb → query_methods.rb} +25 -9
  20. data/lib/ahoy/tracker.rb +101 -42
  21. data/lib/ahoy/utils.rb +7 -0
  22. data/lib/ahoy/version.rb +1 -1
  23. data/lib/ahoy/visit_properties.rb +99 -37
  24. data/lib/generators/ahoy/activerecord_generator.rb +41 -0
  25. data/lib/generators/ahoy/base_generator.rb +13 -0
  26. data/lib/generators/ahoy/install_generator.rb +44 -0
  27. data/lib/generators/ahoy/mongoid_generator.rb +16 -0
  28. data/lib/generators/ahoy/templates/active_record_event_model.rb.tt +10 -0
  29. data/lib/generators/ahoy/templates/active_record_migration.rb.tt +62 -0
  30. data/lib/generators/ahoy/templates/active_record_visit_model.rb.tt +6 -0
  31. data/lib/generators/ahoy/templates/base_store_initializer.rb.tt +20 -0
  32. data/lib/generators/ahoy/templates/database_store_initializer.rb.tt +5 -0
  33. data/lib/generators/ahoy/{stores/templates/mongoid_event_model.rb → templates/mongoid_event_model.rb.tt} +4 -2
  34. data/lib/generators/ahoy/{stores/templates/mongoid_visit_model.rb → templates/mongoid_visit_model.rb.tt} +15 -9
  35. data/vendor/assets/javascripts/ahoy.js +336 -113
  36. metadata +54 -144
  37. data/.gitignore +0 -17
  38. data/Gemfile +0 -6
  39. data/Rakefile +0 -8
  40. data/ahoy_matey.gemspec +0 -38
  41. data/lib/ahoy/deckhands/location_deckhand.rb +0 -49
  42. data/lib/ahoy/deckhands/request_deckhand.rb +0 -52
  43. data/lib/ahoy/deckhands/technology_deckhand.rb +0 -47
  44. data/lib/ahoy/deckhands/traffic_source_deckhand.rb +0 -22
  45. data/lib/ahoy/deckhands/utm_parameter_deckhand.rb +0 -23
  46. data/lib/ahoy/geocode_job.rb +0 -13
  47. data/lib/ahoy/logger_silencer.rb +0 -75
  48. data/lib/ahoy/stores/active_record_store.rb +0 -60
  49. data/lib/ahoy/stores/active_record_token_store.rb +0 -113
  50. data/lib/ahoy/stores/base_store.rb +0 -88
  51. data/lib/ahoy/stores/bunny_store.rb +0 -33
  52. data/lib/ahoy/stores/fluentd_store.rb +0 -17
  53. data/lib/ahoy/stores/kafka_store.rb +0 -40
  54. data/lib/ahoy/stores/kinesis_firehose_store.rb +0 -42
  55. data/lib/ahoy/stores/log_store.rb +0 -53
  56. data/lib/ahoy/stores/mongoid_store.rb +0 -63
  57. data/lib/ahoy/subscribers/active_record.rb +0 -19
  58. data/lib/ahoy/throttle.rb +0 -17
  59. data/lib/generators/ahoy/stores/active_record_events_generator.rb +0 -53
  60. data/lib/generators/ahoy/stores/active_record_generator.rb +0 -16
  61. data/lib/generators/ahoy/stores/active_record_visits_generator.rb +0 -43
  62. data/lib/generators/ahoy/stores/bunny_generator.rb +0 -15
  63. data/lib/generators/ahoy/stores/custom_generator.rb +0 -15
  64. data/lib/generators/ahoy/stores/fluentd_generator.rb +0 -15
  65. data/lib/generators/ahoy/stores/kafka_generator.rb +0 -15
  66. data/lib/generators/ahoy/stores/kinesis_firehose_generator.rb +0 -15
  67. data/lib/generators/ahoy/stores/log_generator.rb +0 -15
  68. data/lib/generators/ahoy/stores/mongoid_events_generator.rb +0 -19
  69. data/lib/generators/ahoy/stores/mongoid_generator.rb +0 -14
  70. data/lib/generators/ahoy/stores/mongoid_visits_generator.rb +0 -27
  71. data/lib/generators/ahoy/stores/templates/active_record_event_model.rb +0 -12
  72. data/lib/generators/ahoy/stores/templates/active_record_events_migration.rb +0 -19
  73. data/lib/generators/ahoy/stores/templates/active_record_initializer.rb +0 -3
  74. data/lib/generators/ahoy/stores/templates/active_record_visit_model.rb +0 -4
  75. data/lib/generators/ahoy/stores/templates/active_record_visits_migration.rb +0 -57
  76. data/lib/generators/ahoy/stores/templates/bunny_initializer.rb +0 -9
  77. data/lib/generators/ahoy/stores/templates/custom_initializer.rb +0 -10
  78. data/lib/generators/ahoy/stores/templates/fluentd_initializer.rb +0 -3
  79. data/lib/generators/ahoy/stores/templates/kafka_initializer.rb +0 -9
  80. data/lib/generators/ahoy/stores/templates/kinesis_firehose_initializer.rb +0 -17
  81. data/lib/generators/ahoy/stores/templates/log_initializer.rb +0 -3
  82. data/lib/generators/ahoy/stores/templates/mongoid_initializer.rb +0 -3
  83. data/test/properties/mysql_json_test.rb +0 -18
  84. data/test/properties/mysql_text_test.rb +0 -19
  85. data/test/properties/postgresql_hstore_test.rb +0 -18
  86. data/test/properties/postgresql_json_test.rb +0 -18
  87. data/test/properties/postgresql_jsonb_test.rb +0 -18
  88. data/test/properties/postgresql_text_test.rb +0 -19
  89. data/test/test_helper.rb +0 -99
  90. data/test/visit_properties_test.rb +0 -44
@@ -0,0 +1,89 @@
1
+ module Ahoy
2
+ class DatabaseStore < BaseStore
3
+ def track_visit(data)
4
+ @visit = visit_model.create!(slice_data(visit_model, data))
5
+ rescue => e
6
+ raise e unless unique_exception?(e)
7
+
8
+ # so next call to visit will try to fetch from DB
9
+ if defined?(@visit)
10
+ remove_instance_variable(:@visit)
11
+ end
12
+ end
13
+
14
+ def track_event(data)
15
+ visit = visit_or_create(started_at: data[:time])
16
+ if visit
17
+ event = event_model.new(slice_data(event_model, data))
18
+ event.visit = visit
19
+ event.time = visit.started_at if event.time < visit.started_at
20
+ begin
21
+ event.save!
22
+ rescue => e
23
+ raise e unless unique_exception?(e)
24
+ end
25
+ else
26
+ Ahoy.log "Event excluded since visit not created: #{data[:visit_token]}"
27
+ end
28
+ end
29
+
30
+ def geocode(data)
31
+ visit_token = data.delete(:visit_token)
32
+ data = slice_data(visit_model, data)
33
+ if defined?(Mongoid::Document) && visit_model < Mongoid::Document
34
+ # upsert since visit might not be found due to eventual consistency
35
+ visit_model.where(visit_token: visit_token).find_one_and_update({"$set": data}, {upsert: true})
36
+ elsif visit
37
+ visit.update!(data)
38
+ else
39
+ Ahoy.log "Visit for geocode not found: #{visit_token}"
40
+ end
41
+ end
42
+
43
+ def authenticate(_)
44
+ if visit && visit.respond_to?(:user) && !visit.user
45
+ begin
46
+ visit.user = user
47
+ visit.save!
48
+ rescue ActiveRecord::AssociationTypeMismatch
49
+ # do nothing
50
+ end
51
+ end
52
+ end
53
+
54
+ def visit
55
+ unless defined?(@visit)
56
+ @visit = visit_model.where(visit_token: ahoy.visit_token).first if ahoy.visit_token
57
+ end
58
+ @visit
59
+ end
60
+
61
+ # if we don't have a visit, let's try to create one first
62
+ def visit_or_create(started_at: nil)
63
+ ahoy.track_visit(started_at: started_at) if !visit && Ahoy.server_side_visits
64
+ visit
65
+ end
66
+
67
+ protected
68
+
69
+ def visit_model
70
+ ::Ahoy::Visit
71
+ end
72
+
73
+ def event_model
74
+ ::Ahoy::Event
75
+ end
76
+
77
+ def slice_data(model, data)
78
+ column_names = model.try(:column_names) || model.attribute_names
79
+ data.slice(*column_names.map(&:to_sym)).select { |_, v| !v.nil? }
80
+ end
81
+
82
+ def unique_exception?(e)
83
+ return true if defined?(ActiveRecord::RecordNotUnique) && e.is_a?(ActiveRecord::RecordNotUnique)
84
+ return true if defined?(PG::UniqueViolation) && e.is_a?(PG::UniqueViolation)
85
+ return true if defined?(Mongo::Error::OperationFailure) && e.is_a?(Mongo::Error::OperationFailure) && e.message.include?("duplicate key error")
86
+ false
87
+ end
88
+ end
89
+ end
data/lib/ahoy/engine.rb CHANGED
@@ -1,10 +1,10 @@
1
1
  module Ahoy
2
2
  class Engine < ::Rails::Engine
3
- initializer "ahoy.middleware", after: "sprockets.environment" do |app|
4
- if Ahoy.throttle
5
- require "ahoy/throttle"
6
- app.middleware.use Ahoy::Throttle
7
- end
3
+ initializer "ahoy", after: "sprockets.environment" do
4
+ Ahoy.logger ||= Rails.logger
5
+
6
+ # allow Devise to be loaded after Ahoy
7
+ require "ahoy/warden" if defined?(Warden)
8
8
 
9
9
  next unless Ahoy.quiet
10
10
 
@@ -14,8 +14,8 @@ module Ahoy
14
14
  # Just create an alias for call in middleware
15
15
  Rails::Rack::Logger.class_eval do
16
16
  def call_with_quiet_ahoy(env)
17
- if env["PATH_INFO"].start_with?(AHOY_PREFIX) && logger.respond_to?(:silence_logger)
18
- logger.silence_logger do
17
+ if env["PATH_INFO"].start_with?(AHOY_PREFIX) && logger.respond_to?(:silence)
18
+ logger.silence do
19
19
  call_without_quiet_ahoy(env)
20
20
  end
21
21
  else
@@ -0,0 +1,40 @@
1
+ module Ahoy
2
+ module Helper
3
+ def amp_event(name, properties = {})
4
+ url = Ahoy::Engine.routes.url_helpers.events_url(
5
+ url_options.slice(:host, :port, :protocol).merge(
6
+ name: name,
7
+ properties: properties,
8
+ screen_width: "SCREEN_WIDTH",
9
+ screen_height: "SCREEN_HEIGHT",
10
+ platform: "Web",
11
+ landing_page: "AMPDOC_URL",
12
+ referrer: "DOCUMENT_REFERRER",
13
+ random: "RANDOM"
14
+ )
15
+ )
16
+ url = "#{url}&visit_token=${clientId(ahoy_visit)}&visitor_token=${clientId(ahoy_visitor)}"
17
+
18
+ content_tag "amp-analytics" do
19
+ content_tag "script", type: "application/json" do
20
+ json_escape({
21
+ requests: {
22
+ pageview: url
23
+ },
24
+ triggers: {
25
+ trackPageview: {
26
+ on: "visible",
27
+ request: "pageview"
28
+ }
29
+ },
30
+ transport: {
31
+ beacon: true,
32
+ xhrpost: true,
33
+ image: false
34
+ }
35
+ }.to_json).html_safe
36
+ end
37
+ end
38
+ end
39
+ end
40
+ end
data/lib/ahoy/model.rb CHANGED
@@ -1,37 +1,15 @@
1
1
  module Ahoy
2
2
  module Model
3
- def visitable(name = nil, options = {})
4
- if name.is_a?(Hash)
5
- name = nil
6
- options = name
7
- end
8
- name ||= :visit
3
+ def visitable(name = :visit, **options)
9
4
  class_eval do
10
- belongs_to name, options
11
- before_create :set_visit
5
+ belongs_to(name, class_name: "Ahoy::Visit", optional: true, **options)
6
+ before_create :set_ahoy_visit
12
7
  end
13
8
  class_eval %{
14
- def set_visit
15
- self.#{name} ||= RequestStore.store[:ahoy].try(:visit)
9
+ def set_ahoy_visit
10
+ self.#{name} ||= Thread.current[:ahoy].try(:visit_or_create)
16
11
  end
17
12
  }
18
13
  end
19
-
20
- # deprecated
21
-
22
- def ahoy_visit
23
- class_eval do
24
- warn "[DEPRECATION] ahoy_visit is deprecated"
25
-
26
- belongs_to :user, polymorphic: true
27
-
28
- def landing_params
29
- @landing_params ||= begin
30
- warn "[DEPRECATION] landing_params is deprecated"
31
- Deckhands::UtmParameterDeckhand.new(landing_page).landing_params
32
- end
33
- end
34
- end
35
- end
36
14
  end
37
15
  end
@@ -1,13 +1,23 @@
1
1
  module Ahoy
2
- module Properties
2
+ module QueryMethods
3
3
  extend ActiveSupport::Concern
4
4
 
5
5
  module ClassMethods
6
- def where_properties(properties)
6
+ def where_event(name, properties = {})
7
+ where(name: name).where_props(properties)
8
+ end
9
+
10
+ def where_props(properties)
7
11
  relation = self
8
- column_type = columns_hash["properties"].type
9
- adapter_name = connection.adapter_name.downcase
12
+ if respond_to?(:columns_hash)
13
+ column_type = columns_hash["properties"].type
14
+ adapter_name = connection.adapter_name.downcase
15
+ else
16
+ adapter_name = "mongoid"
17
+ end
10
18
  case adapter_name
19
+ when "mongoid"
20
+ relation = where(Hash[properties.map { |k, v| ["properties.#{k}", v] }])
11
21
  when /mysql/
12
22
  if column_type == :json
13
23
  properties.each do |k, v|
@@ -17,15 +27,17 @@ module Ahoy
17
27
  v = "true"
18
28
  end
19
29
 
20
- relation = relation.where("JSON_UNQUOTE(properties -> ?) = ?", "$.#{k.to_s}", v.as_json)
30
+ relation = relation.where("JSON_UNQUOTE(properties -> ?) = ?", "$.#{k}", v.as_json)
21
31
  end
22
32
  else
23
33
  properties.each do |k, v|
24
- relation = relation.where("properties REGEXP ?", "[{,]#{{k.to_s => v}.to_json.sub(/\A\{/, "").sub(/\}\z/, "")}[,}]")
34
+ relation = relation.where("properties REGEXP ?", "[{,]#{{k.to_s => v}.to_json.sub(/\A\{/, "").sub(/\}\z/, "").gsub("+", "\\\\+")}[,}]")
25
35
  end
26
36
  end
27
- when /postgres/
28
- if column_type == :jsonb || column_type == :json
37
+ when /postgres|postgis/
38
+ if column_type == :jsonb
39
+ relation = relation.where("properties @> ?", properties.to_json)
40
+ elsif column_type == :json
29
41
  properties.each do |k, v|
30
42
  relation =
31
43
  if v.nil?
@@ -45,7 +57,7 @@ module Ahoy
45
57
  end
46
58
  else
47
59
  properties.each do |k, v|
48
- relation = relation.where("properties SIMILAR TO ?", "%[{,]#{{k.to_s => v}.to_json.sub(/\A\{/, "").sub(/\}\z/, "")}[,}]%")
60
+ relation = relation.where("properties SIMILAR TO ?", "%[{,]#{{k.to_s => v}.to_json.sub(/\A\{/, "").sub(/\}\z/, "").gsub("+", "\\\\+")}[,}]%")
49
61
  end
50
62
  end
51
63
  else
@@ -53,6 +65,10 @@ module Ahoy
53
65
  end
54
66
  relation
55
67
  end
68
+ alias_method :where_properties, :where_props
56
69
  end
57
70
  end
58
71
  end
72
+
73
+ # backward compatibility
74
+ Ahoy::Properties = Ahoy::QueryMethods
data/lib/ahoy/tracker.rb CHANGED
@@ -1,46 +1,64 @@
1
+ require "active_support/core_ext/digest/uuid"
2
+
1
3
  module Ahoy
2
4
  class Tracker
5
+ UUID_NAMESPACE = "a82ae811-5011-45ab-a728-569df7499c5f"
6
+
3
7
  attr_reader :request, :controller
4
8
 
5
- def initialize(options = {})
9
+ def initialize(**options)
6
10
  @store = Ahoy::Store.new(options.merge(ahoy: self))
7
11
  @controller = options[:controller]
8
12
  @request = options[:request] || @controller.try(:request)
13
+ @visit_token = options[:visit_token]
14
+ @user = options[:user]
9
15
  @options = options
10
16
  end
11
17
 
18
+ # can't use keyword arguments here
12
19
  def track(name, properties = {}, options = {})
13
20
  if exclude?
14
21
  debug "Event excluded"
15
22
  elsif missing_params?
16
23
  debug "Missing required parameters"
17
24
  else
18
- options = options.dup
19
-
20
- options[:time] = trusted_time(options[:time])
21
- options[:id] = ensure_uuid(options[:id] || generate_id)
22
-
23
- @store.track_event(name, properties, options)
25
+ data = {
26
+ visit_token: visit_token,
27
+ user_id: user.try(:id),
28
+ name: name.to_s,
29
+ properties: properties,
30
+ time: trusted_time(options[:time]),
31
+ event_id: options[:id] || generate_id
32
+ }.select { |_, v| v }
33
+
34
+ @store.track_event(data)
24
35
  end
25
36
  true
26
37
  rescue => e
27
38
  report_exception(e)
28
39
  end
29
40
 
30
- def track_visit(options = {})
41
+ def track_visit(defer: false, started_at: nil)
31
42
  if exclude?
32
43
  debug "Visit excluded"
33
44
  elsif missing_params?
34
45
  debug "Missing required parameters"
35
46
  else
36
- if options[:defer]
47
+ if defer
37
48
  set_cookie("ahoy_track", true, nil, false)
38
49
  else
39
- options = options.dup
50
+ delete_cookie("ahoy_track")
40
51
 
41
- options[:started_at] ||= Time.zone.now
52
+ data = {
53
+ visit_token: visit_token,
54
+ visitor_token: visitor_token,
55
+ user_id: user.try(:id),
56
+ started_at: trusted_time(started_at),
57
+ }.merge(visit_properties).select { |_, v| v }
42
58
 
43
- @store.track_visit(options)
59
+ @store.track_visit(data)
60
+
61
+ Ahoy::GeocodeV2Job.perform_later(visit_token, data[:ip]) if Ahoy.geocode && data[:ip]
44
62
  end
45
63
  end
46
64
  true
@@ -48,11 +66,32 @@ module Ahoy
48
66
  report_exception(e)
49
67
  end
50
68
 
69
+ def geocode(data)
70
+ if exclude?
71
+ debug "Geocode excluded"
72
+ else
73
+ data = {
74
+ visit_token: visit_token
75
+ }.merge(data).select { |_, v| v }
76
+
77
+ @store.geocode(data)
78
+ true
79
+ end
80
+ rescue => e
81
+ report_exception(e)
82
+ end
83
+
51
84
  def authenticate(user)
52
85
  if exclude?
53
86
  debug "Authentication excluded"
54
87
  else
55
- @store.authenticate(user)
88
+ @store.user = user
89
+
90
+ data = {
91
+ visit_token: visit_token,
92
+ user_id: user.try(:id)
93
+ }
94
+ @store.authenticate(data)
56
95
  end
57
96
  true
58
97
  rescue => e
@@ -63,16 +102,12 @@ module Ahoy
63
102
  @visit ||= @store.visit
64
103
  end
65
104
 
66
- def visit_id
67
- @visit_id ||= ensure_uuid(visit_token_helper)
68
- end
69
-
70
- def visitor_id
71
- @visitor_id ||= ensure_uuid(visitor_token_helper)
105
+ def visit_or_create
106
+ @visit ||= @store.visit_or_create
72
107
  end
73
108
 
74
109
  def new_visit?
75
- !existing_visit_token
110
+ Ahoy.cookies ? !existing_visit_token : visit.nil?
76
111
  end
77
112
 
78
113
  def new_visitor?
@@ -80,12 +115,12 @@ module Ahoy
80
115
  end
81
116
 
82
117
  def set_visit_cookie
83
- set_cookie("ahoy_visit", visit_id, Ahoy.visit_duration)
118
+ set_cookie("ahoy_visit", visit_token, Ahoy.visit_duration)
84
119
  end
85
120
 
86
121
  def set_visitor_cookie
87
122
  if new_visitor?
88
- set_cookie("ahoy_visitor", visitor_id, Ahoy.visitor_duration)
123
+ set_cookie("ahoy_visitor", visitor_token, Ahoy.visitor_duration)
89
124
  end
90
125
  end
91
126
 
@@ -93,18 +128,30 @@ module Ahoy
93
128
  @user ||= @store.user
94
129
  end
95
130
 
96
- # TODO better name
97
131
  def visit_properties
98
- @visit_properties ||= Ahoy::VisitProperties.new(request, api: api?)
132
+ @visit_properties ||= request ? Ahoy::VisitProperties.new(request, api: api?).generate : {}
99
133
  end
100
134
 
101
135
  def visit_token
102
136
  @visit_token ||= ensure_token(visit_token_helper)
103
137
  end
138
+ alias_method :visit_id, :visit_token
104
139
 
105
140
  def visitor_token
106
141
  @visitor_token ||= ensure_token(visitor_token_helper)
107
142
  end
143
+ alias_method :visitor_id, :visitor_token
144
+
145
+ def reset
146
+ reset_visit
147
+ delete_cookie("ahoy_visitor")
148
+ end
149
+
150
+ def reset_visit
151
+ delete_cookie("ahoy_visit")
152
+ delete_cookie("ahoy_events")
153
+ delete_cookie("ahoy_track")
154
+ end
108
155
 
109
156
  protected
110
157
 
@@ -113,7 +160,7 @@ module Ahoy
113
160
  end
114
161
 
115
162
  def missing_params?
116
- if api? && Ahoy.protect_from_forgery
163
+ if Ahoy.cookies && api? && Ahoy.protect_from_forgery
117
164
  !(existing_visit_token && existing_visitor_token)
118
165
  else
119
166
  false
@@ -121,18 +168,25 @@ module Ahoy
121
168
  end
122
169
 
123
170
  def set_cookie(name, value, duration = nil, use_domain = true)
171
+ # safety net
172
+ return unless Ahoy.cookies && request
173
+
124
174
  cookie = {
125
175
  value: value
126
176
  }
127
177
  cookie[:expires] = duration.from_now if duration
128
- domain = Ahoy.cookie_domain || Ahoy.domain
178
+ domain = Ahoy.cookie_domain
129
179
  cookie[:domain] = domain if domain && use_domain
130
180
  request.cookie_jar[name] = cookie
131
181
  end
132
182
 
133
- def trusted_time(time)
183
+ def delete_cookie(name)
184
+ request.cookie_jar.delete(name) if request && request.cookie_jar[name]
185
+ end
186
+
187
+ def trusted_time(time = nil)
134
188
  if !time || (api? && !(1.minute.ago..Time.now).cover?(time))
135
- Time.zone.now
189
+ Time.current
136
190
  else
137
191
  time
138
192
  end
@@ -142,14 +196,12 @@ module Ahoy
142
196
  @store.exclude?
143
197
  end
144
198
 
145
- # odd pattern for backwards compatibility
146
- # TODO remove this method in next major release
147
199
  def report_exception(e)
148
- Safely.safely do
149
- @store.report_exception(e)
150
- if Rails.env.development? || Rails.env.test?
151
- raise e
152
- end
200
+ if defined?(ActionDispatch::RemoteIp::IpSpoofAttackError) && e.is_a?(ActionDispatch::RemoteIp::IpSpoofAttackError)
201
+ debug "Tracking excluded due to IP spoofing"
202
+ else
203
+ raise e if !defined?(Rails) || Rails.env.development? || Rails.env.test?
204
+ Safely.report_exception(e)
153
205
  end
154
206
  end
155
207
 
@@ -160,6 +212,7 @@ module Ahoy
160
212
  def visit_token_helper
161
213
  @visit_token_helper ||= begin
162
214
  token = existing_visit_token
215
+ token ||= visit_anonymity_set unless Ahoy.cookies
163
216
  token ||= generate_id unless Ahoy.api_only
164
217
  token
165
218
  end
@@ -168,6 +221,7 @@ module Ahoy
168
221
  def visitor_token_helper
169
222
  @visitor_token_helper ||= begin
170
223
  token = existing_visitor_token
224
+ token ||= visitor_anonymity_set unless Ahoy.cookies
171
225
  token ||= generate_id unless Ahoy.api_only
172
226
  token
173
227
  end
@@ -176,7 +230,7 @@ module Ahoy
176
230
  def existing_visit_token
177
231
  @existing_visit_token ||= begin
178
232
  token = visit_header
179
- token ||= visit_cookie unless api? && Ahoy.protect_from_forgery
233
+ token ||= visit_cookie if Ahoy.cookies && !(api? && Ahoy.protect_from_forgery)
180
234
  token ||= visit_param if api?
181
235
  token
182
236
  end
@@ -185,12 +239,20 @@ module Ahoy
185
239
  def existing_visitor_token
186
240
  @existing_visitor_token ||= begin
187
241
  token = visitor_header
188
- token ||= visitor_cookie unless api? && Ahoy.protect_from_forgery
242
+ token ||= visitor_cookie if Ahoy.cookies && !(api? && Ahoy.protect_from_forgery)
189
243
  token ||= visitor_param if api?
190
244
  token
191
245
  end
192
246
  end
193
247
 
248
+ def visit_anonymity_set
249
+ @visit_anonymity_set ||= Digest::UUID.uuid_v5(UUID_NAMESPACE, ["visit", Ahoy.mask_ip(request.remote_ip), request.user_agent].join("/"))
250
+ end
251
+
252
+ def visitor_anonymity_set
253
+ @visitor_anonymity_set ||= Digest::UUID.uuid_v5(UUID_NAMESPACE, ["visitor", Ahoy.mask_ip(request.remote_ip), request.user_agent].join("/"))
254
+ end
255
+
194
256
  def visit_cookie
195
257
  @visit_cookie ||= request && request.cookies["ahoy_visit"]
196
258
  end
@@ -215,16 +277,13 @@ module Ahoy
215
277
  @visitor_param ||= request && request.params["visitor_token"]
216
278
  end
217
279
 
218
- def ensure_uuid(id)
219
- Ahoy.ensure_uuid(id) if id
220
- end
221
-
222
280
  def ensure_token(token)
281
+ token = Ahoy::Utils.ensure_utf8(token)
223
282
  token.to_s.gsub(/[^a-z0-9\-]/i, "").first(64) if token
224
283
  end
225
284
 
226
285
  def debug(message)
227
- Rails.logger.debug { "[ahoy] #{message}" }
286
+ Ahoy.log message
228
287
  end
229
288
  end
230
289
  end