Skip to content

Commit b679d80

Browse files
committed
Initial Commit
0 parents  commit b679d80

16 files changed

+423
-0
lines changed

.editorconfig

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
root = true
2+
3+
[*.cr]
4+
charset = utf-8
5+
end_of_line = lf
6+
insert_final_newline = true
7+
indent_style = space
8+
indent_size = 2
9+
trim_trailing_whitespace = true

.gitignore

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
/docs/
2+
/lib/
3+
/bin/
4+
/.shards/
5+
*.dwarf
6+
7+
# Libraries don't need dependency lock
8+
# Dependencies will be locked in applications that use them
9+
/shard.lock
10+
test.cr

.travis.yml

+6
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,6 @@
1+
language: crystal
2+
3+
# Uncomment the following if you'd like Travis to run specs and check code formatting
4+
# script:
5+
# - crystal spec
6+
# - crystal tool format --check

LICENSE

+21
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,21 @@
1+
The MIT License (MIT)
2+
3+
Copyright (c) 2019 Dainel Vera
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.

README.md

+39
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,39 @@
1+
# hunter.cr
2+
3+
A Crystal-Lang wrapper for Hunter.io API V2
4+
5+
## Installation
6+
7+
1. Add the dependency to your `shard.yml`:
8+
9+
```yaml
10+
dependencies:
11+
hunter:
12+
github: arubinofaux/hunter.cr
13+
```
14+
15+
2. Run `shards install`
16+
17+
## Usage
18+
19+
```crystal
20+
require "hunter"
21+
```
22+
23+
TODO: Write usage instructions here
24+
25+
## Development
26+
27+
TODO: Write development instructions here
28+
29+
## Contributing
30+
31+
1. Fork it (<https://github.com/arubinofaux/hunter.cr/fork>)
32+
2. Create your feature branch (`git checkout -b my-new-feature`)
33+
3. Commit your changes (`git commit -am 'Add some feature'`)
34+
4. Push to the branch (`git push origin my-new-feature`)
35+
5. Create a new Pull Request
36+
37+
## Contributors
38+
39+
- [arubinofaux](https://github.com/arubinofaux) - creator and maintainer

shard.yml

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
name: hunter
2+
version: 0.1.0
3+
4+
authors:
5+
- Dainel Vera <[email protected]>
6+
7+
crystal: 0.31.1
8+
9+
license: MIT

spec/hunter_spec.cr

+9
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,9 @@
1+
require "./spec_helper"
2+
3+
describe Hunter do
4+
# TODO: Write tests
5+
6+
it "works" do
7+
false.should eq(true)
8+
end
9+
end

spec/spec_helper.cr

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
require "spec"
2+
require "../src/hunter"

src/hunter.cr

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
require "http/client"
2+
require "json"
3+
4+
require "./hunter/**"
5+
6+
module Hunter
7+
VERSION = "0.1.0"
8+
9+
def self.new(apiKey : String)
10+
Client.new(apiKey)
11+
end
12+
end

src/hunter/api.cr

+45
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,45 @@
1+
module Hunter
2+
class Api
3+
APIVersion = "v2"
4+
Headers = HTTP::Headers{"Content-Type" => "application/json", "Accept" => "application/json", "User-Agent" => "hunter.cr"}
5+
6+
def initialize(@httpClient : HTTP::Client, @apiKey : String)
7+
end
8+
9+
# 200 - OK / The request was successful
10+
# 201 - Created / The request was successful and the resource was created
11+
# 202 - Email verification in progress / The Email verification is still in progress. Feel free to make the same verification again as often as you want, it will only count as a single request until we return the response
12+
# 204 - No content / The request was successful and no additional content was sent
13+
# 222 - Email verification failed / The Email verification failed because of an unexpected response from the remote SMTP server. This failure is outside of our control. We advise you to retry later
14+
# 400 - Bad request / Your request was not valid
15+
# 401 - Unauthorized / No valid API key was provided
16+
# 403 - Forbidden / You have reached the rate limit
17+
# 404 - Not found / The requested resource does not exist
18+
# 422 - Unprocessable entity / Your request is valid but the creation of the resource failed. Check the errors
19+
# 429 - Too many requests / You have reached your usage limit. Upgrade your plan if necessary
20+
# 451 - Unavailable for legal reasons / The person behind the requested resource has asked us directly or indirectly to stop the processing of this resource. For this reason, you shouldn't process this resource yourself in any way
21+
# 5XX - Server errors / Something went wrong on Hunter's end
22+
def handleResponse(response : HTTP::Client::Response)
23+
case response.status_code
24+
when 200, 201, 202, 222
25+
JSON.parse(response.body)
26+
when 204
27+
{"ok"}
28+
when 400, 401, 403, 404, 422, 429, 451
29+
JSON.parse(response.body)
30+
when 500
31+
JSON.parse(response.body)
32+
else
33+
{
34+
"errors": [
35+
{
36+
"id": "total_meltdown",
37+
"code": 9000,
38+
"details": "Something bad happened"
39+
}
40+
]
41+
}
42+
end
43+
end
44+
end
45+
end

src/hunter/api/account.cr

+12
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,12 @@
1+
module Hunter
2+
class Api::Account < Api
3+
def me()
4+
params = Hash(String, String).new
5+
params["api_key"] = @apiKey
6+
params = HTTP::Params.encode(params)
7+
8+
response = @httpClient.get("/#{APIVersion}/account?" + params, headers: Headers)
9+
handleResponse(response)
10+
end
11+
end
12+
end

src/hunter/api/domain.cr

+19
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,19 @@
1+
module Hunter
2+
class Api::Domain < Api
3+
def search(domain = nil, company = nil, limit = nil, offset = nil, type = nil, seniority = nil, department = nil)
4+
params = Hash(String, String).new
5+
params["domain"] = domain if domain
6+
params["company"] = company if company
7+
params["limit"] = limit if limit
8+
params["offset"] = offset if offset
9+
params["type"] = type if type
10+
params["seniority"] = seniority if seniority
11+
params["department"] = department if department
12+
params["api_key"] = @apiKey
13+
params = HTTP::Params.encode(params)
14+
15+
response = @httpClient.get("/#{APIVersion}/domain-search?" + params, headers: Headers)
16+
handleResponse(response)
17+
end
18+
end
19+
end

src/hunter/api/email.cr

+38
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,38 @@
1+
module Hunter
2+
class Api::Email < Api
3+
def finder(domain = nil, company = nil, first_name = nil, last_name = nil, full_name = nil)
4+
params = Hash(String, String).new
5+
params["domain"] = domain if domain
6+
params["company"] = company if company
7+
params["first_name"] = first_name if first_name
8+
params["last_name"] = last_name if last_name
9+
params["full_name"] = full_name if full_name
10+
params["api_key"] = @apiKey
11+
params = HTTP::Params.encode(params)
12+
13+
response = @httpClient.get("/#{APIVersion}/email-finder?" + params, headers: Headers)
14+
handleResponse(response)
15+
end
16+
17+
def verifier(email = nil)
18+
params = Hash(String, String).new
19+
params["email"] = email if email
20+
params["api_key"] = @apiKey
21+
params = HTTP::Params.encode(params)
22+
23+
response = @httpClient.get("/#{APIVersion}/email-verifier?" + params, headers: Headers)
24+
handleResponse(response)
25+
end
26+
27+
def count(domain = nil, company = nil, type = nil)
28+
params = Hash(String, String).new
29+
params["domain"] = domain if domain
30+
params["company"] = company if company
31+
params["type"] = type if type
32+
params = HTTP::Params.encode(params)
33+
34+
response = @httpClient.get("/#{APIVersion}/email-count?" + params, headers: Headers)
35+
handleResponse(response)
36+
end
37+
end
38+
end

src/hunter/api/leads.cr

+104
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,104 @@
1+
module Hunter
2+
class Api::Leads < Api
3+
def all(limit = nil, lead_list_id = nil, email = nil, first_name = nil, last_name = nil, position = nil, company = nil, industry = nil, website = nil, country_code = nil, company_size = nil, source = nil, twitter = nil, linkedin_url = nil, phone_number = nil, sync_status = nil, sending_status = nil, last_activity_at = nil, query = nil, offset = nil)
4+
params = Hash(String, String).new
5+
params["limit"] = limit if limit
6+
params["lead_list_id"] = lead_list_id if lead_list_id
7+
params["email"] = email if email
8+
params["first_name"] = first_name if first_name
9+
params["last_name"] = last_name if last_name
10+
params["position"] = position if position
11+
params["company"] = company if company
12+
params["industry"] = industry if industry
13+
params["website"] = website if website
14+
params["country_code"] = country_code if country_code
15+
params["company_size"] = company_size if company_size
16+
params["source"] = source if source
17+
params["twitter"] = twitter if twitter
18+
params["linkedin_url"] = linkedin_url if linkedin_url
19+
params["phone_number"] = phone_number if phone_number
20+
params["sync_status"] = sync_status if sync_status
21+
params["sending_status"] = sending_status if sending_status
22+
params["last_activity_at"] = last_activity_at if last_activity_at
23+
params["query"] = query if query
24+
params["offset"] = offset if offset
25+
params["api_key"] = @apiKey
26+
params = HTTP::Params.encode(params)
27+
28+
response = @httpClient.get("/#{APIVersion}/leads?" + params, headers: Headers)
29+
handleResponse(response)
30+
end
31+
32+
def get(id = nil)
33+
params = Hash(String, String).new
34+
params["api_key"] = @apiKey
35+
params = HTTP::Params.encode(params)
36+
37+
response = @httpClient.get("/#{APIVersion}/leads/#{id}?" + params, headers: Headers)
38+
handleResponse(response)
39+
end
40+
41+
def create(email = nil, first_name = nil, last_name = nil, position = nil, company = nil, company_industry = nil, company_size = nil, confidence_score = nil, website = nil, country_code = nil, linkedin_url = nil, phone_number = nil, twitter = nil, notes = nil, source = nil, lead_list_id = nil)
42+
params = Hash(String, String).new
43+
params["api_key"] = @apiKey
44+
params = HTTP::Params.encode(params)
45+
46+
body = Hash(String, String).new
47+
body["email"] = email if email
48+
body["first_name"] = first_name if first_name
49+
body["last_name"] = last_name if last_name
50+
body["position"] = position if position
51+
body["company"] = company if company
52+
body["company_industry"] = company_industry if company_industry
53+
body["company_size"] = company_size if company_size
54+
body["confidence_score"] = confidence_score if confidence_score
55+
body["website"] = website if website
56+
body["country_code"] = country_code if country_code
57+
body["linkedin_url"] = linkedin_url if linkedin_url
58+
body["phone_number"] = phone_number if phone_number
59+
body["twitter"] = twitter if twitter
60+
body["notes"] = notes if notes
61+
body["source"] = source if source
62+
body["lead_list_id"] = lead_list_id if lead_list_id
63+
64+
response = @httpClient.post("/#{APIVersion}/leads?" + params, headers: Headers, body: body.to_json)
65+
handleResponse(response)
66+
end
67+
68+
def update(id = nil, email = nil, first_name = nil, last_name = nil, position = nil, company = nil, company_industry = nil, company_size = nil, confidence_score = nil, website = nil, country_code = nil, linkedin_url = nil, phone_number = nil, twitter = nil, notes = nil, source = nil, lead_list_id = nil)
69+
params = Hash(String, String).new
70+
params["api_key"] = @apiKey
71+
params = HTTP::Params.encode(params)
72+
73+
body = Hash(String, String).new
74+
body["email"] = email if email
75+
body["first_name"] = first_name if first_name
76+
body["last_name"] = last_name if last_name
77+
body["position"] = position if position
78+
body["company"] = company if company
79+
body["company_industry"] = company_industry if company_industry
80+
body["company_size"] = company_size if company_size
81+
body["confidence_score"] = confidence_score if confidence_score
82+
body["website"] = website if website
83+
body["country_code"] = country_code if country_code
84+
body["linkedin_url"] = linkedin_url if linkedin_url
85+
body["phone_number"] = phone_number if phone_number
86+
body["twitter"] = twitter if twitter
87+
body["notes"] = notes if notes
88+
body["source"] = source if source
89+
body["lead_list_id"] = lead_list_id if lead_list_id
90+
91+
response = @httpClient.put("/#{APIVersion}/leads/#{id}?" + params, headers: Headers, body: body.to_json)
92+
handleResponse(response)
93+
end
94+
95+
def delete(id = nil)
96+
params = Hash(String, String).new
97+
params["api_key"] = @apiKey
98+
params = HTTP::Params.encode(params)
99+
100+
response = @httpClient.delete("/#{APIVersion}/leads/#{id}?" + params, headers: Headers)
101+
handleResponse(response)
102+
end
103+
end
104+
end

0 commit comments

Comments
 (0)