browser 5.3.1 → 6.2.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.
Files changed (57) hide show
  1. checksums.yaml +4 -4
  2. data/{FUNDING.yml → .github/FUNDING.yml} +2 -1
  3. data/.github/workflows/{tests.yml → ruby-tests.yml} +17 -17
  4. data/.rubocop.yml +3 -23
  5. data/CHANGELOG.md +23 -7
  6. data/LICENSE.md +20 -0
  7. data/README.md +12 -4
  8. data/bots.yml +23 -0
  9. data/browser.gemspec +5 -5
  10. data/gemfiles/{rails6_0.gemfile → 7_0.gemfile} +1 -1
  11. data/gemfiles/{rails6_1.gemfile → 7_1.gemfile} +1 -1
  12. data/lib/browser/accept_language.rb +1 -1
  13. data/lib/browser/base.rb +39 -29
  14. data/lib/browser/browser.rb +25 -20
  15. data/lib/browser/chrome.rb +4 -0
  16. data/lib/browser/device.rb +10 -10
  17. data/lib/browser/edge.rb +1 -1
  18. data/lib/browser/epiphany.rb +21 -0
  19. data/lib/browser/meta/base.rb +1 -1
  20. data/lib/browser/platform/ios.rb +1 -1
  21. data/lib/browser/platform.rb +1 -1
  22. data/lib/browser/safari.rb +1 -0
  23. data/lib/browser/testing.rb +3 -3
  24. data/lib/browser/version.rb +1 -1
  25. data/lib/browser.rb +1 -1
  26. data/search_engines.yml +1 -0
  27. data/test/browser_test.rb +3 -3
  28. data/test/rails_test.rb +1 -1
  29. data/test/sample_app.rb +2 -2
  30. data/test/test_helper.rb +4 -2
  31. data/test/ua.yml +11 -4
  32. data/test/ua_bots.yml +18 -0
  33. data/test/unit/accept_language_test.rb +1 -1
  34. data/test/unit/android_app_test.rb +3 -3
  35. data/test/unit/android_test.rb +17 -17
  36. data/test/unit/blackberry_test.rb +1 -1
  37. data/test/unit/bots_test.rb +15 -4
  38. data/test/unit/chrome_test.rb +7 -0
  39. data/test/unit/device_test.rb +25 -25
  40. data/test/unit/edge_test.rb +6 -0
  41. data/test/unit/electron_test.rb +1 -1
  42. data/test/unit/epiphany_test.rb +21 -0
  43. data/test/unit/facebook_test.rb +3 -3
  44. data/test/unit/generic_test.rb +3 -3
  45. data/test/unit/instagram_test.rb +2 -2
  46. data/test/unit/internet_explorer_test.rb +18 -18
  47. data/test/unit/ios_app_test.rb +3 -3
  48. data/test/unit/ios_test.rb +18 -12
  49. data/test/unit/kindle_test.rb +15 -0
  50. data/test/unit/meta_test.rb +13 -13
  51. data/test/unit/platform_test.rb +33 -25
  52. data/test/unit/proxy_test.rb +2 -2
  53. data/test/unit/safari_test.rb +14 -9
  54. data/test/unit/snapchat_test.rb +3 -3
  55. data/test/unit/uc_browser_test.rb +1 -1
  56. data/test/unit/windows_test.rb +11 -11
  57. metadata +15 -77
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 837e072e8544da56ee51cf187b35850444d17c931c535115cf50c679d2227a7c
4
- data.tar.gz: b789e49dbd45164c42ff5a25c5bd5a5d0c54a32ac945bcbf97ee41682ca23693
3
+ metadata.gz: cc6ec7dfcb388ec7584be57709ff9c01fbef5cc8ce65cab9810a27d42adafd84
4
+ data.tar.gz: 34979727435f7b5a8e6fb65c758f144925c167abc192abcdefc939353eee9ed6
5
5
  SHA512:
6
- metadata.gz: 4adb59b646efa17033f385ad0b935f067c9b67a8718a1362f0c5ef49a3c28decd064ec788739c4d38f9969a2189ffc7090ae5d62d3979c249e3237f6b547bfe1
7
- data.tar.gz: c0712d54d08e48522b26ce1c162a238711f20bced68b578e300657ec25ff313d9637a2a881dd2797cd8295b90d100a60789364b17cb9b008e4e3da092e071d0a
6
+ metadata.gz: 4859b7dcca0706511efeaa14a5eccda8c3d8754aa5223fcb1b930d93436c8cdf547f8511ce0b9e519109626c56b0b3b22f2fc527964c537500e463eead5a8709
7
+ data.tar.gz: 5f37c58194e893bb10a5d6ad95ec0f106e9d6b7f0a1db4754d69844a7d0d560620bf876199be511b75e460a0dad8227434b785e4718d84d09f6d3e78108df398
@@ -1,3 +1,4 @@
1
1
  # These are supported funding model platforms
2
-
2
+ ---
3
3
  github: [fnando]
4
+ custom: ["https://paypal.me/nandovieira/🍕"]
@@ -1,48 +1,47 @@
1
- name: Tests
1
+ ---
2
+ name: ruby-tests
2
3
 
3
4
  on:
4
- pull_request:
5
- branches:
6
- - main
5
+ pull_request_target:
7
6
  push:
8
7
  branches:
9
8
  - main
10
-
11
- schedule:
12
- - cron: "0 10 * * *"
9
+ workflow_dispatch:
10
+ inputs: {}
13
11
 
14
12
  jobs:
15
13
  build:
16
14
  name: Tests with Ruby ${{ matrix.ruby }} and ${{ matrix.gemfile }}
17
15
  runs-on: "ubuntu-latest"
16
+ if: |
17
+ github.actor == 'dependabot[bot]' && github.event_name == 'pull_request_target' ||
18
+ github.actor != 'dependabot[bot]'
18
19
  strategy:
19
20
  fail-fast: false
20
21
  matrix:
21
- ruby: ["3.0.x", "2.7.x", "2.6.x", "2.5.x"]
22
+ ruby: ["3.2", "3.3"]
22
23
  gemfile:
23
24
  - Gemfile
24
- - gemfiles/rails6_1.gemfile
25
- - gemfiles/rails6_0.gemfile
25
+ - gemfiles/7_1.gemfile
26
+ - gemfiles/7_0.gemfile
26
27
 
27
28
  steps:
28
- - uses: actions/checkout@v1
29
+ - uses: actions/checkout@v2.4.0
29
30
 
30
31
  - uses: actions/cache@v2
31
32
  with:
32
33
  path: vendor/bundle
33
34
  key: >
34
- ${{ runner.os }}-${{ matrix.ruby }}-gems-${{ hashFiles(matrix.gemfile) }}
35
- restore-keys: >
36
- ${{ runner.os }}-${{ matrix.ruby }}-gems-${{ hashFiles(matrix.gemfile) }}
35
+ ${{ runner.os }}-${{ matrix.ruby }}-gems-${{
36
+ hashFiles(matrix.gemfile) }}
37
37
 
38
38
  - name: Set up Ruby
39
- uses: actions/setup-ruby@v1
39
+ uses: ruby/setup-ruby@v1
40
40
  with:
41
41
  ruby-version: ${{ matrix.ruby }}
42
42
 
43
43
  - name: Install gem dependencies
44
44
  env:
45
- RAILS_ENV: test
46
45
  BUNDLE_GEMFILE: ${{ matrix.gemfile }}
47
46
  run: |
48
47
  gem install bundler
@@ -51,7 +50,8 @@ jobs:
51
50
 
52
51
  - name: Run Tests
53
52
  env:
54
- RAILS_ENV: test
53
+ PGHOST: localhost
54
+ PGUSER: postgres
55
55
  BUNDLE_GEMFILE: ${{ matrix.gemfile }}
56
56
  run: |
57
57
  bundle exec rake
data/.rubocop.yml CHANGED
@@ -4,6 +4,7 @@ inherit_gem:
4
4
 
5
5
  AllCops:
6
6
  TargetRubyVersion: 2.5
7
+ NewCops: enable
7
8
  Exclude:
8
9
  - bin/**/*
9
10
  - gemfiles/**/*
@@ -13,29 +14,8 @@ AllCops:
13
14
  - "*.gemspec"
14
15
  - config.ru
15
16
 
16
- Metrics/ClassLength:
17
- Enabled: false
18
-
19
- Layout/LineLength:
20
- Max: 80
21
-
22
- Metrics/MethodLength:
23
- Enabled: false
24
-
25
- Style/Alias:
26
- EnforcedStyle: prefer_alias_method
27
-
28
- Lint/RedundantCopDisableDirective:
29
- Enabled: false
30
-
31
- Metrics/AbcSize:
32
- Enabled: false
33
-
34
- Metrics/CyclomaticComplexity:
35
- Enabled: false
36
-
37
- Metrics/PerceivedComplexity:
17
+ Naming/VariableNumber:
38
18
  Enabled: false
39
19
 
40
- Naming/VariableNumber:
20
+ Minitest/EmptyLineBeforeAssertionMethods:
41
21
  Enabled: false
data/CHANGELOG.md CHANGED
@@ -1,5 +1,23 @@
1
1
  # Changelog
2
2
 
3
+ ## 6.2.0
4
+
5
+ - Add `Browser::Base#epiphany?`.
6
+ - Don't ever match linux-based browsers as Safari.
7
+
8
+ ## 6.1.0
9
+
10
+ - Add more bots (mostly related to AI crawlers)
11
+ - Add AppSignal bot
12
+ - Add Checkly bot
13
+ - Detect Chrome when requesting desktop site.
14
+
15
+ ## 6.0.0
16
+
17
+ - Add `Browser::Base#chromium_based?`.
18
+ - Change how browser detection works to avoid re-instantiating classes.
19
+ - Bump up minimum required ruby version to 3.2.0.
20
+
3
21
  ## 5.3.1
4
22
 
5
23
  - Remove Stripe webhooks from bot list.
@@ -90,13 +108,11 @@
90
108
  - You can now define new bot matchers by adding a callable object to
91
109
  `Browser::Bot.matchers`.
92
110
  - Fix `browser.yandex?` and `browser.sputnik?`.
93
- - [BREAKING CHANGE] Removed methods to enable the bot's empty user agent
94
- detection (`Browser::Bot.detect_empty_ua!` and
95
- `Browser::Bot.detect_empty_ua?`).
96
- - [BREAKING CHANGE] Bot detection is now more aggressive by default. It matches
97
- empty user agents, anything that matches
98
- `crawl|fetch|search|monitoring|spider|bot`, and anything listed under
99
- https://github.com/fnando/browser/blob/master/bots.yml.
111
+ - [BREAKING CHANGE] Removed methods to enable the bot's empty user agent detection
112
+ (`Browser::Bot.detect_empty_ua!` and `Browser::Bot.detect_empty_ua?`).
113
+ - [BREAKING CHANGE] Bot detection is now more aggressive by default. It matches empty
114
+ user agents, anything that matches `crawl|fetch|search|monitoring|spider|bot`,
115
+ and anything listed under https://github.com/fnando/browser/blob/master/bots.yml.
100
116
  - Add Jaunt to the bot list.
101
117
 
102
118
  ## 2.7.1
data/LICENSE.md ADDED
@@ -0,0 +1,20 @@
1
+ # The MIT License (MIT)
2
+
3
+ Copyright (c) 2010 Nando Vieira
4
+
5
+ Permission is hereby granted, free of charge, to any person obtaining a copy of
6
+ this software and associated documentation files (the "Software"), to deal in
7
+ the Software without restriction, including without limitation the rights to
8
+ use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
9
+ the Software, and to permit persons to whom the Software is furnished to do so,
10
+ subject to the following conditions:
11
+
12
+ The above copyright notice and this permission notice shall be included in all
13
+ copies or substantial portions of the Software.
14
+
15
+ THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
16
+ IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
17
+ FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
18
+ COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
19
+ IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
20
+ CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
data/README.md CHANGED
@@ -1,7 +1,6 @@
1
1
  # Browser
2
2
 
3
- [![Tests](https://github.com/fnando/browser/workflows/Tests/badge.svg)](https://github.com/fnando/browser)
4
- [![Code Climate](https://codeclimate.com/github/fnando/browser/badges/gpa.svg)](https://codeclimate.com/github/fnando/browser)
3
+ [![Tests](https://github.com/fnando/browser/workflows/ruby-tests/badge.svg)](https://github.com/fnando/browser)
5
4
  [![Gem](https://img.shields.io/gem/v/browser.svg)](https://rubygems.org/gems/browser)
6
5
  [![Gem](https://img.shields.io/gem/dt/browser.svg)](https://rubygems.org/gems/browser)
7
6
 
@@ -23,6 +22,7 @@ browser = Browser.new("Some User Agent", accept_language: "en-us")
23
22
  # General info
24
23
  browser.bot?
25
24
  browser.chrome?
25
+ browser.chromium_based?
26
26
  browser.core_media?
27
27
  browser.duck_duck_go?
28
28
  browser.edge? # Newest MS browser
@@ -53,9 +53,9 @@ browser.yandex?
53
53
  browser.wechat?
54
54
  browser.qq?
55
55
  browser.weibo?
56
- browser.yandex?
57
56
  browser.sputnik?
58
57
  browser.sougou_browser?
58
+ browser.epiphany?
59
59
 
60
60
  # Get bot info
61
61
  browser.bot.name
@@ -323,6 +323,14 @@ Browser::Bot.matchers.delete(Browser::Bot::KeywordMatcher)
323
323
  Browser::Bot.matchers.delete(Browser::Bot::EmptyUserAgentMatcher)
324
324
  ```
325
325
 
326
+ To extend the bot list, you can manipulate the methods below:
327
+
328
+ ```ruby
329
+ Browser::Bot.bots.merge!(new_bots_hash)
330
+ Browser::Bot.bot_exceptions += new_exceptions
331
+ Browser::Bot.search_engines.merge!(new_search_engines_hash)
332
+ ```
333
+
326
334
  ### Middleware
327
335
 
328
336
  You can use the `Browser::Middleware` to redirect user agents.
@@ -425,7 +433,7 @@ information.
425
433
 
426
434
  ## Maintainer
427
435
 
428
- - Nando Vieira - http://nandovieira.com
436
+ - Nando Vieira - https://nandovieira.com
429
437
 
430
438
  ## Contributors
431
439
 
data/bots.yml CHANGED
@@ -16,6 +16,7 @@ apachebench: ApacheBench (ab)
16
16
  apis-google: APIs-Google
17
17
  appengine-google: Google App Engine
18
18
  applebot: Apple Bot
19
+ appsignal: AppSignal Bot
19
20
  archive.org_bot: Internet Archive (archive.org)
20
21
  archiveteam archivebot: ArchiveTeam ArchiveBot
21
22
  ask jeeves: Ask Jeeves
@@ -38,6 +39,7 @@ buzzbot: Buzzbot
38
39
  buzztalk: buzztalk
39
40
  catchbot: CatchBot (catchbot.com)
40
41
  check_http: Nagios monitor
42
+ checkly: Checkly
41
43
  chrome-lighthouse: Chrome-Lighthouse
42
44
  cipacrawler: CipaCrawler
43
45
  cliqzbot: Cliqzbot
@@ -291,7 +293,28 @@ zoombot: ZoomBot
291
293
  zoominfobot: ZoominfoBot
292
294
  zyborg: Zyborg
293
295
 
296
+ # AI Crawlers
297
+ # https://darkvisitors.com
298
+ amazonbot: Amazon
299
+ anthropic-ai: Anthropic-AI
300
+ applebot: Apple
301
+ bytespider: TikTok
302
+ ccbot: Common Crawl
303
+ chatgpt-user: ChatGPT
304
+ claude-web: Anthropic-AI
305
+ cohere-ai: Cohere
306
+ diffbot: Diffbot
307
+ facebookbot: Facebook
308
+ google-extended: Google
309
+ googleother: Google
310
+ gptbot: ChatGPT
311
+ omgili: Webz.io
312
+ perplexitybot: Perplexity
313
+ webz.io: Webz.io
314
+ youbot: You.com
315
+
294
316
  # Generic lib user agents go here.
317
+ httpie: HTTPie
295
318
  eventmachine httpclient: Ruby http library
296
319
  go 1.1 package http: Go 1.1 package http
297
320
  htmlparser: HTMLParser
data/browser.gemspec CHANGED
@@ -7,27 +7,27 @@ Gem::Specification.new do |s|
7
7
  s.version = Browser::VERSION
8
8
  s.platform = Gem::Platform::RUBY
9
9
  s.authors = ["Nando Vieira"]
10
- s.email = ["fnando.vieira@gmail.com"]
10
+ s.email = ["me@fnando.com"]
11
11
  s.homepage = "https://github.com/fnando/browser"
12
12
  s.summary = "Do some browser detection with Ruby."
13
13
  s.description = s.summary
14
14
  s.license = "MIT"
15
- s.required_ruby_version = ">= 2.5.0"
15
+ s.required_ruby_version = ">= 3.2.0"
16
16
 
17
+ s.metadata["rubygems_mfa_required"] = "true"
17
18
  s.metadata["changelog_uri"] = "https://github.com/fnando/browser/blob/main/CHANGELOG.md"
19
+ s.metadata["funding_uri"] = "https://github.com/sponsors/fnando"
18
20
 
19
21
  s.files = `git ls-files`.split("\n")
20
- s.test_files = `git ls-files -- {test,spec,features}/*`.split("\n")
21
22
  s.executables = `git ls-files -- exe/*`
22
23
  .split("\n")
23
24
  .map {|f| File.basename(f) }
24
25
  s.require_paths = ["lib"]
25
26
 
26
- s.add_development_dependency "bundler", ">= 0"
27
+ s.add_development_dependency "bundler"
27
28
  s.add_development_dependency "minitest"
28
29
  s.add_development_dependency "minitest-autotest"
29
30
  s.add_development_dependency "minitest-utils"
30
- s.add_development_dependency "pry-meta"
31
31
  s.add_development_dependency "rack-test"
32
32
  s.add_development_dependency "rails"
33
33
  s.add_development_dependency "rake"
@@ -3,4 +3,4 @@
3
3
  source "https://rubygems.org"
4
4
  gemspec path: ".."
5
5
 
6
- gem "rails", "~> 6.0"
6
+ gem "rails", "~> 7.0.0"
@@ -3,4 +3,4 @@
3
3
  source "https://rubygems.org"
4
4
  gemspec path: ".."
5
5
 
6
- gem "rails", "~> 6.1"
6
+ gem "rails", "~> 7.1.0"
@@ -55,7 +55,7 @@ module Browser
55
55
 
56
56
  private def quality_value
57
57
  qvalue = part[/;q=([\d.]+)/, 1]
58
- qvalue = /\A0\.0?\z/.match?(qvalue) ? "0.0" : qvalue
58
+ qvalue = "0.0" if /\A0\.0?\z/.match?(qvalue)
59
59
  qvalue = qvalue.squeeze(".") if qvalue
60
60
  qvalue
61
61
  end
data/lib/browser/base.rb CHANGED
@@ -26,7 +26,7 @@ module Browser
26
26
  end
27
27
  end
28
28
 
29
- alias_method :to_a, :meta
29
+ alias to_a meta
30
30
 
31
31
  # Return meta representation as string.
32
32
  def to_s
@@ -59,13 +59,13 @@ module Browser
59
59
 
60
60
  # Detect if browser is Microsoft Internet Explorer.
61
61
  def ie?(expected_version = nil)
62
- InternetExplorer.new(ua).match? &&
62
+ instance_of?(InternetExplorer) &&
63
63
  detect_version?(full_version, expected_version)
64
64
  end
65
65
 
66
66
  # Detect if browser is Microsoft Edge.
67
67
  def edge?(expected_version = nil)
68
- Edge.new(ua).match? && detect_version?(full_version, expected_version)
68
+ instance_of?(Edge) && detect_version?(full_version, expected_version)
69
69
  end
70
70
 
71
71
  def compatibility_view?
@@ -82,32 +82,32 @@ module Browser
82
82
 
83
83
  # Detect if browser is Instagram.
84
84
  def instagram?(expected_version = nil)
85
- Instagram.new(ua).match? &&
85
+ instance_of?(Instagram) &&
86
86
  detect_version?(full_version, expected_version)
87
87
  end
88
88
 
89
89
  # Detect if browser is Snapchat.
90
90
  def snapchat?(expected_version = nil)
91
- Snapchat.new(ua).match? &&
91
+ instance_of?(Snapchat) &&
92
92
  detect_version?(full_version, expected_version)
93
93
  end
94
94
 
95
95
  # Detect if browser if Facebook.
96
96
  def facebook?(expected_version = nil)
97
- Facebook.new(ua).match? &&
97
+ instance_of?(Facebook) &&
98
98
  detect_version?(full_version, expected_version)
99
99
  end
100
100
 
101
101
  # Detect if browser is Otter.
102
102
  def otter?(expected_version = nil)
103
- Otter.new(ua).match? &&
103
+ instance_of?(Otter) &&
104
104
  detect_version?(full_version, expected_version)
105
105
  end
106
106
 
107
107
  # Detect if browser is WebKit-based.
108
108
  def webkit?(expected_version = nil)
109
109
  ua.match?(/AppleWebKit/i) &&
110
- (!edge? || Edge.new(ua).chrome_based?) &&
110
+ (!edge? || chromium_based?) &&
111
111
  detect_version?(webkit_full_version, expected_version)
112
112
  end
113
113
 
@@ -124,13 +124,13 @@ module Browser
124
124
 
125
125
  # Detect if browser is PhantomJS
126
126
  def phantom_js?(expected_version = nil)
127
- PhantomJS.new(ua).match? &&
127
+ instance_of?(PhantomJS) &&
128
128
  detect_version?(full_version, expected_version)
129
129
  end
130
130
 
131
131
  # Detect if browser is Safari.
132
132
  def safari?(expected_version = nil)
133
- Safari.new(ua).match? && detect_version?(full_version, expected_version)
133
+ instance_of?(Safari) && detect_version?(full_version, expected_version)
134
134
  end
135
135
 
136
136
  def safari_webapp_mode?
@@ -139,55 +139,55 @@ module Browser
139
139
 
140
140
  # Detect if browser is Firefox.
141
141
  def firefox?(expected_version = nil)
142
- Firefox.new(ua).match? && detect_version?(full_version, expected_version)
142
+ instance_of?(Firefox) && detect_version?(full_version, expected_version)
143
143
  end
144
144
 
145
145
  # Detect if browser is Chrome.
146
146
  def chrome?(expected_version = nil)
147
- Chrome.new(ua).match? && detect_version?(full_version, expected_version)
147
+ instance_of?(Chrome) && detect_version?(full_version, expected_version)
148
148
  end
149
149
 
150
150
  # Detect if browser is Opera.
151
151
  def opera?(expected_version = nil)
152
- Opera.new(ua).match? && detect_version?(full_version, expected_version)
152
+ instance_of?(Opera) && detect_version?(full_version, expected_version)
153
153
  end
154
154
 
155
155
  # Detect if browser is Sputnik.
156
156
  def sputnik?(expected_version = nil)
157
- Sputnik.new(ua).match? && detect_version?(full_version, expected_version)
157
+ instance_of?(Sputnik) && detect_version?(full_version, expected_version)
158
158
  end
159
159
 
160
160
  # Detect if browser is Yandex.
161
161
  def yandex?(expected_version = nil)
162
- Yandex.new(ua).match? && detect_version?(full_version, expected_version)
162
+ instance_of?(Yandex) && detect_version?(full_version, expected_version)
163
163
  end
164
- alias_method :yandex_browser?, :yandex?
164
+ alias yandex_browser? yandex?
165
165
 
166
166
  # Detect if browser is UCBrowser.
167
167
  def uc_browser?(expected_version = nil)
168
- UCBrowser.new(ua).match? &&
168
+ instance_of?(UCBrowser) &&
169
169
  detect_version?(full_version, expected_version)
170
170
  end
171
171
 
172
172
  # Detect if browser is Nokia S40 Ovi Browser.
173
173
  def nokia?(expected_version = nil)
174
- Nokia.new(ua).match? && detect_version?(full_version, expected_version)
174
+ instance_of?(Nokia) && detect_version?(full_version, expected_version)
175
175
  end
176
176
 
177
177
  # Detect if browser is MicroMessenger.
178
178
  def micro_messenger?(expected_version = nil)
179
- MicroMessenger.new(ua).match? &&
179
+ instance_of?(MicroMessenger) &&
180
180
  detect_version?(full_version, expected_version)
181
181
  end
182
182
 
183
- alias_method :wechat?, :micro_messenger?
183
+ alias wechat? micro_messenger?
184
184
 
185
185
  def weibo?(expected_version = nil)
186
- Weibo.new(ua).match? && detect_version?(full_version, expected_version)
186
+ instance_of?(Weibo) && detect_version?(full_version, expected_version)
187
187
  end
188
188
 
189
189
  def alipay?(expected_version = nil)
190
- Alipay.new(ua).match? && detect_version?(full_version, expected_version)
190
+ instance_of?(Alipay) && detect_version?(full_version, expected_version)
191
191
  end
192
192
 
193
193
  # Detect if browser is Opera Mini.
@@ -210,29 +210,29 @@ module Browser
210
210
 
211
211
  # Detect if browser is Huawei.
212
212
  def huawei_browser?(expected_version = nil)
213
- HuaweiBrowser.new(ua).match? &&
213
+ instance_of?(HuaweiBrowser) &&
214
214
  detect_version?(full_version, expected_version)
215
215
  end
216
216
 
217
217
  # Detect if browser is Xiaomi Miui.
218
218
  def miui_browser?(expected_version = nil)
219
- MiuiBrowser.new(ua).match? &&
219
+ instance_of?(MiuiBrowser) &&
220
220
  detect_version?(full_version, expected_version)
221
221
  end
222
222
 
223
223
  # Detect if browser is Maxthon.
224
224
  def maxthon?(expected_version = nil)
225
- Maxthon.new(ua).match? && detect_version?(full_version, expected_version)
225
+ instance_of?(Maxthon) && detect_version?(full_version, expected_version)
226
226
  end
227
227
 
228
228
  # Detect if browser is QQ.
229
229
  def qq?(expected_version = nil)
230
- QQ.new(ua).match? && detect_version?(full_version, expected_version)
230
+ instance_of?(QQ) && detect_version?(full_version, expected_version)
231
231
  end
232
232
 
233
233
  # Detect if browser is Sougou.
234
234
  def sougou_browser?(expected_version = nil)
235
- SougouBrowser.new(ua).match? &&
235
+ instance_of?(SougouBrowser) &&
236
236
  detect_version?(full_version, expected_version)
237
237
  end
238
238
 
@@ -241,6 +241,11 @@ module Browser
241
241
  ua.include?("GSA") && detect_version?(full_version, expected_version)
242
242
  end
243
243
 
244
+ # Detect if browser is Chromium-based.
245
+ def chromium_based?
246
+ false
247
+ end
248
+
244
249
  def webkit_full_version
245
250
  ua[%r{AppleWebKit/([\d.]+)}, 1] || "0.0"
246
251
  end
@@ -260,12 +265,17 @@ module Browser
260
265
 
261
266
  # Detect if the browser is Electron.
262
267
  def electron?(expected_version = nil)
263
- Electron.new(ua).match? && detect_version?(full_version, expected_version)
268
+ instance_of?(Electron) && detect_version?(full_version, expected_version)
269
+ end
270
+
271
+ # Detect if the browser is Epiphany.
272
+ def epiphany?(expected_version = nil)
273
+ instance_of?(Epiphany) && detect_version?(full_version, expected_version)
264
274
  end
265
275
 
266
276
  private def validate_size(subject, input)
267
277
  actual_bytesize = input.bytesize
268
- size_limit = Browser.public_send("#{subject}_size_limit")
278
+ size_limit = Browser.public_send(:"#{subject}_size_limit")
269
279
 
270
280
  return if actual_bytesize < size_limit
271
281
 
@@ -37,6 +37,7 @@ require_relative "miui_browser"
37
37
  require_relative "maxthon"
38
38
  require_relative "sougou_browser"
39
39
  require_relative "google_search_app"
40
+ require_relative "epiphany"
40
41
 
41
42
  require_relative "bot"
42
43
  require_relative "bot/empty_user_agent_matcher"
@@ -77,23 +78,26 @@ module Browser
77
78
  InternetExplorer,
78
79
  Firefox,
79
80
  Otter,
80
- Facebook, # must be placed before Chrome and Safari
81
- Instagram, # must be placed before Chrome and Safari
82
- Snapchat, # must be placed before Chrome and Safari
83
- Weibo, # must be placed before Chrome and Safari
84
- MicroMessenger, # must be placed before QQ
85
- QQ, # must be placed before Chrome and Safari
86
- Alipay, # must be placed before Chrome and Safari
87
- Electron, # must be placed before Chrome and Safari
88
- Yandex, # must be placed before Chrome and Safari
89
- Sputnik, # must be placed before Chrome and Safari
90
- DuckDuckGo, # must be placed before Chrome and Safari
91
- SamsungBrowser, # must be placed before Chrome and Safari
92
- HuaweiBrowser, # must be placed before Chrome and Safari
93
- MiuiBrowser, # must be placed before Chrome and Safari
94
- Maxthon, # must be placed before Chrome and Safari
95
- SougouBrowser, # must be placed before Chrome and Safari
96
- GoogleSearchApp, # must be placed before Chrome and Safari
81
+ Epiphany,
82
+ Facebook,
83
+ Instagram,
84
+ Snapchat,
85
+ Weibo,
86
+ MicroMessenger, # must be placed before QQ
87
+ QQ,
88
+ Alipay,
89
+ Electron,
90
+ Yandex,
91
+ Sputnik,
92
+ DuckDuckGo,
93
+ SamsungBrowser,
94
+ HuaweiBrowser,
95
+ MiuiBrowser,
96
+ Maxthon,
97
+ SougouBrowser,
98
+ GoogleSearchApp,
99
+
100
+ # Must be last
97
101
  Chrome,
98
102
  Safari,
99
103
  Unknown
@@ -101,8 +105,9 @@ module Browser
101
105
  end
102
106
 
103
107
  def self.new(user_agent, **kwargs)
104
- matchers
105
- .map {|klass| klass.new(user_agent || EMPTY_STRING, **kwargs) }
106
- .find(&:match?)
108
+ matchers.each do |matcher_class|
109
+ matcher = matcher_class.new(user_agent || EMPTY_STRING, **kwargs)
110
+ return matcher if matcher.match?
111
+ end
107
112
  end
108
113
  end
@@ -19,6 +19,10 @@ module Browser
19
19
  "0.0"
20
20
  end
21
21
 
22
+ def chromium_based?
23
+ true
24
+ end
25
+
22
26
  def match?
23
27
  ua.match?(/Chrome|CriOS/) &&
24
28
  !ua.match?(/PhantomJS|FxiOS|ArchiveBot/) &&