emailable 3.0.1
Sign up to get free protection for your applications and to get access to all the features.
- checksums.yaml +7 -0
- data/.gitignore +8 -0
- data/.ruby-gemset +1 -0
- data/.ruby-version +1 -0
- data/.travis.yml +7 -0
- data/Gemfile +4 -0
- data/Gemfile.lock +70 -0
- data/LICENSE.txt +21 -0
- data/README.md +135 -0
- data/Rakefile +10 -0
- data/bin/console +14 -0
- data/bin/setup +8 -0
- data/config/locales/en.yml +10 -0
- data/emailable.gemspec +40 -0
- data/lib/emailable/batch.rb +51 -0
- data/lib/emailable/client.rb +70 -0
- data/lib/emailable/email_validator.rb +70 -0
- data/lib/emailable/resources/account.rb +5 -0
- data/lib/emailable/resources/api_resource.rb +20 -0
- data/lib/emailable/resources/batch_status.rb +10 -0
- data/lib/emailable/resources/verification.rb +15 -0
- data/lib/emailable/version.rb +3 -0
- data/lib/emailable.rb +65 -0
- data/test/email_validator_test.rb +85 -0
- data/test/emailable/batch_test.rb +30 -0
- data/test/emailable_test.rb +71 -0
- data/test/test_helper.rb +12 -0
- metadata +218 -0
checksums.yaml
ADDED
@@ -0,0 +1,7 @@
|
|
1
|
+
---
|
2
|
+
SHA256:
|
3
|
+
metadata.gz: 0e7635cc0f814700d29137678bd7b53fd91f07f042ff5435b8777cc269a51fb6
|
4
|
+
data.tar.gz: 808f71d2c343495ad48143230d1daa11ba59e334d75a242d25df637abd446b51
|
5
|
+
SHA512:
|
6
|
+
metadata.gz: 8a78fd0ab06eca710c67e2e8c01142d0a4bfdfa19f7357c9f54b7a8d53eddaf7b29f32ded7ad85145f414e95782a61dd4774d0a39821f2194f0228b9b402b6c4
|
7
|
+
data.tar.gz: 99f27b66dc848815689fe3a0869b2b47a0f412c3305b910a044488b65598161a8349f690b5772d4e5181e97a4c55fa794f9593d3fa4b60bca5d5ad9a99b84530
|
data/.gitignore
ADDED
data/.ruby-gemset
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
emailable-ruby
|
data/.ruby-version
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
2.6.5
|
data/.travis.yml
ADDED
data/Gemfile
ADDED
data/Gemfile.lock
ADDED
@@ -0,0 +1,70 @@
|
|
1
|
+
PATH
|
2
|
+
remote: .
|
3
|
+
specs:
|
4
|
+
emailable (3.0.0)
|
5
|
+
faraday
|
6
|
+
faraday_middleware
|
7
|
+
net-http-persistent
|
8
|
+
|
9
|
+
GEM
|
10
|
+
remote: https://rubygems.org/
|
11
|
+
specs:
|
12
|
+
activemodel (6.0.3.4)
|
13
|
+
activesupport (= 6.0.3.4)
|
14
|
+
activesupport (6.0.3.4)
|
15
|
+
concurrent-ruby (~> 1.0, >= 1.0.2)
|
16
|
+
i18n (>= 0.7, < 2)
|
17
|
+
minitest (~> 5.1)
|
18
|
+
tzinfo (~> 1.1)
|
19
|
+
zeitwerk (~> 2.2, >= 2.2.2)
|
20
|
+
ansi (1.5.0)
|
21
|
+
awesome_print (1.8.0)
|
22
|
+
builder (3.2.3)
|
23
|
+
coderay (1.1.2)
|
24
|
+
concurrent-ruby (1.1.7)
|
25
|
+
connection_pool (2.2.3)
|
26
|
+
faraday (1.3.0)
|
27
|
+
faraday-net_http (~> 1.0)
|
28
|
+
multipart-post (>= 1.2, < 3)
|
29
|
+
ruby2_keywords
|
30
|
+
faraday-net_http (1.0.1)
|
31
|
+
faraday_middleware (1.0.0)
|
32
|
+
faraday (~> 1.0)
|
33
|
+
i18n (1.8.5)
|
34
|
+
concurrent-ruby (~> 1.0)
|
35
|
+
method_source (0.9.2)
|
36
|
+
minitest (5.11.3)
|
37
|
+
minitest-reporters (1.3.6)
|
38
|
+
ansi
|
39
|
+
builder
|
40
|
+
minitest (>= 5.0)
|
41
|
+
ruby-progressbar
|
42
|
+
multipart-post (2.1.1)
|
43
|
+
net-http-persistent (4.0.1)
|
44
|
+
connection_pool (~> 2.2)
|
45
|
+
pry (0.12.2)
|
46
|
+
coderay (~> 1.1.0)
|
47
|
+
method_source (~> 0.9.0)
|
48
|
+
rake (13.0.1)
|
49
|
+
ruby-progressbar (1.10.1)
|
50
|
+
ruby2_keywords (0.0.4)
|
51
|
+
thread_safe (0.3.6)
|
52
|
+
tzinfo (1.2.8)
|
53
|
+
thread_safe (~> 0.1)
|
54
|
+
zeitwerk (2.4.1)
|
55
|
+
|
56
|
+
PLATFORMS
|
57
|
+
ruby
|
58
|
+
|
59
|
+
DEPENDENCIES
|
60
|
+
activemodel
|
61
|
+
awesome_print
|
62
|
+
bundler
|
63
|
+
emailable!
|
64
|
+
minitest (~> 5.0)
|
65
|
+
minitest-reporters
|
66
|
+
pry
|
67
|
+
rake (~> 13.0)
|
68
|
+
|
69
|
+
BUNDLED WITH
|
70
|
+
1.17.3
|
data/LICENSE.txt
ADDED
@@ -0,0 +1,21 @@
|
|
1
|
+
The MIT License (MIT)
|
2
|
+
|
3
|
+
Copyright (c) 2019 Emailable. https://emailable.com
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
7
|
+
in the Software without restriction, including without limitation the rights
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
10
|
+
furnished to do so, subject to the following conditions:
|
11
|
+
|
12
|
+
The above copyright notice and this permission notice shall be included in
|
13
|
+
all 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,
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN
|
21
|
+
THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,135 @@
|
|
1
|
+
# Emailable Ruby Library
|
2
|
+
|
3
|
+
[![Build Status](https://travis-ci.com/emailable/emailable-ruby.svg)](https://travis-ci.com/emailable/emailable-ruby)
|
4
|
+
[![Maintainability](https://api.codeclimate.com/v1/badges/2d74c69a9155109058a7/maintainability)](https://codeclimate.com/github/emailable/emailable-ruby/maintainability)
|
5
|
+
|
6
|
+
This is the official ruby wrapper for the Emailable API.
|
7
|
+
|
8
|
+
It also includes an Active Record (Rails) validator to verify email attributes.
|
9
|
+
|
10
|
+
## Documentation
|
11
|
+
|
12
|
+
See the [Ruby API docs](https://emailable.com/docs/api/?ruby).
|
13
|
+
|
14
|
+
## Installation
|
15
|
+
|
16
|
+
Add this line to your application's Gemfile:
|
17
|
+
|
18
|
+
```ruby
|
19
|
+
gem 'emailable'
|
20
|
+
```
|
21
|
+
|
22
|
+
And then execute:
|
23
|
+
|
24
|
+
$ bundle
|
25
|
+
|
26
|
+
Or install it yourself as:
|
27
|
+
|
28
|
+
$ gem install emailable
|
29
|
+
|
30
|
+
## Usage
|
31
|
+
|
32
|
+
The library needs to be configured with your account's API key which is available in your [Emailable Dashboard](https://app.emailable.com/api). Set `Emailable.api_key` to its value:
|
33
|
+
|
34
|
+
### Setup
|
35
|
+
|
36
|
+
```ruby
|
37
|
+
require 'emailable'
|
38
|
+
|
39
|
+
# set api key
|
40
|
+
Emailable.api_key = 'live_...'
|
41
|
+
```
|
42
|
+
|
43
|
+
### Verification
|
44
|
+
|
45
|
+
```ruby
|
46
|
+
# verify an email address
|
47
|
+
Emailable.verify('jarrett@emailable.com')
|
48
|
+
```
|
49
|
+
|
50
|
+
#### Slow Email Server Handling
|
51
|
+
|
52
|
+
Some email servers are slow to respond. As a result, the timeout may be reached
|
53
|
+
before we are able to complete the verification process. If this happens, the
|
54
|
+
verification will continue in the background on our servers, and a
|
55
|
+
`Emailable::TimeoutError` will be raised. We recommend sleeping for at least
|
56
|
+
one second and trying your request again. Re-requesting the same verification
|
57
|
+
with the same options will not impact your credit allocation within a 5 minute
|
58
|
+
window. You can test this behavior using a test key and the special
|
59
|
+
email `slow@example.com`.
|
60
|
+
|
61
|
+
### Batch Verification
|
62
|
+
|
63
|
+
#### Start a batch
|
64
|
+
|
65
|
+
```ruby
|
66
|
+
emails = ['jarrett@emailable.com', 'support@emailable.com', ...]
|
67
|
+
batch = Emailable::Batch.new(emails)
|
68
|
+
|
69
|
+
# you can optionally pass in a callback url that we'll POST to when the
|
70
|
+
# batch is complete.
|
71
|
+
batch = Emailable::Batch.new(emails, callback: 'https://emailable.com/')
|
72
|
+
|
73
|
+
# start verifying the batch
|
74
|
+
batch.verify
|
75
|
+
```
|
76
|
+
|
77
|
+
#### Get the status / results of a batch
|
78
|
+
|
79
|
+
Calling `status` on a batch will return the status. It will contain the results as well once complete. You can also `results` to get just the results.
|
80
|
+
|
81
|
+
```ruby
|
82
|
+
id = '5cfcbfdeede34200693c4319'
|
83
|
+
batch = Emailable::Batch.new(id)
|
84
|
+
|
85
|
+
# get status of batch
|
86
|
+
batch.status
|
87
|
+
|
88
|
+
# gets the results
|
89
|
+
batch.status.emails
|
90
|
+
|
91
|
+
# get the counts
|
92
|
+
batch.status.total_counts
|
93
|
+
batch.status.reason_counts
|
94
|
+
|
95
|
+
# returns true / false
|
96
|
+
batch.complete?
|
97
|
+
```
|
98
|
+
|
99
|
+
### Active Record Validator
|
100
|
+
|
101
|
+
Define a validator on an Active Record model for your email attribute(s).
|
102
|
+
It'll validate the attribute only when it's present and has changed.
|
103
|
+
|
104
|
+
#### Options
|
105
|
+
|
106
|
+
* `smtp`, `timeout`: Passed directly to API as options.
|
107
|
+
* `states`: An array of states you'd like to be considered valid.
|
108
|
+
* `free`, `role`, `disposable`, `accept_all`: If you'd like any of these to be valid.
|
109
|
+
|
110
|
+
```ruby
|
111
|
+
validates :email, email: {
|
112
|
+
smtp: true, states: %i[deliverable risky unknown],
|
113
|
+
free: true, role: true, disposable: false, accept_all: true, timeout: 3
|
114
|
+
}
|
115
|
+
```
|
116
|
+
|
117
|
+
#### Access Verification Result
|
118
|
+
|
119
|
+
You can define an `attr_accessor` with the following format to gain
|
120
|
+
access to the verification result.
|
121
|
+
|
122
|
+
```ruby
|
123
|
+
# [attribute_name]_verification_result
|
124
|
+
attr_accessor :email_verification_result
|
125
|
+
```
|
126
|
+
|
127
|
+
## Development
|
128
|
+
|
129
|
+
After checking out the repo, run `bin/setup` to install dependencies. Then, run `rake test` to run the tests. You can also run `bin/console` for an interactive prompt that will allow you to experiment.
|
130
|
+
|
131
|
+
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`, which will create a git tag for the version, push git commits and tags, and push the `.gem` file to [rubygems.org](https://rubygems.org).
|
132
|
+
|
133
|
+
## Contributing
|
134
|
+
|
135
|
+
Bug reports and pull requests are welcome on GitHub at https://github.com/emailable/emailable-ruby.
|
data/Rakefile
ADDED
data/bin/console
ADDED
@@ -0,0 +1,14 @@
|
|
1
|
+
#!/usr/bin/env ruby
|
2
|
+
|
3
|
+
require "bundler/setup"
|
4
|
+
require "emailable/ruby"
|
5
|
+
|
6
|
+
# You can add fixtures and/or initialization code here to make experimenting
|
7
|
+
# with your gem easier. You can also use a different console, if you like.
|
8
|
+
|
9
|
+
# (If you use this, don't forget to add pry to your Gemfile!)
|
10
|
+
# require "pry"
|
11
|
+
# Pry.start
|
12
|
+
|
13
|
+
require "irb"
|
14
|
+
IRB.start(__FILE__)
|
data/bin/setup
ADDED
data/emailable.gemspec
ADDED
@@ -0,0 +1,40 @@
|
|
1
|
+
# frozen_string_literal: true
|
2
|
+
|
3
|
+
$LOAD_PATH.unshift(::File.join(::File.dirname(__FILE__), 'lib'))
|
4
|
+
|
5
|
+
require 'emailable/version'
|
6
|
+
|
7
|
+
Gem::Specification.new do |s|
|
8
|
+
s.name = 'emailable'
|
9
|
+
s.version = Emailable::VERSION
|
10
|
+
s.summary = 'Ruby bindings for the Emailable API'
|
11
|
+
s.description = 'Email Verification that’s astonishingly easy and low-cost. '\
|
12
|
+
'See https://emailable.com for details.'
|
13
|
+
s.homepage = 'https://emailable.com'
|
14
|
+
s.author = 'Emailable'
|
15
|
+
s.email = 'support@emailable.com'
|
16
|
+
s.license = 'MIT'
|
17
|
+
s.metadata = {
|
18
|
+
"bug_tracker_uri" => "https://github.com/emailable/emailable-ruby/issues",
|
19
|
+
"documentation_uri" => "https://docs.emailable.com/?ruby",
|
20
|
+
"source_code_uri" => "https://github.com/emailable/emailable-ruby"
|
21
|
+
}
|
22
|
+
|
23
|
+
s.files = `git ls-files`.split("\n")
|
24
|
+
s.test_files = `git ls-files -- test/*`.split("\n")
|
25
|
+
s.executables = `git ls-files -- bin/*`.split("\n").map do |f|
|
26
|
+
::File.basename(f)
|
27
|
+
end
|
28
|
+
s.require_paths = ['lib']
|
29
|
+
|
30
|
+
s.add_dependency 'faraday'
|
31
|
+
s.add_dependency 'faraday_middleware'
|
32
|
+
s.add_dependency 'net-http-persistent'
|
33
|
+
s.add_development_dependency 'bundler'
|
34
|
+
s.add_development_dependency 'rake', '~> 13.0'
|
35
|
+
s.add_development_dependency 'pry'
|
36
|
+
s.add_development_dependency 'awesome_print'
|
37
|
+
s.add_development_dependency 'minitest', '~> 5.0'
|
38
|
+
s.add_development_dependency 'minitest-reporters'
|
39
|
+
s.add_development_dependency 'activemodel'
|
40
|
+
end
|
@@ -0,0 +1,51 @@
|
|
1
|
+
module Emailable
|
2
|
+
class Batch
|
3
|
+
attr_accessor :id
|
4
|
+
|
5
|
+
def initialize(id_or_emails, callback: nil)
|
6
|
+
if id_or_emails.is_a?(Array)
|
7
|
+
@emails = id_or_emails
|
8
|
+
@callback = callback
|
9
|
+
elsif id_or_emails.is_a?(String)
|
10
|
+
@id = id_or_emails
|
11
|
+
else
|
12
|
+
raise ArgumentError, 'expected an array of emails or batch id'
|
13
|
+
end
|
14
|
+
|
15
|
+
@client = Emailable::Client.new
|
16
|
+
end
|
17
|
+
|
18
|
+
def verify
|
19
|
+
return @id unless @id.nil?
|
20
|
+
|
21
|
+
opts = { emails: @emails.join(','), url: @callback }
|
22
|
+
response = @client.request(:post, 'batch', opts)
|
23
|
+
|
24
|
+
@id = response.body['id']
|
25
|
+
end
|
26
|
+
|
27
|
+
def status
|
28
|
+
return nil unless @id
|
29
|
+
return @status if @status
|
30
|
+
|
31
|
+
response = @client.request(:get, 'batch', { id: @id })
|
32
|
+
bs = BatchStatus.new(response.body)
|
33
|
+
@status = bs if bs.complete?
|
34
|
+
|
35
|
+
bs
|
36
|
+
end
|
37
|
+
|
38
|
+
def complete?
|
39
|
+
!status.complete?
|
40
|
+
end
|
41
|
+
|
42
|
+
def inspect
|
43
|
+
ivars = instance_variables.map do |e|
|
44
|
+
[e.to_s.delete('@'), instance_variable_get(e)]
|
45
|
+
end.to_h
|
46
|
+
"#<#{self.class}:0x#{(object_id << 1).to_s(16)}> JSON: " +
|
47
|
+
JSON.pretty_generate(ivars)
|
48
|
+
end
|
49
|
+
|
50
|
+
end
|
51
|
+
end
|
@@ -0,0 +1,70 @@
|
|
1
|
+
module Emailable
|
2
|
+
class Client
|
3
|
+
|
4
|
+
def initialize
|
5
|
+
@client = Faraday.new('https://api.emailable.com/v1') do |f|
|
6
|
+
f.request :url_encoded
|
7
|
+
f.response :json, content_type: /\bjson$/
|
8
|
+
f.adapter :net_http_persistent
|
9
|
+
end
|
10
|
+
end
|
11
|
+
|
12
|
+
def request(method, endpoint, opts = {})
|
13
|
+
begin
|
14
|
+
tries ||= 0
|
15
|
+
|
16
|
+
@client.params[:api_key] = Emailable.api_key
|
17
|
+
|
18
|
+
response =
|
19
|
+
if method == :get
|
20
|
+
@client.get(endpoint, opts)
|
21
|
+
elsif method == :post
|
22
|
+
@client.post(endpoint, opts)
|
23
|
+
end
|
24
|
+
rescue => e
|
25
|
+
retry if self.class.should_retry?(e, tries)
|
26
|
+
|
27
|
+
raise e
|
28
|
+
end
|
29
|
+
|
30
|
+
status = response.status
|
31
|
+
return response if status.between?(200, 299)
|
32
|
+
|
33
|
+
error_attributes = {
|
34
|
+
message: response.body['message'],
|
35
|
+
code: status
|
36
|
+
}
|
37
|
+
error_map = {
|
38
|
+
'400' => BadRequestError,
|
39
|
+
'401' => UnauthorizedError,
|
40
|
+
'402' => PaymentRequiredError,
|
41
|
+
'403' => ForbiddenError,
|
42
|
+
'404' => NotFoundError,
|
43
|
+
'429' => TooManyRequestsError,
|
44
|
+
'500' => InternalServerError,
|
45
|
+
'503' => ServiceUnavailableError
|
46
|
+
}
|
47
|
+
|
48
|
+
raise error_map[status.to_s].new(error_attributes)
|
49
|
+
end
|
50
|
+
|
51
|
+
def self.should_retry?(error, num_retries)
|
52
|
+
return false if num_retries >= Emailable.max_network_retries
|
53
|
+
|
54
|
+
# Retry on timeout-related problems (either on open or read).
|
55
|
+
return true if error.is_a?(Faraday::TimeoutError)
|
56
|
+
|
57
|
+
# Destination refused the connection, the connection was reset, or a
|
58
|
+
# variety of other connection failures. This could occur from a single
|
59
|
+
# saturated server, so retry in case it's intermittent.
|
60
|
+
return true if error.is_a?(Faraday::ConnectionFailed)
|
61
|
+
|
62
|
+
if error.is_a?(Faraday::ClientError) && error.response
|
63
|
+
# 409 conflict
|
64
|
+
return true if error.response[:status] == 409
|
65
|
+
end
|
66
|
+
|
67
|
+
false
|
68
|
+
end
|
69
|
+
end
|
70
|
+
end
|
@@ -0,0 +1,70 @@
|
|
1
|
+
# ActiveRecord validator for validating an email address with Emailable
|
2
|
+
#
|
3
|
+
# Usage:
|
4
|
+
# validates :email, presence: true, email: {
|
5
|
+
# smtp: true, states: %i[deliverable risky unknown],
|
6
|
+
# free: true, role: true, disposable: false, accept_all: true,
|
7
|
+
# timeout: 3
|
8
|
+
# }
|
9
|
+
#
|
10
|
+
# Define an attr_accessor to access verification results.
|
11
|
+
# attr_accessor :email_verification_result
|
12
|
+
#
|
13
|
+
class EmailValidator < ActiveModel::EachValidator
|
14
|
+
|
15
|
+
def validate_each(record, attribute, value)
|
16
|
+
smtp = boolean_option_or_raise_error(:smtp, true)
|
17
|
+
|
18
|
+
states = options.fetch(:states, %i(deliverable risky unknown))
|
19
|
+
allowed_states = %i[deliverable undeliverable risky unknown]
|
20
|
+
unless (states - allowed_states).empty?
|
21
|
+
raise ArgumentError, ":states must be an array of symbols containing "\
|
22
|
+
"any or all of :#{allowed_states.join(', :')}"
|
23
|
+
end
|
24
|
+
|
25
|
+
free = boolean_option_or_raise_error(:free, true)
|
26
|
+
role = boolean_option_or_raise_error(:role, true)
|
27
|
+
disposable = boolean_option_or_raise_error(:disposable, false)
|
28
|
+
accept_all = boolean_option_or_raise_error(:accept_all, true)
|
29
|
+
|
30
|
+
timeout = options.fetch(:timeout, 3)
|
31
|
+
unless timeout.is_a?(Integer) && timeout > 1
|
32
|
+
raise ArgumentError, ":timeout must be an Integer greater than 1"
|
33
|
+
end
|
34
|
+
|
35
|
+
return if record.errors[attribute].present?
|
36
|
+
return unless value.present?
|
37
|
+
return unless record.changes.include?(attribute.to_sym)
|
38
|
+
|
39
|
+
api_options = { timeout: timeout, smtp: smtp }
|
40
|
+
api_options[:accept_all] = true unless accept_all
|
41
|
+
ev = Emailable.verify(value, api_options)
|
42
|
+
|
43
|
+
result_accessor = "#{attribute}_verification_result"
|
44
|
+
if record.respond_to?(result_accessor)
|
45
|
+
record.instance_variable_set("@#{result_accessor}", ev)
|
46
|
+
end
|
47
|
+
|
48
|
+
error ||= ev.state.to_sym unless states.include?(ev.state.to_sym)
|
49
|
+
error ||= :free if ev.free? && !free
|
50
|
+
error ||= :role if ev.role? && !role
|
51
|
+
error ||= :disposable if ev.disposable? && !disposable
|
52
|
+
error ||= :accept_all if ev.accept_all? && !accept_all
|
53
|
+
|
54
|
+
record.errors.add(attribute, error) if error
|
55
|
+
rescue Emailable::Error
|
56
|
+
# silence errors
|
57
|
+
end
|
58
|
+
|
59
|
+
private
|
60
|
+
|
61
|
+
def boolean_option_or_raise_error(name, default)
|
62
|
+
option = options.fetch(name, default)
|
63
|
+
unless [true, false].include?(option)
|
64
|
+
raise ArgumentError, ":#{name} must by a Boolean"
|
65
|
+
end
|
66
|
+
|
67
|
+
option
|
68
|
+
end
|
69
|
+
|
70
|
+
end
|
@@ -0,0 +1,20 @@
|
|
1
|
+
module Emailable
|
2
|
+
class APIResource
|
3
|
+
|
4
|
+
def initialize(attributes = {})
|
5
|
+
attributes.each do |attr, value|
|
6
|
+
instance_variable_set("@#{attr}", value)
|
7
|
+
end
|
8
|
+
end
|
9
|
+
|
10
|
+
def inspect
|
11
|
+
ivars = instance_variables.map do |e|
|
12
|
+
[e.to_s.delete('@'), instance_variable_get(e)]
|
13
|
+
end.to_h
|
14
|
+
fmtted_email = @email ? " #{@email}" : ''
|
15
|
+
"#<#{self.class}:0x#{(object_id << 1).to_s(16)}#{fmtted_email}> JSON: " +
|
16
|
+
JSON.pretty_generate(ivars)
|
17
|
+
end
|
18
|
+
|
19
|
+
end
|
20
|
+
end
|
@@ -0,0 +1,15 @@
|
|
1
|
+
module Emailable
|
2
|
+
class Verification < APIResource
|
3
|
+
attr_accessor :accept_all, :did_you_mean, :disposable, :domain, :duration,
|
4
|
+
:email, :free, :mx_record, :reason, :role, :score,
|
5
|
+
:smtp_provider, :state, :tag, :user, :first_name, :last_name,
|
6
|
+
:full_name, :gender
|
7
|
+
|
8
|
+
%w(accept_all disposable free role).each do |method|
|
9
|
+
define_method("#{method}?") do
|
10
|
+
instance_variable_get "@#{method}"
|
11
|
+
end
|
12
|
+
end
|
13
|
+
|
14
|
+
end
|
15
|
+
end
|
data/lib/emailable.rb
ADDED
@@ -0,0 +1,65 @@
|
|
1
|
+
require 'faraday'
|
2
|
+
require 'faraday_middleware'
|
3
|
+
require 'emailable/version'
|
4
|
+
require 'emailable/client'
|
5
|
+
require 'emailable/batch'
|
6
|
+
require 'emailable/resources/api_resource'
|
7
|
+
require 'emailable/resources/account'
|
8
|
+
require 'emailable/resources/batch_status'
|
9
|
+
require 'emailable/resources/verification'
|
10
|
+
if defined?(ActiveModel)
|
11
|
+
require 'emailable/email_validator'
|
12
|
+
I18n.load_path += Dir.glob(File.expand_path('../../config/locales/**/*', __FILE__))
|
13
|
+
end
|
14
|
+
|
15
|
+
module Emailable
|
16
|
+
@max_network_retries = 1
|
17
|
+
|
18
|
+
class << self
|
19
|
+
attr_accessor :api_key, :max_network_retries
|
20
|
+
end
|
21
|
+
|
22
|
+
module_function
|
23
|
+
|
24
|
+
def verify(email, smtp: nil, accept_all: nil, timeout: nil)
|
25
|
+
opts = {
|
26
|
+
email: email, smtp: smtp, accept_all: accept_all, timeout: timeout
|
27
|
+
}
|
28
|
+
|
29
|
+
client = Emailable::Client.new
|
30
|
+
response = client.request(:get, 'verify', opts)
|
31
|
+
|
32
|
+
if response.status == 249
|
33
|
+
raise Emailable::TimeoutError.new(
|
34
|
+
code: response.status, message: response.body
|
35
|
+
)
|
36
|
+
else
|
37
|
+
Verification.new(response.body)
|
38
|
+
end
|
39
|
+
end
|
40
|
+
|
41
|
+
def account
|
42
|
+
client = Emailable::Client.new
|
43
|
+
response = client.request(:get, 'account')
|
44
|
+
Account.new(response.body)
|
45
|
+
end
|
46
|
+
|
47
|
+
|
48
|
+
class Error < StandardError
|
49
|
+
attr_accessor :code, :message
|
50
|
+
|
51
|
+
def initialize(code: nil, message: nil)
|
52
|
+
@code = code
|
53
|
+
@message = message
|
54
|
+
end
|
55
|
+
end
|
56
|
+
class BadRequestError < Error; end
|
57
|
+
class UnauthorizedError < Error; end
|
58
|
+
class PaymentRequiredError < Error; end
|
59
|
+
class ForbiddenError < Error; end
|
60
|
+
class NotFoundError < Error; end
|
61
|
+
class TooManyRequestsError < Error; end
|
62
|
+
class InternalServerError < Error; end
|
63
|
+
class ServiceUnavailableError < Error; end
|
64
|
+
class TimeoutError < Error; end
|
65
|
+
end
|
@@ -0,0 +1,85 @@
|
|
1
|
+
require 'test_helper'
|
2
|
+
|
3
|
+
class EmailValidatorTest < Minitest::Test
|
4
|
+
|
5
|
+
def user_class(
|
6
|
+
smtp: true, states: %i[deliverable risky unknown], free: true, role: true,
|
7
|
+
accept_all: true, disposable: true, timeout: 3
|
8
|
+
)
|
9
|
+
Class.new do
|
10
|
+
include ActiveModel::Model
|
11
|
+
attr_accessor :email, :email_verification_result
|
12
|
+
|
13
|
+
validates :email, presence: true, email: {
|
14
|
+
smtp: smtp, states: states,
|
15
|
+
free: free, role: role, disposable: disposable, accept_all: accept_all,
|
16
|
+
timeout: timeout
|
17
|
+
}
|
18
|
+
|
19
|
+
def self.name
|
20
|
+
'TestClass'
|
21
|
+
end
|
22
|
+
|
23
|
+
# stub changes to always be true
|
24
|
+
def changes
|
25
|
+
{ email: true }
|
26
|
+
end
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
def setup
|
31
|
+
Emailable.api_key = 'test_7aff7fc0142c65f86a00'
|
32
|
+
sleep(0.25)
|
33
|
+
end
|
34
|
+
|
35
|
+
def test_valid
|
36
|
+
@user = user_class.new(email: 'deliverable@example.com')
|
37
|
+
|
38
|
+
assert @user.valid?
|
39
|
+
assert @user.errors.empty?
|
40
|
+
end
|
41
|
+
|
42
|
+
def test_invalid
|
43
|
+
@user = user_class.new(email: 'undeliverable@example.com')
|
44
|
+
|
45
|
+
assert !@user.valid?
|
46
|
+
assert @user.errors[:email].present?
|
47
|
+
end
|
48
|
+
|
49
|
+
def test_verification_result
|
50
|
+
@user = user_class.new(email: 'undeliverable@example.com')
|
51
|
+
@user.valid?
|
52
|
+
|
53
|
+
refute_nil @user.email_verification_result
|
54
|
+
assert @user.email_verification_result.state, :undeliverable
|
55
|
+
end
|
56
|
+
|
57
|
+
def test_boolean_options
|
58
|
+
%i[smtp free role disposable accept_all].each do |option|
|
59
|
+
invalid_user = user_class(option => 'string').new
|
60
|
+
valid_user = user_class.new
|
61
|
+
|
62
|
+
assert !valid_user.valid?
|
63
|
+
assert_raises(ArgumentError) { invalid_user.valid? }
|
64
|
+
end
|
65
|
+
end
|
66
|
+
|
67
|
+
def test_states_option
|
68
|
+
invalid_user = user_class(states: %i[invalid_state]).new
|
69
|
+
valid_user = user_class.new
|
70
|
+
|
71
|
+
assert !valid_user.valid?
|
72
|
+
assert_raises(ArgumentError) { invalid_user.valid? }
|
73
|
+
end
|
74
|
+
|
75
|
+
def test_timeout_option
|
76
|
+
invalid_user1 = user_class(timeout: 'string').new
|
77
|
+
invalid_user2 = user_class(timeout: 1).new
|
78
|
+
valid_user = user_class.new
|
79
|
+
|
80
|
+
assert !valid_user.valid?
|
81
|
+
assert_raises(ArgumentError) { invalid_user1.valid? }
|
82
|
+
assert_raises(ArgumentError) { invalid_user2.valid? }
|
83
|
+
end
|
84
|
+
|
85
|
+
end
|
@@ -0,0 +1,30 @@
|
|
1
|
+
require 'test_helper'
|
2
|
+
|
3
|
+
module Emailable
|
4
|
+
class BatchTest < Minitest::Test
|
5
|
+
|
6
|
+
def setup
|
7
|
+
sleep(1)
|
8
|
+
Emailable.api_key = 'test_7aff7fc0142c65f86a00'
|
9
|
+
@emails = ['jarrett@emailable.com', 'support@emailable.com']
|
10
|
+
@batch = Emailable::Batch.new(@emails)
|
11
|
+
@batch_id ||= @batch.verify
|
12
|
+
sleep(1)
|
13
|
+
end
|
14
|
+
|
15
|
+
def test_start_batch
|
16
|
+
refute_nil @batch_id
|
17
|
+
end
|
18
|
+
|
19
|
+
def test_batch_status
|
20
|
+
status = @batch.status
|
21
|
+
refute_nil status.message
|
22
|
+
end
|
23
|
+
|
24
|
+
def test_batch_complete
|
25
|
+
complete = @batch.complete?
|
26
|
+
assert complete == true || complete == false
|
27
|
+
end
|
28
|
+
|
29
|
+
end
|
30
|
+
end
|
@@ -0,0 +1,71 @@
|
|
1
|
+
require 'test_helper'
|
2
|
+
|
3
|
+
class EmailableTest < Minitest::Test
|
4
|
+
|
5
|
+
def setup
|
6
|
+
Emailable.api_key = 'test_7aff7fc0142c65f86a00'
|
7
|
+
@result ||= Emailable.verify('jarrett@emailable.com')
|
8
|
+
sleep(0.25)
|
9
|
+
end
|
10
|
+
|
11
|
+
def test_verification
|
12
|
+
refute_nil @result.domain
|
13
|
+
refute_nil @result.email
|
14
|
+
refute_nil @result.reason
|
15
|
+
refute_nil @result.score
|
16
|
+
refute_nil @result.state
|
17
|
+
refute_nil @result.user
|
18
|
+
refute_nil @result.duration
|
19
|
+
end
|
20
|
+
|
21
|
+
def test_verification_state
|
22
|
+
assert %w(deliverable undeliverable risky unknown).include?(@result.state)
|
23
|
+
end
|
24
|
+
|
25
|
+
def test_verification_role
|
26
|
+
result = Emailable.verify('support@emailable.com')
|
27
|
+
assert result.role?
|
28
|
+
refute @result.role?
|
29
|
+
end
|
30
|
+
|
31
|
+
def test_verification_did_you_mean
|
32
|
+
result1 = Emailable.verify('jarrett@gmali.com')
|
33
|
+
result2 = Emailable.verify('jarrett@gmail.com')
|
34
|
+
assert result1.did_you_mean, 'jarrett@gmail.com'
|
35
|
+
assert_nil result2.did_you_mean
|
36
|
+
end
|
37
|
+
|
38
|
+
def test_verification_tag
|
39
|
+
result = Emailable.verify('jarrett+marketing@emailable.com')
|
40
|
+
assert result.tag == 'marketing'
|
41
|
+
end
|
42
|
+
|
43
|
+
def test_account
|
44
|
+
account = Emailable.account
|
45
|
+
|
46
|
+
refute_nil account.owner_email
|
47
|
+
refute_nil account.available_credits
|
48
|
+
end
|
49
|
+
|
50
|
+
def test_name_and_gender
|
51
|
+
result = Emailable.verify('johndoe@emailable.com')
|
52
|
+
if %w(deliverable risky unknown).include?(result.state)
|
53
|
+
assert result.first_name, 'John'
|
54
|
+
assert result.last_name, 'Doe'
|
55
|
+
assert result.full_name, 'John Doe'
|
56
|
+
assert result.gender, 'male'
|
57
|
+
else
|
58
|
+
assert_nil result.first_name
|
59
|
+
assert_nil result.last_name
|
60
|
+
assert_nil result.full_name
|
61
|
+
assert_nil result.gender
|
62
|
+
end
|
63
|
+
end
|
64
|
+
|
65
|
+
def test_slow_verification
|
66
|
+
assert_raises(Emailable::TimeoutError) do
|
67
|
+
Emailable.verify('slow@example.com')
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
end
|
data/test/test_helper.rb
ADDED
@@ -0,0 +1,12 @@
|
|
1
|
+
$LOAD_PATH.unshift File.expand_path("../../lib", __FILE__)
|
2
|
+
require 'active_model'
|
3
|
+
require 'emailable'
|
4
|
+
|
5
|
+
require 'pry'
|
6
|
+
require 'minitest/autorun'
|
7
|
+
require 'minitest/reporters'
|
8
|
+
Minitest::Reporters.use! [
|
9
|
+
Minitest::Reporters::ProgressReporter.new(color: true)
|
10
|
+
]
|
11
|
+
Minitest.load_plugins
|
12
|
+
Minitest::PrideIO.pride!
|
metadata
ADDED
@@ -0,0 +1,218 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: emailable
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 3.0.1
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Emailable
|
8
|
+
autorequire:
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
date: 2021-03-19 00:00:00.000000000 Z
|
12
|
+
dependencies:
|
13
|
+
- !ruby/object:Gem::Dependency
|
14
|
+
name: faraday
|
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: faraday_middleware
|
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: net-http-persistent
|
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: bundler
|
57
|
+
requirement: !ruby/object:Gem::Requirement
|
58
|
+
requirements:
|
59
|
+
- - ">="
|
60
|
+
- !ruby/object:Gem::Version
|
61
|
+
version: '0'
|
62
|
+
type: :development
|
63
|
+
prerelease: false
|
64
|
+
version_requirements: !ruby/object:Gem::Requirement
|
65
|
+
requirements:
|
66
|
+
- - ">="
|
67
|
+
- !ruby/object:Gem::Version
|
68
|
+
version: '0'
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
name: rake
|
71
|
+
requirement: !ruby/object:Gem::Requirement
|
72
|
+
requirements:
|
73
|
+
- - "~>"
|
74
|
+
- !ruby/object:Gem::Version
|
75
|
+
version: '13.0'
|
76
|
+
type: :development
|
77
|
+
prerelease: false
|
78
|
+
version_requirements: !ruby/object:Gem::Requirement
|
79
|
+
requirements:
|
80
|
+
- - "~>"
|
81
|
+
- !ruby/object:Gem::Version
|
82
|
+
version: '13.0'
|
83
|
+
- !ruby/object:Gem::Dependency
|
84
|
+
name: pry
|
85
|
+
requirement: !ruby/object:Gem::Requirement
|
86
|
+
requirements:
|
87
|
+
- - ">="
|
88
|
+
- !ruby/object:Gem::Version
|
89
|
+
version: '0'
|
90
|
+
type: :development
|
91
|
+
prerelease: false
|
92
|
+
version_requirements: !ruby/object:Gem::Requirement
|
93
|
+
requirements:
|
94
|
+
- - ">="
|
95
|
+
- !ruby/object:Gem::Version
|
96
|
+
version: '0'
|
97
|
+
- !ruby/object:Gem::Dependency
|
98
|
+
name: awesome_print
|
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
|
+
- !ruby/object:Gem::Dependency
|
112
|
+
name: minitest
|
113
|
+
requirement: !ruby/object:Gem::Requirement
|
114
|
+
requirements:
|
115
|
+
- - "~>"
|
116
|
+
- !ruby/object:Gem::Version
|
117
|
+
version: '5.0'
|
118
|
+
type: :development
|
119
|
+
prerelease: false
|
120
|
+
version_requirements: !ruby/object:Gem::Requirement
|
121
|
+
requirements:
|
122
|
+
- - "~>"
|
123
|
+
- !ruby/object:Gem::Version
|
124
|
+
version: '5.0'
|
125
|
+
- !ruby/object:Gem::Dependency
|
126
|
+
name: minitest-reporters
|
127
|
+
requirement: !ruby/object:Gem::Requirement
|
128
|
+
requirements:
|
129
|
+
- - ">="
|
130
|
+
- !ruby/object:Gem::Version
|
131
|
+
version: '0'
|
132
|
+
type: :development
|
133
|
+
prerelease: false
|
134
|
+
version_requirements: !ruby/object:Gem::Requirement
|
135
|
+
requirements:
|
136
|
+
- - ">="
|
137
|
+
- !ruby/object:Gem::Version
|
138
|
+
version: '0'
|
139
|
+
- !ruby/object:Gem::Dependency
|
140
|
+
name: activemodel
|
141
|
+
requirement: !ruby/object:Gem::Requirement
|
142
|
+
requirements:
|
143
|
+
- - ">="
|
144
|
+
- !ruby/object:Gem::Version
|
145
|
+
version: '0'
|
146
|
+
type: :development
|
147
|
+
prerelease: false
|
148
|
+
version_requirements: !ruby/object:Gem::Requirement
|
149
|
+
requirements:
|
150
|
+
- - ">="
|
151
|
+
- !ruby/object:Gem::Version
|
152
|
+
version: '0'
|
153
|
+
description: Email Verification that’s astonishingly easy and low-cost. See https://emailable.com
|
154
|
+
for details.
|
155
|
+
email: support@emailable.com
|
156
|
+
executables:
|
157
|
+
- console
|
158
|
+
- setup
|
159
|
+
extensions: []
|
160
|
+
extra_rdoc_files: []
|
161
|
+
files:
|
162
|
+
- ".gitignore"
|
163
|
+
- ".ruby-gemset"
|
164
|
+
- ".ruby-version"
|
165
|
+
- ".travis.yml"
|
166
|
+
- Gemfile
|
167
|
+
- Gemfile.lock
|
168
|
+
- LICENSE.txt
|
169
|
+
- README.md
|
170
|
+
- Rakefile
|
171
|
+
- bin/console
|
172
|
+
- bin/setup
|
173
|
+
- config/locales/en.yml
|
174
|
+
- emailable.gemspec
|
175
|
+
- lib/emailable.rb
|
176
|
+
- lib/emailable/batch.rb
|
177
|
+
- lib/emailable/client.rb
|
178
|
+
- lib/emailable/email_validator.rb
|
179
|
+
- lib/emailable/resources/account.rb
|
180
|
+
- lib/emailable/resources/api_resource.rb
|
181
|
+
- lib/emailable/resources/batch_status.rb
|
182
|
+
- lib/emailable/resources/verification.rb
|
183
|
+
- lib/emailable/version.rb
|
184
|
+
- test/email_validator_test.rb
|
185
|
+
- test/emailable/batch_test.rb
|
186
|
+
- test/emailable_test.rb
|
187
|
+
- test/test_helper.rb
|
188
|
+
homepage: https://emailable.com
|
189
|
+
licenses:
|
190
|
+
- MIT
|
191
|
+
metadata:
|
192
|
+
bug_tracker_uri: https://github.com/emailable/emailable-ruby/issues
|
193
|
+
documentation_uri: https://docs.emailable.com/?ruby
|
194
|
+
source_code_uri: https://github.com/emailable/emailable-ruby
|
195
|
+
post_install_message:
|
196
|
+
rdoc_options: []
|
197
|
+
require_paths:
|
198
|
+
- lib
|
199
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
200
|
+
requirements:
|
201
|
+
- - ">="
|
202
|
+
- !ruby/object:Gem::Version
|
203
|
+
version: '0'
|
204
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
205
|
+
requirements:
|
206
|
+
- - ">="
|
207
|
+
- !ruby/object:Gem::Version
|
208
|
+
version: '0'
|
209
|
+
requirements: []
|
210
|
+
rubygems_version: 3.0.9
|
211
|
+
signing_key:
|
212
|
+
specification_version: 4
|
213
|
+
summary: Ruby bindings for the Emailable API
|
214
|
+
test_files:
|
215
|
+
- test/email_validator_test.rb
|
216
|
+
- test/emailable/batch_test.rb
|
217
|
+
- test/emailable_test.rb
|
218
|
+
- test/test_helper.rb
|