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.
- checksums.yaml +5 -5
- data/CHANGELOG.md +88 -0
- data/CONTRIBUTING.md +42 -0
- data/LICENSE.txt +1 -1
- data/README.md +369 -380
- data/app/controllers/ahoy/base_controller.rb +15 -15
- data/app/controllers/ahoy/events_controller.rb +8 -2
- data/app/controllers/ahoy/visits_controller.rb +8 -1
- data/app/jobs/ahoy/geocode_job.rb +10 -0
- data/app/jobs/ahoy/geocode_v2_job.rb +28 -0
- data/config/routes.rb +1 -1
- data/lib/ahoy.rb +68 -86
- data/lib/ahoy/base_store.rb +97 -0
- data/lib/ahoy/controller.rb +26 -17
- data/lib/ahoy/database_store.rb +89 -0
- data/lib/ahoy/engine.rb +7 -7
- data/lib/ahoy/helper.rb +40 -0
- data/lib/ahoy/model.rb +5 -27
- data/lib/ahoy/{properties.rb → query_methods.rb} +25 -9
- data/lib/ahoy/tracker.rb +101 -42
- data/lib/ahoy/utils.rb +7 -0
- data/lib/ahoy/version.rb +1 -1
- data/lib/ahoy/visit_properties.rb +99 -37
- data/lib/generators/ahoy/activerecord_generator.rb +41 -0
- data/lib/generators/ahoy/base_generator.rb +13 -0
- data/lib/generators/ahoy/install_generator.rb +44 -0
- data/lib/generators/ahoy/mongoid_generator.rb +16 -0
- data/lib/generators/ahoy/templates/active_record_event_model.rb.tt +10 -0
- data/lib/generators/ahoy/templates/active_record_migration.rb.tt +62 -0
- data/lib/generators/ahoy/templates/active_record_visit_model.rb.tt +6 -0
- data/lib/generators/ahoy/templates/base_store_initializer.rb.tt +20 -0
- data/lib/generators/ahoy/templates/database_store_initializer.rb.tt +5 -0
- data/lib/generators/ahoy/{stores/templates/mongoid_event_model.rb → templates/mongoid_event_model.rb.tt} +4 -2
- data/lib/generators/ahoy/{stores/templates/mongoid_visit_model.rb → templates/mongoid_visit_model.rb.tt} +15 -9
- data/vendor/assets/javascripts/ahoy.js +336 -113
- metadata +54 -144
- data/.gitignore +0 -17
- data/Gemfile +0 -6
- data/Rakefile +0 -8
- data/ahoy_matey.gemspec +0 -38
- data/lib/ahoy/deckhands/location_deckhand.rb +0 -49
- data/lib/ahoy/deckhands/request_deckhand.rb +0 -52
- data/lib/ahoy/deckhands/technology_deckhand.rb +0 -47
- data/lib/ahoy/deckhands/traffic_source_deckhand.rb +0 -22
- data/lib/ahoy/deckhands/utm_parameter_deckhand.rb +0 -23
- data/lib/ahoy/geocode_job.rb +0 -13
- data/lib/ahoy/logger_silencer.rb +0 -75
- data/lib/ahoy/stores/active_record_store.rb +0 -60
- data/lib/ahoy/stores/active_record_token_store.rb +0 -113
- data/lib/ahoy/stores/base_store.rb +0 -88
- data/lib/ahoy/stores/bunny_store.rb +0 -33
- data/lib/ahoy/stores/fluentd_store.rb +0 -17
- data/lib/ahoy/stores/kafka_store.rb +0 -40
- data/lib/ahoy/stores/kinesis_firehose_store.rb +0 -42
- data/lib/ahoy/stores/log_store.rb +0 -53
- data/lib/ahoy/stores/mongoid_store.rb +0 -63
- data/lib/ahoy/subscribers/active_record.rb +0 -19
- data/lib/ahoy/throttle.rb +0 -17
- data/lib/generators/ahoy/stores/active_record_events_generator.rb +0 -53
- data/lib/generators/ahoy/stores/active_record_generator.rb +0 -16
- data/lib/generators/ahoy/stores/active_record_visits_generator.rb +0 -43
- data/lib/generators/ahoy/stores/bunny_generator.rb +0 -15
- data/lib/generators/ahoy/stores/custom_generator.rb +0 -15
- data/lib/generators/ahoy/stores/fluentd_generator.rb +0 -15
- data/lib/generators/ahoy/stores/kafka_generator.rb +0 -15
- data/lib/generators/ahoy/stores/kinesis_firehose_generator.rb +0 -15
- data/lib/generators/ahoy/stores/log_generator.rb +0 -15
- data/lib/generators/ahoy/stores/mongoid_events_generator.rb +0 -19
- data/lib/generators/ahoy/stores/mongoid_generator.rb +0 -14
- data/lib/generators/ahoy/stores/mongoid_visits_generator.rb +0 -27
- data/lib/generators/ahoy/stores/templates/active_record_event_model.rb +0 -12
- data/lib/generators/ahoy/stores/templates/active_record_events_migration.rb +0 -19
- data/lib/generators/ahoy/stores/templates/active_record_initializer.rb +0 -3
- data/lib/generators/ahoy/stores/templates/active_record_visit_model.rb +0 -4
- data/lib/generators/ahoy/stores/templates/active_record_visits_migration.rb +0 -57
- data/lib/generators/ahoy/stores/templates/bunny_initializer.rb +0 -9
- data/lib/generators/ahoy/stores/templates/custom_initializer.rb +0 -10
- data/lib/generators/ahoy/stores/templates/fluentd_initializer.rb +0 -3
- data/lib/generators/ahoy/stores/templates/kafka_initializer.rb +0 -9
- data/lib/generators/ahoy/stores/templates/kinesis_firehose_initializer.rb +0 -17
- data/lib/generators/ahoy/stores/templates/log_initializer.rb +0 -3
- data/lib/generators/ahoy/stores/templates/mongoid_initializer.rb +0 -3
- data/test/properties/mysql_json_test.rb +0 -18
- data/test/properties/mysql_text_test.rb +0 -19
- data/test/properties/postgresql_hstore_test.rb +0 -18
- data/test/properties/postgresql_json_test.rb +0 -18
- data/test/properties/postgresql_jsonb_test.rb +0 -18
- data/test/properties/postgresql_text_test.rb +0 -19
- data/test/test_helper.rb +0 -99
- 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
|
4
|
-
|
5
|
-
|
6
|
-
|
7
|
-
|
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?(:
|
18
|
-
logger.
|
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
|
data/lib/ahoy/helper.rb
ADDED
@@ -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 =
|
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
|
11
|
-
before_create :
|
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
|
15
|
-
self.#{name} ||=
|
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
|
2
|
+
module QueryMethods
|
3
3
|
extend ActiveSupport::Concern
|
4
4
|
|
5
5
|
module ClassMethods
|
6
|
-
def
|
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
|
-
|
9
|
-
|
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
|
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
|
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
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
22
|
-
|
23
|
-
|
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(
|
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
|
47
|
+
if defer
|
37
48
|
set_cookie("ahoy_track", true, nil, false)
|
38
49
|
else
|
39
|
-
|
50
|
+
delete_cookie("ahoy_track")
|
40
51
|
|
41
|
-
|
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(
|
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.
|
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
|
67
|
-
@
|
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",
|
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",
|
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
|
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
|
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.
|
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
|
-
|
149
|
-
|
150
|
-
|
151
|
-
|
152
|
-
|
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
|
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
|
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
|
-
|
286
|
+
Ahoy.log message
|
228
287
|
end
|
229
288
|
end
|
230
289
|
end
|