caller_id 0.0.1
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.
- data/Gemfile +4 -0
- data/Gemfile.lock +37 -0
- data/LICENSE +22 -0
- data/README.md +54 -0
- data/Rakefile +20 -0
- data/caller_id.gemspec +26 -0
- data/lib/caller_id.rb +38 -0
- data/lib/caller_id/version.rb +3 -0
- data/spec/caller_id_spec.rb +41 -0
- data/spec/cassettes/CallerId/_lookup.yml +34 -0
- data/spec/cassettes/CallerId/_lookup_uncached_number.yml +34 -0
- data/spec/cassettes/CallerId/_lookup_unlisted_number.yml +34 -0
- data/spec/spec_helper.rb +12 -0
- metadata +129 -0
data/Gemfile
ADDED
data/Gemfile.lock
ADDED
@@ -0,0 +1,37 @@
|
|
1
|
+
PATH
|
2
|
+
remote: .
|
3
|
+
specs:
|
4
|
+
caller_id (0.0.1)
|
5
|
+
json
|
6
|
+
|
7
|
+
GEM
|
8
|
+
remote: https://rubygems.org/
|
9
|
+
specs:
|
10
|
+
addressable (2.2.8)
|
11
|
+
crack (0.3.1)
|
12
|
+
diff-lcs (1.1.3)
|
13
|
+
json (1.6.1)
|
14
|
+
rspec (2.10.0)
|
15
|
+
rspec-core (~> 2.10.0)
|
16
|
+
rspec-expectations (~> 2.10.0)
|
17
|
+
rspec-mocks (~> 2.10.0)
|
18
|
+
rspec-core (2.10.1)
|
19
|
+
rspec-expectations (2.10.0)
|
20
|
+
diff-lcs (~> 1.1.3)
|
21
|
+
rspec-mocks (2.10.1)
|
22
|
+
timecop (0.3.5)
|
23
|
+
vcr (2.2.2)
|
24
|
+
webmock (1.8.6)
|
25
|
+
addressable (>= 2.2.7)
|
26
|
+
crack (>= 0.1.7)
|
27
|
+
|
28
|
+
PLATFORMS
|
29
|
+
ruby
|
30
|
+
|
31
|
+
DEPENDENCIES
|
32
|
+
bundler (>= 1.0.0)
|
33
|
+
caller_id!
|
34
|
+
rspec (~> 2.6)
|
35
|
+
timecop (~> 0.3)
|
36
|
+
vcr (~> 2.1)
|
37
|
+
webmock (= 1.8.6)
|
data/LICENSE
ADDED
@@ -0,0 +1,22 @@
|
|
1
|
+
Copyright (c) 2012 Ben Ellis
|
2
|
+
|
3
|
+
MIT License
|
4
|
+
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
6
|
+
a copy of this software and associated documentation files (the
|
7
|
+
"Software"), to deal in the Software without restriction, including
|
8
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
9
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
10
|
+
permit persons to whom the Software is furnished to do so, subject to
|
11
|
+
the following conditions:
|
12
|
+
|
13
|
+
The above copyright notice and this permission notice shall be
|
14
|
+
included in all copies or substantial portions of the Software.
|
15
|
+
|
16
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
17
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
18
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
19
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
20
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
21
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
22
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,54 @@
|
|
1
|
+
# CallerId
|
2
|
+
|
3
|
+
A ruby wrapper around OpenCNAM api. Lookup cell phone caller ID in supported countries. See http://docs.opencnam.com/#what-s-our-coverage
|
4
|
+
|
5
|
+
## Installation
|
6
|
+
|
7
|
+
Add this line to your application's Gemfile:
|
8
|
+
|
9
|
+
gem 'caller_id'
|
10
|
+
|
11
|
+
And then execute:
|
12
|
+
|
13
|
+
$ bundle
|
14
|
+
|
15
|
+
Or install it yourself as:
|
16
|
+
|
17
|
+
$ gem install caller_id
|
18
|
+
|
19
|
+
## Usage
|
20
|
+
|
21
|
+
c = CallerId::ReversePhoneLookup "+1 (202) 456-1414", open_cnam_username, open_cnam_api_key
|
22
|
+
# Username and key are optional. Sign up at http://opencnam.com to get them.
|
23
|
+
c.lookup
|
24
|
+
# {:cnam => "WHITE HOUSE SWI", :number => "2024561414"}
|
25
|
+
|
26
|
+
Note: if a number has not yet been mapped by OpenCNAM, you'll get a response containing :retry => true (see the spec).
|
27
|
+
|
28
|
+
In that case you should wait a few seconds and then try the call again. You may want to do this in a background queue.
|
29
|
+
|
30
|
+
Also note, the "cnam" value returned by a successful lookup is just a caller id string, all caps, with varying semantics.
|
31
|
+
|
32
|
+
I've seen:
|
33
|
+
* [first last]
|
34
|
+
* [last first]
|
35
|
+
* [last, first middle]
|
36
|
+
|
37
|
+
If you do need to parse it into first and last name, be aware that the string format varies wildly. One possible
|
38
|
+
approach would be to do this:
|
39
|
+
|
40
|
+
first_and_maybe_middle_initial, last = result[:cname].split(",",2).reverse.join(" ").split(" ",2)
|
41
|
+
|
42
|
+
But be warned that this will not always parse them correctly, and that in some cases the string returned might not even be a name
|
43
|
+
(see the White House example). You might want to treat it as a "best guess" that should then be confirmed via other sources.
|
44
|
+
|
45
|
+
If you do not sign up for a username and API key with OpenCNAM, be aware that they currently throttle free requests to 60 per hour.
|
46
|
+
Also note the username and API keys will soon be associated with a price plan.
|
47
|
+
|
48
|
+
## Contributing
|
49
|
+
|
50
|
+
1. Fork it
|
51
|
+
2. Create your feature branch (`git checkout -b my-new-feature`)
|
52
|
+
3. Commit your changes (`git commit -am 'Added some feature'`)
|
53
|
+
4. Push to the branch (`git push origin my-new-feature`)
|
54
|
+
5. Create new Pull Request
|
data/Rakefile
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
#!/usr/bin/env rake
|
2
|
+
require "bundler/gem_tasks"
|
3
|
+
require 'rake'
|
4
|
+
require 'rspec/core/rake_task'
|
5
|
+
|
6
|
+
task :default => :spec
|
7
|
+
|
8
|
+
|
9
|
+
desc "Run specs"
|
10
|
+
RSpec::Core::RakeTask.new do |t|
|
11
|
+
t.pattern = "./spec/**/*_spec.rb" # don't need this, it's default.
|
12
|
+
# Put spec opts in a file named .rspec in root
|
13
|
+
end
|
14
|
+
|
15
|
+
desc "Generate code coverage"
|
16
|
+
RSpec::Core::RakeTask.new(:coverage) do |t|
|
17
|
+
t.pattern = "./spec/**/*_spec.rb" # don't need this, it's default.
|
18
|
+
t.rcov = true
|
19
|
+
t.rcov_opts = ['--exclude', 'spec']
|
20
|
+
end
|
data/caller_id.gemspec
ADDED
@@ -0,0 +1,26 @@
|
|
1
|
+
# -*- encoding: utf-8 -*-
|
2
|
+
require File.expand_path('../lib/caller_id/version', __FILE__)
|
3
|
+
|
4
|
+
Gem::Specification.new do |gem|
|
5
|
+
gem.authors = ["Ben Ellis"]
|
6
|
+
gem.email = ["bellis@on-site.com"]
|
7
|
+
gem.description = %q{Ruby wrapper for OpenCNAM}
|
8
|
+
gem.summary = %q{Do reverse phone lookups via OpenCNAM ( http://www.opencnam.com/ )}
|
9
|
+
gem.homepage = "https://github.com/on-site/caller_id"
|
10
|
+
|
11
|
+
gem.files = `git ls-files`.split($\)
|
12
|
+
gem.executables = gem.files.grep(%r{^bin/}).map{ |f| File.basename(f) }
|
13
|
+
gem.test_files = gem.files.grep(%r{^(test|spec|features)/})
|
14
|
+
gem.name = "caller_id"
|
15
|
+
gem.require_paths = ["lib"]
|
16
|
+
gem.version = CallerId::VERSION
|
17
|
+
|
18
|
+
gem.add_development_dependency "bundler", ">= 1.0.0"
|
19
|
+
gem.add_development_dependency "rspec", "~> 2.6"
|
20
|
+
gem.add_development_dependency "vcr", "~> 2.1"
|
21
|
+
gem.add_development_dependency "webmock", "1.8.6"
|
22
|
+
gem.add_development_dependency "timecop", "~> 0.3"
|
23
|
+
|
24
|
+
gem.add_dependency "json"
|
25
|
+
|
26
|
+
end
|
data/lib/caller_id.rb
ADDED
@@ -0,0 +1,38 @@
|
|
1
|
+
require "caller_id/version"
|
2
|
+
require "json"
|
3
|
+
require "open-uri"
|
4
|
+
|
5
|
+
module CallerId
|
6
|
+
# This class leverages the OpenCNAM service to lookup names from phone numbers.
|
7
|
+
# http://docs.opencnam.com/guide.html
|
8
|
+
class ReversePhoneLookup
|
9
|
+
def initialize number, user = nil, key = nil
|
10
|
+
@phone_number = number.to_s.gsub(/\D/,'').gsub(/\A1/,'') # strip spaces, punctuation, and leading 1s.
|
11
|
+
# TODO: consider raising a specific Exception here.
|
12
|
+
raise "invalid format" if @phone_number.nil? || @phone_number.strip.empty? || @phone_number.length != 10
|
13
|
+
if user && key
|
14
|
+
@api_user = user
|
15
|
+
@api_key = key
|
16
|
+
end
|
17
|
+
end
|
18
|
+
# Derived from https://github.com/EricR/sinatra-cnam-lookup/blob/master/cnam_checker.rb
|
19
|
+
def lookup
|
20
|
+
begin
|
21
|
+
response = open(get_url)
|
22
|
+
rescue OpenURI::HTTPError
|
23
|
+
return {:not_found => true}
|
24
|
+
end
|
25
|
+
return {:retry => true, :now => Time.now} if response.status[0] == "202"
|
26
|
+
return if response.status[0] != "200"
|
27
|
+
json_response = response.read
|
28
|
+
results = JSON.parse(json_response)
|
29
|
+
{:cnam => results["cnam"].strip, :number => results["number"]}
|
30
|
+
end
|
31
|
+
private
|
32
|
+
def get_url
|
33
|
+
url = "https://api.opencnam.com/v1/phone/#{@phone_number}?format=json"
|
34
|
+
url += "&username=#{@api_user}&api_key=#{@api_key}" if @api_user && @api_key
|
35
|
+
url
|
36
|
+
end
|
37
|
+
end
|
38
|
+
end
|
@@ -0,0 +1,41 @@
|
|
1
|
+
require 'spec_helper'
|
2
|
+
require 'caller_id'
|
3
|
+
|
4
|
+
describe CallerId do
|
5
|
+
let(:white_house_num) { "2024561414" }
|
6
|
+
let(:white_house_lookup_response) { {:cnam => "WHITE HOUSE SWI", :number => white_house_num} }
|
7
|
+
let(:kevin_brower) { "(510) 883-1085"}
|
8
|
+
let(:not_found) { "5034871414" }
|
9
|
+
let(:now) { Time.now }
|
10
|
+
describe "#lookup" do
|
11
|
+
use_vcr_cassette
|
12
|
+
it "returns name" do
|
13
|
+
CallerId::ReversePhoneLookup.new(white_house_num).lookup.should == white_house_lookup_response
|
14
|
+
end
|
15
|
+
it "ignores leading 1" do
|
16
|
+
CallerId::ReversePhoneLookup.new("1#{white_house_num}").lookup.should == white_house_lookup_response
|
17
|
+
end
|
18
|
+
it "ignores anything other than numbers" do
|
19
|
+
CallerId::ReversePhoneLookup.new("+1(#{white_house_num})...").lookup.should == white_house_lookup_response
|
20
|
+
end
|
21
|
+
end
|
22
|
+
describe "#lookup uncached number" do
|
23
|
+
use_vcr_cassette
|
24
|
+
it "returns retry" do
|
25
|
+
Timecop.freeze(now) do
|
26
|
+
CallerId::ReversePhoneLookup.new(kevin_brower).lookup.should == {:retry => true, :now => now}
|
27
|
+
end
|
28
|
+
end
|
29
|
+
end
|
30
|
+
describe "#lookup unlisted number" do
|
31
|
+
use_vcr_cassette
|
32
|
+
it "returns not found" do
|
33
|
+
CallerId::ReversePhoneLookup.new(not_found).lookup.should == {:not_found => true}
|
34
|
+
end
|
35
|
+
end
|
36
|
+
describe "#lookup invalid number" do
|
37
|
+
it "raises an error before even trying the lookup" do
|
38
|
+
expect { CallerId::ReversePhoneLookup.new("foo") }.to raise_error
|
39
|
+
end
|
40
|
+
end
|
41
|
+
end
|
@@ -0,0 +1,34 @@
|
|
1
|
+
---
|
2
|
+
http_interactions:
|
3
|
+
- request:
|
4
|
+
method: get
|
5
|
+
uri: https://api.opencnam.com/v1/phone/2024561414?format=json
|
6
|
+
body:
|
7
|
+
encoding: US-ASCII
|
8
|
+
string: ""
|
9
|
+
headers:
|
10
|
+
Accept:
|
11
|
+
- "*/*"
|
12
|
+
User-Agent:
|
13
|
+
- Ruby
|
14
|
+
response:
|
15
|
+
status:
|
16
|
+
code: 200
|
17
|
+
message: OK
|
18
|
+
headers:
|
19
|
+
Content-Type:
|
20
|
+
- application/json; charset=utf-8
|
21
|
+
Date:
|
22
|
+
- Mon, 18 Jun 2012 10:19:24 GMT
|
23
|
+
Server:
|
24
|
+
- gunicorn/0.14.2
|
25
|
+
Content-Length:
|
26
|
+
- "51"
|
27
|
+
Connection:
|
28
|
+
- keep-alive
|
29
|
+
body:
|
30
|
+
encoding: US-ASCII
|
31
|
+
string: "{\"cnam\": \"WHITE HOUSE SWI\", \"number\": \"2024561414\"}"
|
32
|
+
http_version:
|
33
|
+
recorded_at: Mon, 18 Jun 2012 10:19:25 GMT
|
34
|
+
recorded_with: VCR 2.2.2
|
@@ -0,0 +1,34 @@
|
|
1
|
+
---
|
2
|
+
http_interactions:
|
3
|
+
- request:
|
4
|
+
method: get
|
5
|
+
uri: https://api.opencnam.com/v1/phone/5108831085?format=json
|
6
|
+
body:
|
7
|
+
encoding: US-ASCII
|
8
|
+
string: ""
|
9
|
+
headers:
|
10
|
+
Accept:
|
11
|
+
- "*/*"
|
12
|
+
User-Agent:
|
13
|
+
- Ruby
|
14
|
+
response:
|
15
|
+
status:
|
16
|
+
code: 202
|
17
|
+
message: ACCEPTED
|
18
|
+
headers:
|
19
|
+
Content-Type:
|
20
|
+
- text/html; charset=utf-8
|
21
|
+
Date:
|
22
|
+
- Fri, 15 Jun 2012 15:09:25 GMT
|
23
|
+
Server:
|
24
|
+
- gunicorn/0.14.2
|
25
|
+
Content-Length:
|
26
|
+
- '86'
|
27
|
+
Connection:
|
28
|
+
- keep-alive
|
29
|
+
body:
|
30
|
+
encoding: US-ASCII
|
31
|
+
string: Currently running a lookup for phone '5108831085'. Please check back in a few seconds.
|
32
|
+
http_version:
|
33
|
+
recorded_at: Fri, 15 Jun 2012 15:09:25 GMT
|
34
|
+
recorded_with: VCR 2.2.2
|
@@ -0,0 +1,34 @@
|
|
1
|
+
---
|
2
|
+
http_interactions:
|
3
|
+
- request:
|
4
|
+
method: get
|
5
|
+
uri: https://api.opencnam.com/v1/phone/5034871414?format=json
|
6
|
+
body:
|
7
|
+
encoding: US-ASCII
|
8
|
+
string: ""
|
9
|
+
headers:
|
10
|
+
Accept:
|
11
|
+
- "*/*"
|
12
|
+
User-Agent:
|
13
|
+
- Ruby
|
14
|
+
response:
|
15
|
+
status:
|
16
|
+
code: 404
|
17
|
+
message: NOT FOUND
|
18
|
+
headers:
|
19
|
+
Content-Type:
|
20
|
+
- text/html; charset=utf-8
|
21
|
+
Date:
|
22
|
+
- Mon, 18 Jun 2012 10:19:27 GMT
|
23
|
+
Server:
|
24
|
+
- gunicorn/0.14.2
|
25
|
+
Content-Length:
|
26
|
+
- "0"
|
27
|
+
Connection:
|
28
|
+
- keep-alive
|
29
|
+
body:
|
30
|
+
encoding: US-ASCII
|
31
|
+
string: ""
|
32
|
+
http_version:
|
33
|
+
recorded_at: Mon, 18 Jun 2012 10:19:28 GMT
|
34
|
+
recorded_with: VCR 2.2.2
|
data/spec/spec_helper.rb
ADDED
metadata
ADDED
@@ -0,0 +1,129 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: caller_id
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.0.1
|
5
|
+
prerelease:
|
6
|
+
platform: ruby
|
7
|
+
authors:
|
8
|
+
- Ben Ellis
|
9
|
+
autorequire:
|
10
|
+
bindir: bin
|
11
|
+
cert_chain: []
|
12
|
+
date: 2012-06-18 00:00:00.000000000Z
|
13
|
+
dependencies:
|
14
|
+
- !ruby/object:Gem::Dependency
|
15
|
+
name: bundler
|
16
|
+
requirement: &2153249760 !ruby/object:Gem::Requirement
|
17
|
+
none: false
|
18
|
+
requirements:
|
19
|
+
- - ! '>='
|
20
|
+
- !ruby/object:Gem::Version
|
21
|
+
version: 1.0.0
|
22
|
+
type: :development
|
23
|
+
prerelease: false
|
24
|
+
version_requirements: *2153249760
|
25
|
+
- !ruby/object:Gem::Dependency
|
26
|
+
name: rspec
|
27
|
+
requirement: &2153249140 !ruby/object:Gem::Requirement
|
28
|
+
none: false
|
29
|
+
requirements:
|
30
|
+
- - ~>
|
31
|
+
- !ruby/object:Gem::Version
|
32
|
+
version: '2.6'
|
33
|
+
type: :development
|
34
|
+
prerelease: false
|
35
|
+
version_requirements: *2153249140
|
36
|
+
- !ruby/object:Gem::Dependency
|
37
|
+
name: vcr
|
38
|
+
requirement: &2153248560 !ruby/object:Gem::Requirement
|
39
|
+
none: false
|
40
|
+
requirements:
|
41
|
+
- - ~>
|
42
|
+
- !ruby/object:Gem::Version
|
43
|
+
version: '2.1'
|
44
|
+
type: :development
|
45
|
+
prerelease: false
|
46
|
+
version_requirements: *2153248560
|
47
|
+
- !ruby/object:Gem::Dependency
|
48
|
+
name: webmock
|
49
|
+
requirement: &2153247980 !ruby/object:Gem::Requirement
|
50
|
+
none: false
|
51
|
+
requirements:
|
52
|
+
- - =
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: 1.8.6
|
55
|
+
type: :development
|
56
|
+
prerelease: false
|
57
|
+
version_requirements: *2153247980
|
58
|
+
- !ruby/object:Gem::Dependency
|
59
|
+
name: timecop
|
60
|
+
requirement: &2153247400 !ruby/object:Gem::Requirement
|
61
|
+
none: false
|
62
|
+
requirements:
|
63
|
+
- - ~>
|
64
|
+
- !ruby/object:Gem::Version
|
65
|
+
version: '0.3'
|
66
|
+
type: :development
|
67
|
+
prerelease: false
|
68
|
+
version_requirements: *2153247400
|
69
|
+
- !ruby/object:Gem::Dependency
|
70
|
+
name: json
|
71
|
+
requirement: &2153246920 !ruby/object:Gem::Requirement
|
72
|
+
none: false
|
73
|
+
requirements:
|
74
|
+
- - ! '>='
|
75
|
+
- !ruby/object:Gem::Version
|
76
|
+
version: '0'
|
77
|
+
type: :runtime
|
78
|
+
prerelease: false
|
79
|
+
version_requirements: *2153246920
|
80
|
+
description: Ruby wrapper for OpenCNAM
|
81
|
+
email:
|
82
|
+
- bellis@on-site.com
|
83
|
+
executables: []
|
84
|
+
extensions: []
|
85
|
+
extra_rdoc_files: []
|
86
|
+
files:
|
87
|
+
- Gemfile
|
88
|
+
- Gemfile.lock
|
89
|
+
- LICENSE
|
90
|
+
- README.md
|
91
|
+
- Rakefile
|
92
|
+
- caller_id.gemspec
|
93
|
+
- lib/caller_id.rb
|
94
|
+
- lib/caller_id/version.rb
|
95
|
+
- spec/caller_id_spec.rb
|
96
|
+
- spec/cassettes/CallerId/_lookup.yml
|
97
|
+
- spec/cassettes/CallerId/_lookup_uncached_number.yml
|
98
|
+
- spec/cassettes/CallerId/_lookup_unlisted_number.yml
|
99
|
+
- spec/spec_helper.rb
|
100
|
+
homepage: https://github.com/on-site/caller_id
|
101
|
+
licenses: []
|
102
|
+
post_install_message:
|
103
|
+
rdoc_options: []
|
104
|
+
require_paths:
|
105
|
+
- lib
|
106
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
107
|
+
none: false
|
108
|
+
requirements:
|
109
|
+
- - ! '>='
|
110
|
+
- !ruby/object:Gem::Version
|
111
|
+
version: '0'
|
112
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
113
|
+
none: false
|
114
|
+
requirements:
|
115
|
+
- - ! '>='
|
116
|
+
- !ruby/object:Gem::Version
|
117
|
+
version: '0'
|
118
|
+
requirements: []
|
119
|
+
rubyforge_project:
|
120
|
+
rubygems_version: 1.8.11
|
121
|
+
signing_key:
|
122
|
+
specification_version: 3
|
123
|
+
summary: Do reverse phone lookups via OpenCNAM ( http://www.opencnam.com/ )
|
124
|
+
test_files:
|
125
|
+
- spec/caller_id_spec.rb
|
126
|
+
- spec/cassettes/CallerId/_lookup.yml
|
127
|
+
- spec/cassettes/CallerId/_lookup_uncached_number.yml
|
128
|
+
- spec/cassettes/CallerId/_lookup_unlisted_number.yml
|
129
|
+
- spec/spec_helper.rb
|