traffic_channelizer 1.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- checksums.yaml +7 -0
- data/.gitignore +9 -0
- data/.rspec +2 -0
- data/.travis.yml +3 -0
- data/Gemfile +4 -0
- data/README.md +55 -0
- data/Rakefile +1 -0
- data/bin/console +7 -0
- data/bin/setup +7 -0
- data/lib/traffic_channelizer.rb +14 -0
- data/lib/traffic_channelizer/channel_group.rb +18 -0
- data/lib/traffic_channelizer/landing_page.rb +38 -0
- data/lib/traffic_channelizer/referrer.rb +55 -0
- data/lib/traffic_channelizer/version.rb +3 -0
- data/lib/traffic_channelizer/visit.rb +57 -0
- data/traffic_channelizer.gemspec +30 -0
- metadata +161 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA1:
|
3
|
+
metadata.gz: 6e209cd5064bf38df0b29712c13814fc060305e2
|
4
|
+
data.tar.gz: 359b711366ecc0be0d1c8da126adb3b7f852e6cf
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: f4ec19f2539aeb901e3cd462a4108cec72a995c6dbb284ad537c55d77fbc0c04a5235dd8341491ebf7e065ddb6ee07b72f50bad35ed8677a85e4dc5d8691ef62
|
7
|
+
data.tar.gz: 97edaba49ac1e050bf31c1903f4870c353be5129520be7ffb3f76dbc97c7d2240ea375cc4ed9877504e67ae2d1faf6e039d595f4f0c9e2a93b525a82f1e9694d
|
data/.gitignore
ADDED
data/.rspec
ADDED
data/.travis.yml
ADDED
data/Gemfile
ADDED
data/README.md
ADDED
@@ -0,0 +1,55 @@
|
|
1
|
+
# TrafficChannelizer
|
2
|
+
|
3
|
+
Given a referrer_url and destination landing_page_url, this gem will determine the
|
4
|
+
marketing attribution for the visit (source, medium, campaign) and channel group
|
5
|
+
(eg Social, Direct, Organic Search, Paid Search).
|
6
|
+
|
7
|
+
The categorizations are based on Google Analytics' [default channel definitions](https://support.google.com/analytics/answer/3297892)
|
8
|
+
|
9
|
+
## Limitations
|
10
|
+
|
11
|
+
1. Display and Paid Search groupings do not take into account the Ad Distribution Network
|
12
|
+
2. The list of Social Networks and Search Engines may differ from Google's lists.
|
13
|
+
|
14
|
+
## Usage
|
15
|
+
|
16
|
+
```Ruby
|
17
|
+
TrafficChannelizer.analyze(referrer_url: 'https://www.google.com?s=some-query', landing_page_url: 'https://your-site.com')
|
18
|
+
=> {:referrer_domain=>"google.com", :landing_page_domain=>"your-site.com", :medium=>"search", :source=>"Google", :term=>nil, :campaign=>nil, :content=>nil, :channel_group=>"Organic Search"}
|
19
|
+
|
20
|
+
TrafficChannelizer.analyze(referrer_url: 'https://www.some-site.com', landing_page_url: 'https://your-site.com?utm_medium=cpc')
|
21
|
+
=> {:referrer_domain=>"some-site.com", :landing_page_domain=>"your-site.com", :medium=>"cpc", :source=>"some-site.com", :term=>nil, :campaign=>nil, :content=>nil, :channel_group=>"Paid Search"}
|
22
|
+
|
23
|
+
TrafficChannelizer.analyze(referrer_url: nil, landing_page_url: 'https://your-site.com')
|
24
|
+
=> {:referrer_domain=>nil, :landing_page_domain=>"your-site.com", :medium=>"(none)", :source=>"(direct)", :term=>nil, :campaign=>nil, :content=>nil, :channel_group=>"Direct"}
|
25
|
+
```
|
26
|
+
|
27
|
+
## Installation
|
28
|
+
|
29
|
+
Add this line to your application's Gemfile:
|
30
|
+
|
31
|
+
```ruby
|
32
|
+
gem 'traffic_channelizer'
|
33
|
+
```
|
34
|
+
|
35
|
+
And then execute:
|
36
|
+
|
37
|
+
$ bundle
|
38
|
+
|
39
|
+
Or install it yourself as:
|
40
|
+
|
41
|
+
$ gem install traffic_channelizer
|
42
|
+
|
43
|
+
## Development
|
44
|
+
|
45
|
+
After checking out the repo, run `bin/setup` to install dependencies. Then, run `bin/console` for an interactive prompt that will allow you to experiment.
|
46
|
+
|
47
|
+
To install this gem onto your local machine, run `bundle exec rake install`. To release a new version, update the version number in `version.rb`, and then run `bundle exec rake release` to create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
|
48
|
+
|
49
|
+
## Contributing
|
50
|
+
|
51
|
+
1. Fork it ( https://github.com/[my-github-username]/traffic_channelizer/fork )
|
52
|
+
2. Create your feature branch (`git checkout -b my-new-feature`)
|
53
|
+
3. Commit your changes (`git commit -am 'Add some feature'`)
|
54
|
+
4. Push to the branch (`git push origin my-new-feature`)
|
55
|
+
5. Create a new Pull Request
|
data/Rakefile
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
require 'bundler/gem_tasks'
|
data/bin/console
ADDED
data/bin/setup
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
require 'traffic_channelizer/version'
|
2
|
+
require 'traffic_channelizer/referrer'
|
3
|
+
require 'traffic_channelizer/landing_page'
|
4
|
+
require 'traffic_channelizer/channel_group'
|
5
|
+
require 'traffic_channelizer/visit'
|
6
|
+
|
7
|
+
module TrafficChannelizer
|
8
|
+
def self.analyze(referrer_url:, landing_page_url:)
|
9
|
+
Visit.new(
|
10
|
+
referrer_url: referrer_url,
|
11
|
+
landing_page_url: landing_page_url
|
12
|
+
).attribution_data
|
13
|
+
end
|
14
|
+
end
|
@@ -0,0 +1,18 @@
|
|
1
|
+
module TrafficChannelizer
|
2
|
+
class ChannelGroup
|
3
|
+
def self.identify(source:, medium:)
|
4
|
+
return 'Email' if medium =~ /^email$/i
|
5
|
+
return 'Affiliates' if medium =~ /^affiliate$/i
|
6
|
+
return 'Referral' if medium =~ /^referral$/i
|
7
|
+
return 'Organic Search' if medium =~ /^(organic|search)$/i
|
8
|
+
return 'Paid Search' if medium =~ /^(cpc|ppc|paidsearch)$/i
|
9
|
+
return 'Other Advertising' if medium =~ /^(cpv|cpa|cpp)$/i
|
10
|
+
return 'Display' if medium =~ /^(display|cpm|banner)$/i
|
11
|
+
return 'Social' if medium =~ /^(social|social-network|social-media|sm|social network|social media)$/i
|
12
|
+
return 'Other' unless medium == '(none)'
|
13
|
+
return 'Direct' if source == '(direct)'
|
14
|
+
return 'Internal' if source == '(internal)'
|
15
|
+
'Referral'
|
16
|
+
end
|
17
|
+
end
|
18
|
+
end
|
@@ -0,0 +1,38 @@
|
|
1
|
+
require 'addressable/uri'
|
2
|
+
|
3
|
+
module TrafficChannelizer
|
4
|
+
class LandingPage
|
5
|
+
attr_reader :url
|
6
|
+
|
7
|
+
def initialize(url)
|
8
|
+
@url = url
|
9
|
+
end
|
10
|
+
|
11
|
+
%w(utm_source utm_medium utm_campaign utm_term utm_content).each do |utm_param|
|
12
|
+
define_method utm_param do
|
13
|
+
query_params[utm_param] unless query_params[utm_param] == ''
|
14
|
+
end
|
15
|
+
end
|
16
|
+
|
17
|
+
def domain
|
18
|
+
domain_without_www
|
19
|
+
end
|
20
|
+
|
21
|
+
private
|
22
|
+
|
23
|
+
def domain_without_www
|
24
|
+
return unless uri.host
|
25
|
+
uri.host.gsub(/^www\./, '')
|
26
|
+
end
|
27
|
+
|
28
|
+
def uri
|
29
|
+
@uri ||= Addressable::URI.parse(url.to_s)
|
30
|
+
rescue
|
31
|
+
nil
|
32
|
+
end
|
33
|
+
|
34
|
+
def query_params
|
35
|
+
@query_params ||= (uri && uri.query_values) || {}
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
@@ -0,0 +1,55 @@
|
|
1
|
+
require 'referer-parser'
|
2
|
+
require 'addressable/uri'
|
3
|
+
|
4
|
+
module TrafficChannelizer
|
5
|
+
class Referrer
|
6
|
+
attr_reader :url
|
7
|
+
|
8
|
+
def self.parser
|
9
|
+
@parser ||= RefererParser::Parser.new
|
10
|
+
end
|
11
|
+
|
12
|
+
class << self
|
13
|
+
attr_writer :parser
|
14
|
+
end
|
15
|
+
|
16
|
+
def initialize(url)
|
17
|
+
@url = url
|
18
|
+
end
|
19
|
+
|
20
|
+
def medium
|
21
|
+
referrer_data[:medium]
|
22
|
+
end
|
23
|
+
|
24
|
+
def source
|
25
|
+
referrer_data[:source] || domain_without_www
|
26
|
+
end
|
27
|
+
|
28
|
+
def domain
|
29
|
+
domain_without_www
|
30
|
+
end
|
31
|
+
|
32
|
+
def term
|
33
|
+
referrer_data[:term]
|
34
|
+
end
|
35
|
+
|
36
|
+
private
|
37
|
+
|
38
|
+
def domain_without_www
|
39
|
+
return unless normalized_uri.host
|
40
|
+
normalized_uri.host.gsub(/^www\./, '')
|
41
|
+
end
|
42
|
+
|
43
|
+
def normalized_uri
|
44
|
+
@normalized_uri ||= Addressable::URI.parse(url.to_s).normalize
|
45
|
+
rescue
|
46
|
+
nil
|
47
|
+
end
|
48
|
+
|
49
|
+
def referrer_data
|
50
|
+
@referrer_data ||= self.class.parser.parse(normalized_uri.to_s)
|
51
|
+
rescue RefererParser::InvalidUriError
|
52
|
+
{}
|
53
|
+
end
|
54
|
+
end
|
55
|
+
end
|
@@ -0,0 +1,57 @@
|
|
1
|
+
module TrafficChannelizer
|
2
|
+
class Visit
|
3
|
+
def initialize(referrer_url:, landing_page_url:)
|
4
|
+
@referrer_url = referrer_url
|
5
|
+
@landing_page_url = landing_page_url
|
6
|
+
end
|
7
|
+
|
8
|
+
def attribution_data
|
9
|
+
{
|
10
|
+
referrer_domain: referrer.domain,
|
11
|
+
landing_page_domain: landing_page.domain,
|
12
|
+
medium: medium,
|
13
|
+
source: source,
|
14
|
+
term: term,
|
15
|
+
campaign: landing_page.utm_campaign,
|
16
|
+
content: landing_page.utm_content,
|
17
|
+
channel_group: channel_group
|
18
|
+
}
|
19
|
+
end
|
20
|
+
|
21
|
+
def medium
|
22
|
+
landing_page.utm_medium || referrer.medium || '(none)'
|
23
|
+
end
|
24
|
+
|
25
|
+
def source
|
26
|
+
s = landing_page.utm_source
|
27
|
+
s ||= '(internal)' if internal?
|
28
|
+
s ||= referrer.source
|
29
|
+
s ||= '(direct)'
|
30
|
+
s
|
31
|
+
end
|
32
|
+
|
33
|
+
def term
|
34
|
+
landing_page.utm_term || referrer.term
|
35
|
+
end
|
36
|
+
|
37
|
+
def channel_group
|
38
|
+
ChannelGroup.identify(medium: medium, source: source)
|
39
|
+
end
|
40
|
+
|
41
|
+
private
|
42
|
+
|
43
|
+
def internal?
|
44
|
+
landing_page.domain == referrer.domain
|
45
|
+
end
|
46
|
+
|
47
|
+
def landing_page
|
48
|
+
@landing_page ||= LandingPage.new(landing_page_url)
|
49
|
+
end
|
50
|
+
|
51
|
+
def referrer
|
52
|
+
@referrer ||= Referrer.new(referrer_url)
|
53
|
+
end
|
54
|
+
|
55
|
+
attr_reader :referrer_url, :landing_page_url
|
56
|
+
end
|
57
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
# coding: utf-8
|
2
|
+
lib = File.expand_path('../lib', __FILE__)
|
3
|
+
$LOAD_PATH.unshift(lib) unless $LOAD_PATH.include?(lib)
|
4
|
+
require 'traffic_channelizer/version'
|
5
|
+
|
6
|
+
Gem::Specification.new do |spec|
|
7
|
+
spec.name = 'traffic_channelizer'
|
8
|
+
spec.version = TrafficChannelizer::VERSION
|
9
|
+
spec.authors = ['Ian Yamey']
|
10
|
+
spec.email = ['ian@policygenius.com']
|
11
|
+
|
12
|
+
spec.summary = 'Analyzes website visits to determine the source, medium, campaign and channel grouping, similar to the classifications used by Google Analytics'
|
13
|
+
spec.description = "Given a referrer_url and destination landing_page_url, this gem will determine the traffic's " \
|
14
|
+
'source, medium, campaign, term, content and channel (eg Social, Direct, Organic Search, Paid Search). The categorization is ' \
|
15
|
+
"based on Google Analytics' default channel definitions (see https://support.google.com/analytics/answer/3297892)"
|
16
|
+
spec.homepage = 'https://github.com/ianyamey/traffic_channelizer'
|
17
|
+
spec.license = 'MIT'
|
18
|
+
spec.files = `git ls-files -z`.split("\x0").reject { |f| f.match(%r{^(test|spec|features)/}) }
|
19
|
+
spec.bindir = 'exe'
|
20
|
+
spec.executables = spec.files.grep(%r{^exe/}) { |f| File.basename(f) }
|
21
|
+
spec.require_paths = ['lib']
|
22
|
+
|
23
|
+
spec.add_dependency 'rack'
|
24
|
+
spec.add_dependency 'referer-parser'
|
25
|
+
spec.add_dependency 'addressable'
|
26
|
+
spec.add_development_dependency 'rspec', '~> 3.2'
|
27
|
+
spec.add_development_dependency 'bundler', '~> 1.9'
|
28
|
+
spec.add_development_dependency 'rake', '~> 10.0'
|
29
|
+
spec.add_development_dependency 'pry'
|
30
|
+
end
|
metadata
ADDED
@@ -0,0 +1,161 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: traffic_channelizer
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 1.0.0
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Ian Yamey
|
8
|
+
autorequire:
|
9
|
+
bindir: exe
|
10
|
+
cert_chain: []
|
11
|
+
date: 2016-05-04 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: rack
|
15
|
+
requirement: !ruby/object:Gem::Requirement
|
16
|
+
requirements:
|
17
|
+
- - ">="
|
18
|
+
- !ruby/object:Gem::Version
|
19
|
+
version: '0'
|
20
|
+
type: :runtime
|
21
|
+
prerelease: false
|
22
|
+
version_requirements: !ruby/object:Gem::Requirement
|
23
|
+
requirements:
|
24
|
+
- - ">="
|
25
|
+
- !ruby/object:Gem::Version
|
26
|
+
version: '0'
|
27
|
+
- !ruby/object:Gem::Dependency
|
28
|
+
name: referer-parser
|
29
|
+
requirement: !ruby/object:Gem::Requirement
|
30
|
+
requirements:
|
31
|
+
- - ">="
|
32
|
+
- !ruby/object:Gem::Version
|
33
|
+
version: '0'
|
34
|
+
type: :runtime
|
35
|
+
prerelease: false
|
36
|
+
version_requirements: !ruby/object:Gem::Requirement
|
37
|
+
requirements:
|
38
|
+
- - ">="
|
39
|
+
- !ruby/object:Gem::Version
|
40
|
+
version: '0'
|
41
|
+
- !ruby/object:Gem::Dependency
|
42
|
+
name: addressable
|
43
|
+
requirement: !ruby/object:Gem::Requirement
|
44
|
+
requirements:
|
45
|
+
- - ">="
|
46
|
+
- !ruby/object:Gem::Version
|
47
|
+
version: '0'
|
48
|
+
type: :runtime
|
49
|
+
prerelease: false
|
50
|
+
version_requirements: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - ">="
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: '0'
|
55
|
+
- !ruby/object:Gem::Dependency
|
56
|
+
name: rspec
|
57
|
+
requirement: !ruby/object:Gem::Requirement
|
58
|
+
requirements:
|
59
|
+
- - "~>"
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '3.2'
|
62
|
+
type: :development
|
63
|
+
prerelease: false
|
64
|
+
version_requirements: !ruby/object:Gem::Requirement
|
65
|
+
requirements:
|
66
|
+
- - "~>"
|
67
|
+
- !ruby/object:Gem::Version
|
68
|
+
version: '3.2'
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
name: bundler
|
71
|
+
requirement: !ruby/object:Gem::Requirement
|
72
|
+
requirements:
|
73
|
+
- - "~>"
|
74
|
+
- !ruby/object:Gem::Version
|
75
|
+
version: '1.9'
|
76
|
+
type: :development
|
77
|
+
prerelease: false
|
78
|
+
version_requirements: !ruby/object:Gem::Requirement
|
79
|
+
requirements:
|
80
|
+
- - "~>"
|
81
|
+
- !ruby/object:Gem::Version
|
82
|
+
version: '1.9'
|
83
|
+
- !ruby/object:Gem::Dependency
|
84
|
+
name: rake
|
85
|
+
requirement: !ruby/object:Gem::Requirement
|
86
|
+
requirements:
|
87
|
+
- - "~>"
|
88
|
+
- !ruby/object:Gem::Version
|
89
|
+
version: '10.0'
|
90
|
+
type: :development
|
91
|
+
prerelease: false
|
92
|
+
version_requirements: !ruby/object:Gem::Requirement
|
93
|
+
requirements:
|
94
|
+
- - "~>"
|
95
|
+
- !ruby/object:Gem::Version
|
96
|
+
version: '10.0'
|
97
|
+
- !ruby/object:Gem::Dependency
|
98
|
+
name: pry
|
99
|
+
requirement: !ruby/object:Gem::Requirement
|
100
|
+
requirements:
|
101
|
+
- - ">="
|
102
|
+
- !ruby/object:Gem::Version
|
103
|
+
version: '0'
|
104
|
+
type: :development
|
105
|
+
prerelease: false
|
106
|
+
version_requirements: !ruby/object:Gem::Requirement
|
107
|
+
requirements:
|
108
|
+
- - ">="
|
109
|
+
- !ruby/object:Gem::Version
|
110
|
+
version: '0'
|
111
|
+
description: Given a referrer_url and destination landing_page_url, this gem will
|
112
|
+
determine the traffic's source, medium, campaign, term, content and channel (eg
|
113
|
+
Social, Direct, Organic Search, Paid Search). The categorization is based on Google
|
114
|
+
Analytics' default channel definitions (see https://support.google.com/analytics/answer/3297892)
|
115
|
+
email:
|
116
|
+
- ian@policygenius.com
|
117
|
+
executables: []
|
118
|
+
extensions: []
|
119
|
+
extra_rdoc_files: []
|
120
|
+
files:
|
121
|
+
- ".gitignore"
|
122
|
+
- ".rspec"
|
123
|
+
- ".travis.yml"
|
124
|
+
- Gemfile
|
125
|
+
- README.md
|
126
|
+
- Rakefile
|
127
|
+
- bin/console
|
128
|
+
- bin/setup
|
129
|
+
- lib/traffic_channelizer.rb
|
130
|
+
- lib/traffic_channelizer/channel_group.rb
|
131
|
+
- lib/traffic_channelizer/landing_page.rb
|
132
|
+
- lib/traffic_channelizer/referrer.rb
|
133
|
+
- lib/traffic_channelizer/version.rb
|
134
|
+
- lib/traffic_channelizer/visit.rb
|
135
|
+
- traffic_channelizer.gemspec
|
136
|
+
homepage: https://github.com/ianyamey/traffic_channelizer
|
137
|
+
licenses:
|
138
|
+
- MIT
|
139
|
+
metadata: {}
|
140
|
+
post_install_message:
|
141
|
+
rdoc_options: []
|
142
|
+
require_paths:
|
143
|
+
- lib
|
144
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
145
|
+
requirements:
|
146
|
+
- - ">="
|
147
|
+
- !ruby/object:Gem::Version
|
148
|
+
version: '0'
|
149
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
150
|
+
requirements:
|
151
|
+
- - ">="
|
152
|
+
- !ruby/object:Gem::Version
|
153
|
+
version: '0'
|
154
|
+
requirements: []
|
155
|
+
rubyforge_project:
|
156
|
+
rubygems_version: 2.4.6
|
157
|
+
signing_key:
|
158
|
+
specification_version: 4
|
159
|
+
summary: Analyzes website visits to determine the source, medium, campaign and channel
|
160
|
+
grouping, similar to the classifications used by Google Analytics
|
161
|
+
test_files: []
|