adamhunter-contacts 1.1.16

Sign up to get free protection for your applications and to get access to all the features.
data/LICENSE ADDED
@@ -0,0 +1,10 @@
1
+ Copyright (c) 2006, Lucas Carlson, MOG
2
+ All rights reserved.
3
+
4
+ Redistribution and use in source and binary forms, with or without modification, are permitted provided that the following conditions are met:
5
+
6
+ Redistributions of source code must retain the above copyright notice, this list of conditions and the following disclaimer.
7
+ Redistributions in binary form must reproduce the above copyright notice, this list of conditions and the following disclaimer in the documentation and/or other materials provided with the distribution.
8
+ Neither the name of the Lucas Carlson nor the names of its contributors may be used to endorse or promote products derived from this software without specific prior written permission.
9
+ THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
10
+
data/README ADDED
@@ -0,0 +1,44 @@
1
+ == Welcome to Contacts
2
+
3
+ Contacts is a universal interface to grab contact list information from various providers including Hotmail, Gmail and Yahoo.
4
+
5
+ == Download
6
+
7
+ * gem install contacts
8
+ * http://rubyforge.org/projects/contacts
9
+ * svn co svn://rubyforge.org/var/svn/contacts
10
+
11
+ == Background
12
+
13
+ For a long time, the only way to get a list of contacts from your free online email accounts was with proprietary PHP scripts that would cost you $50. The act of grabbing that list is a simple matter of screen scrapping and this library gives you all the functionality you need. Thanks to the generosity of the highly popular Rails website MOG (http://mog.com) for allowing this library to be released open-source to the world. It is easy to extend this library to add new free email providers, so please contact the author if you would like to help.
14
+
15
+ == Usage
16
+
17
+ Contacts::Hotmail.new(login, password).contacts # => [["name", "foo@bar.com"], ["another name", "bow@wow.com"]]
18
+ Contacts::Yahoo.new(login, password).contacts
19
+ Contacts::Gmail.new(login, password).contacts
20
+
21
+ Contacts.new(:gmail, login, password).contacts
22
+ Contacts.new(:hotmail, login, password).contacts
23
+ Contacts.new(:yahoo, login, password).contacts
24
+
25
+ Contacts.guess(login, password).contacts
26
+
27
+ Notice there are three ways to use this library so that you can limit the use as much as you would like in your particular application. The Contacts.guess method will automatically concatenate all the address book contacts from each of the successful logins in the case that a username password works across multiple services.
28
+
29
+ == Examples
30
+
31
+ See the examples/ directory.
32
+
33
+ == Authors
34
+
35
+ * Lucas Carlson from MOG (mailto:lucas@rufy.com) - http://mog.com
36
+
37
+ == Contributors
38
+
39
+ * Britt Selvitelle from Twitter (mailto:anotherbritt@gmail.com) - http://twitter.com
40
+ * Tony Targonski from GigPark (mailto:tony@gigpark.com) - http://gigpark.com
41
+ * Waheed Barghouthi from Watwet (mailto:waheed.barghouthi@gmail.com) - http://watwet.com
42
+
43
+ This library is released under the terms of the BSD.
44
+
@@ -0,0 +1,91 @@
1
+ require 'rubygems'
2
+ require 'rake'
3
+ require 'rake/testtask'
4
+ require 'rake/rdoctask'
5
+ require 'rake/gempackagetask'
6
+ require 'rake/contrib/rubyforgepublisher'
7
+ require 'lib/contacts'
8
+
9
+ PKG_VERSION = Contacts::VERSION
10
+
11
+ PKG_FILES = FileList[
12
+ "lib/**/*", "bin/*", "test/**/*", "[A-Z]*", "Rakefile", "doc/**/*", "examples/**/*"
13
+ ] - ["test/accounts.yml"]
14
+
15
+ desc "Default Task"
16
+ task :default => [ :test ]
17
+
18
+ # Run the unit tests
19
+ desc "Run all unit tests"
20
+ Rake::TestTask.new("test") { |t|
21
+ t.libs << "lib"
22
+ t.pattern = 'test/*/*_test.rb'
23
+ t.verbose = true
24
+ }
25
+
26
+ # Make a console, useful when working on tests
27
+ desc "Generate a test console"
28
+ task :console do
29
+ verbose( false ) { sh "irb -I lib/ -r 'contacts'" }
30
+ end
31
+
32
+ # Genereate the RDoc documentation
33
+ desc "Create documentation"
34
+ Rake::RDocTask.new("doc") { |rdoc|
35
+ rdoc.title = "Contact List - ridiculously easy contact list information from various providers including Yahoo, Gmail, and Hotmail"
36
+ rdoc.rdoc_dir = 'doc'
37
+ rdoc.rdoc_files.include('README')
38
+ rdoc.rdoc_files.include('lib/**/*.rb')
39
+ }
40
+
41
+ # Genereate the package
42
+ spec = Gem::Specification.new do |s|
43
+
44
+ #### Basic information.
45
+
46
+ s.name = 'adamhunter-contacts'
47
+ s.version = PKG_VERSION
48
+ s.summary = <<-EOF
49
+ Ridiculously easy contact list information from various providers including Yahoo, Gmail, and Hotmail
50
+ EOF
51
+ s.description = <<-EOF
52
+ Ridiculously easy contact list information from various providers including Yahoo, Gmail, and Hotmail
53
+ EOF
54
+
55
+ #### Which files are to be included in this gem? Everything! (Except CVS directories.)
56
+
57
+ s.files = PKG_FILES
58
+
59
+ #### Load-time details: library and application (you will need one or both).
60
+
61
+ s.require_path = 'lib'
62
+ s.autorequire = 'contacts'
63
+
64
+ s.add_dependency('json', '>= 0.4.1')
65
+ s.add_dependency('gdata', '>= 1.1.0')
66
+ s.requirements << "A json parser, the gdata ruby gem"
67
+
68
+ #### Documentation and testing.
69
+
70
+ s.has_rdoc = true
71
+
72
+ #### Author and project details.
73
+
74
+ s.author = "Lucas Carlson"
75
+ s.email = "lucas@rufy.com"
76
+ s.homepage = "http://rubyforge.org/projects/contacts"
77
+ end
78
+
79
+ Rake::GemPackageTask.new(spec) do |pkg|
80
+ pkg.need_zip = true
81
+ pkg.need_tar = true
82
+ end
83
+
84
+ desc "Report code statistics (KLOCs, etc) from the application"
85
+ task :stats do
86
+ require 'code_statistics'
87
+ CodeStatistics.new(
88
+ ["Library", "lib"],
89
+ ["Units", "test"]
90
+ ).to_s
91
+ end
@@ -0,0 +1,12 @@
1
+ require File.dirname(__FILE__)+"/../lib/contacts"
2
+
3
+ login = ARGV[0]
4
+ password = ARGV[1]
5
+
6
+ Contacts::Gmail.new(login, password).contacts
7
+
8
+ Contacts.new(:gmail, login, password).contacts
9
+
10
+ Contacts.new("gmail", login, password).contacts
11
+
12
+ Contacts.guess(login, password).contacts
@@ -0,0 +1,14 @@
1
+ $:.unshift(File.dirname(__FILE__)+"/contacts/")
2
+
3
+ require 'rubygems'
4
+
5
+ gem 'gdata', '1.1.0'
6
+ require 'gdata'
7
+
8
+ require 'base'
9
+ require 'gmail'
10
+ require 'hotmail'
11
+ require 'yahoo'
12
+ require 'plaxo'
13
+
14
+ require 'gdatamail'
@@ -0,0 +1,210 @@
1
+ require "cgi"
2
+ require "net/http"
3
+ require "net/https"
4
+ require "uri"
5
+ require "zlib"
6
+ require "stringio"
7
+ require "thread"
8
+
9
+ class Contacts
10
+ TYPES = {}
11
+ VERSION = "1.1.16"
12
+
13
+ class Base
14
+ def initialize(login, password)
15
+ @login = login
16
+ @password = password
17
+ @connections = {}
18
+ connect
19
+ end
20
+
21
+ def connect
22
+ raise AuthenticationError, "Login and password must not be nil, login: #{@login.inspect}, password: #{@password.inspect}" if @login.nil? || @password.nil?
23
+ real_connect
24
+ end
25
+
26
+ def connected?
27
+ @cookies && !@cookies.empty?
28
+ end
29
+
30
+ def contacts(options = {})
31
+ return @contacts if @contacts
32
+ if connected?
33
+ url = URI.parse(contact_list_url)
34
+ http = open_http(url)
35
+ resp, data = http.get("#{url.path}?#{url.query}",
36
+ "Cookie" => @cookies
37
+ )
38
+
39
+ if resp.code_type != Net::HTTPOK
40
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
41
+ end
42
+
43
+ parse(data, options)
44
+ end
45
+ end
46
+
47
+ def login
48
+ @attempt ||= 0
49
+ @attempt += 1
50
+
51
+ if @attempt == 1
52
+ @login
53
+ else
54
+ if @login.include?("@#{domain}")
55
+ @login.sub("@#{domain}","")
56
+ else
57
+ "#{@login}@#{domain}"
58
+ end
59
+ end
60
+ end
61
+
62
+ def password
63
+ @password
64
+ end
65
+
66
+ private
67
+
68
+ def domain
69
+ @d ||= URI.parse(self.class.const_get(:URL)).host.sub(/^www\./,'')
70
+ end
71
+
72
+ def contact_list_url
73
+ self.class.const_get(:CONTACT_LIST_URL)
74
+ end
75
+
76
+ def address_book_url
77
+ self.class.const_get(:ADDRESS_BOOK_URL)
78
+ end
79
+
80
+ def open_http(url)
81
+ c = @connections[Thread.current.object_id] ||= {}
82
+ http = c["#{url.host}:#{url.port}"]
83
+ unless http
84
+ http = Net::HTTP.new(url.host, url.port)
85
+ if url.port == 443
86
+ http.use_ssl = true
87
+ http.verify_mode = OpenSSL::SSL::VERIFY_NONE
88
+ end
89
+ c["#{url.host}:#{url.port}"] = http
90
+ end
91
+ http.start unless http.started?
92
+ http
93
+ end
94
+
95
+ def parse_cookies(data, existing="")
96
+ return existing if data.nil?
97
+
98
+ cookies = existing.split(";").map{|i|i.split("=", 2).map{|j|j.strip}}.inject({}){|h,i|h[i[0]]=i[1];h}
99
+
100
+ data.gsub!(/ ?[\w]+=EXPIRED;/,'')
101
+ data.gsub!(/ ?expires=(.*?, .*?)[;,$]/i, ';')
102
+ data.gsub!(/ ?(domain|path)=[\S]*?[;,$]/i,';')
103
+ data.gsub!(/[,;]?\s*(secure|httponly)/i,'')
104
+ data.gsub!(/(;\s*){2,}/,', ')
105
+ data.gsub!(/(,\s*){2,}/,', ')
106
+ data.sub!(/^,\s*/,'')
107
+ data.sub!(/\s*,$/,'')
108
+
109
+ data.split(", ").map{|t|t.to_s.split(";").first}.each do |data|
110
+ k, v = data.split("=", 2).map{|j|j.strip}
111
+ if cookies[k] && v.empty?
112
+ cookies.delete(k)
113
+ elsif v && !v.empty?
114
+ cookies[k] = v
115
+ end
116
+ end
117
+
118
+ cookies.map{|k,v| "#{k}=#{v}"}.join("; ")
119
+ end
120
+
121
+ def remove_cookie(cookie, cookies)
122
+ parse_cookies("#{cookie}=", cookies)
123
+ end
124
+
125
+ def post(url, postdata, cookies="", referer="")
126
+ url = URI.parse(url)
127
+ http = open_http(url)
128
+ resp, data = http.post(url.path, postdata,
129
+ "User-Agent" => "Mozilla/5.0 (Macintosh; U; Intel Mac OS X; en-US; rv:1.8.1) Gecko/20061010 Firefox/2.0",
130
+ "Accept-Encoding" => "gzip",
131
+ "Cookie" => cookies,
132
+ "Referer" => referer,
133
+ "Content-Type" => 'application/x-www-form-urlencoded'
134
+ )
135
+ data = uncompress(resp, data)
136
+ cookies = parse_cookies(resp.response['set-cookie'], cookies)
137
+ forward = resp.response['Location']
138
+ forward ||= (data =~ /<meta.*?url='([^']+)'/ ? CGI.unescapeHTML($1) : nil)
139
+ if (not forward.nil?) && URI.parse(forward).host.nil?
140
+ forward = url.scheme.to_s + "://" + url.host.to_s + forward
141
+ end
142
+ return data, resp, cookies, forward
143
+ end
144
+
145
+ def get(url, cookies="", referer="")
146
+ url = URI.parse(url)
147
+ http = open_http(url)
148
+ resp, data = http.get("#{url.path}?#{url.query}",
149
+ "User-Agent" => "Mozilla/5.0 (Macintosh; U; Intel Mac OS X; en-US; rv:1.8.1) Gecko/20061010 Firefox/2.0",
150
+ "Accept-Encoding" => "gzip",
151
+ "Cookie" => cookies,
152
+ "Referer" => referer
153
+ )
154
+ data = uncompress(resp, data)
155
+ cookies = parse_cookies(resp.response['set-cookie'], cookies)
156
+ forward = resp.response['Location']
157
+ if (not forward.nil?) && URI.parse(forward).host.nil?
158
+ forward = url.scheme.to_s + "://" + url.host.to_s + forward
159
+ end
160
+ return data, resp, cookies, forward
161
+ end
162
+
163
+ def uncompress(resp, data)
164
+ case resp.response['content-encoding']
165
+ when 'gzip':
166
+ gz = Zlib::GzipReader.new(StringIO.new(data))
167
+ data = gz.read
168
+ gz.close
169
+ resp.response['content-encoding'] = nil
170
+ # FIXME: Not sure what Hotmail was feeding me with their 'deflate',
171
+ # but the headers definitely were not right
172
+ when 'deflate':
173
+ data = Zlib::Inflate.inflate(data)
174
+ resp.response['content-encoding'] = nil
175
+ end
176
+
177
+ data
178
+ end
179
+ end
180
+
181
+ class ContactsError < StandardError
182
+ end
183
+
184
+ class AuthenticationError < ContactsError
185
+ end
186
+
187
+ class ConnectionError < ContactsError
188
+ end
189
+
190
+ class TypeNotFound < ContactsError
191
+ end
192
+
193
+ def self.new(type, login, password)
194
+ if TYPES.include?(type.to_s.intern)
195
+ TYPES[type.to_s.intern].new(login, password)
196
+ else
197
+ raise TypeNotFound, "#{type.inspect} is not a valid type, please choose one of the following: #{TYPES.keys.inspect}"
198
+ end
199
+ end
200
+
201
+ def self.guess(login, password)
202
+ TYPES.inject([]) do |a, t|
203
+ begin
204
+ a + t[1].new(login, password).contacts
205
+ rescue AuthenticationError
206
+ a
207
+ end
208
+ end.uniq
209
+ end
210
+ end
@@ -0,0 +1,33 @@
1
+ class Contacts
2
+ class GDataMail < Base
3
+
4
+ CONTACTS_SCOPE = 'http://www.google.com/m8/feeds/'
5
+ CONTACTS_FEED = CONTACTS_SCOPE + 'contacts/default/full/'
6
+
7
+ def contacts
8
+ return @contacts if @contacts
9
+ real_connect
10
+ end
11
+
12
+ def real_connect
13
+ @client = GData::Client::Contacts.new
14
+ @client.clientlogin(@login, @password)
15
+
16
+ feed = @client.get(CONTACTS_FEED).to_xml
17
+
18
+ @contacts = []
19
+ feed.elements.each('entry') do |entry|
20
+ title = entry.elements['title'].text
21
+ email = nil
22
+ entry.elements.each('gd:email') do |e|
23
+ if e.attribute('primary')
24
+ email = e.attribute('address').value
25
+ end
26
+ end
27
+ @contacts << [title, email] unless email.nil?
28
+ end
29
+ @contacts
30
+ end
31
+
32
+ end
33
+ end
@@ -0,0 +1,93 @@
1
+ # Use ActiveSupport's version of JSON if available
2
+ if Object.const_defined?('ActiveSupport') && ActiveSupport.const_defined?('JSON')
3
+ class JSON
4
+ def self.parse(i)
5
+ ActiveSupport::JSON.decode(i)
6
+ end
7
+ end
8
+ else
9
+ require 'json/add/rails'
10
+ end
11
+
12
+ class Contacts
13
+ class Gmail < Base
14
+ URL = "https://mail.google.com/mail/"
15
+ LOGIN_URL = "https://www.google.com/accounts/ServiceLoginAuth"
16
+ LOGIN_REFERER_URL = "https://www.google.com/accounts/ServiceLogin?service=mail&passive=true&rm=false&continue=http%3A%2F%2Fmail.google.com%2Fmail%3Fui%3Dhtml%26zy%3Dl&ltmpl=yj_blanco&ltmplcache=2&hl=en"
17
+ CONTACT_LIST_URL = "https://mail.google.com/mail/contacts/data/contacts?thumb=true&show=ALL&enums=true&psort=Name&max=10000&out=js&rf=&jsx=true"
18
+ PROTOCOL_ERROR = "Gmail has changed its protocols, please upgrade this library first. If that does not work, dive into the code and submit a patch at http://github.com/cardmagic/contacts"
19
+
20
+ def real_connect
21
+ postdata = "ltmpl=yj_blanco"
22
+ postdata += "&continue=%s" % CGI.escape(URL)
23
+ postdata += "&ltmplcache=2"
24
+ postdata += "&service=mail"
25
+ postdata += "&rm=false"
26
+ postdata += "&ltmpl=yj_blanco"
27
+ postdata += "&hl=en"
28
+ postdata += "&Email=%s" % CGI.escape(login)
29
+ postdata += "&Passwd=%s" % CGI.escape(password)
30
+ postdata += "&rmShown=1"
31
+ postdata += "&null=Sign+in"
32
+
33
+ time = Time.now.to_i
34
+ time_past = Time.now.to_i - 8 - rand(12)
35
+ cookie = "GMAIL_LOGIN=T#{time_past}/#{time_past}/#{time}"
36
+
37
+ data, resp, cookies, forward, old_url = post(LOGIN_URL, postdata, cookie, LOGIN_REFERER_URL) + [LOGIN_URL]
38
+
39
+ cookies = remove_cookie("GMAIL_LOGIN", cookies)
40
+
41
+ if data.index("Username and password do not match") || data.index("New to Gmail? It's free and easy")
42
+ raise AuthenticationError, "Username and password do not match"
43
+ elsif data.index("The username or password you entered is incorrect")
44
+ raise AuthenticationError, "Username and password do not match"
45
+ elsif data.index("Required field must not be blank")
46
+ raise AuthenticationError, "Login and password must not be blank"
47
+ elsif data.index("errormsg_0_logincaptcha")
48
+ raise AuthenticationError, "Captcha error"
49
+ elsif data.index("Invalid request")
50
+ raise ConnectionError, PROTOCOL_ERROR
51
+ elsif cookies == ""
52
+ raise ConnectionError, PROTOCOL_ERROR
53
+ end
54
+
55
+ cookies = remove_cookie("LSID", cookies)
56
+ cookies = remove_cookie("GV", cookies)
57
+
58
+ @cookies = cookies
59
+ end
60
+
61
+ private
62
+
63
+ def parse(data, options)
64
+ data.gsub!(/^while \(true\); &&&START&&&/, '')
65
+ data.gsub!(/ &&&END&&&$/, '')
66
+ data.gsub!(/\t/, ' ') # tabs in the note field cause errors with JSON.parse
67
+ data.gsub!(/[\t\x00-\x1F]/, " ") # strip control characters
68
+
69
+ @contacts = JSON.parse(data)['Body']['Contacts'] || {}
70
+
71
+ # Determine in which format to return the data.
72
+
73
+ # Return the full JSON Hash.
74
+ return @contacts if(options[:details])
75
+
76
+ # Default format.
77
+ # ['Name', 'Email1', 'Email2', ...]
78
+ if @contacts != nil
79
+ @contacts = @contacts.delete_if {|c| c["Emails"].nil?}.map do |c|
80
+ name, emails = c.values_at "Name", "Emails"
81
+ # emails are returned in a form of
82
+ # [{"Address"=>"home.email@gmail.com"}, {"Type"=>{"Id"=>"WORK"}, "Address"=>"work.email@gmail.com"}]
83
+ emails = emails.collect{|a| a.values_at("Address")}
84
+ [name, emails].flatten
85
+ end
86
+ else
87
+ []
88
+ end
89
+ end
90
+ end
91
+
92
+ TYPES[:gmail] = Gmail
93
+ end
@@ -0,0 +1,125 @@
1
+ class Contacts
2
+ class Hotmail < Base
3
+ URL = "https://login.live.com/login.srf?id=2"
4
+ OLD_CONTACT_LIST_URL = "http://%s/cgi-bin/addresses"
5
+ NEW_CONTACT_LIST_URL = "http://%s/mail/GetContacts.aspx"
6
+ CONTACT_LIST_URL = "http://mpeople.live.com/default.aspx?pg=0"
7
+ COMPOSE_URL = "http://%s/cgi-bin/compose?"
8
+ PROTOCOL_ERROR = "Hotmail has changed its protocols, please upgrade this library first. If that does not work, report this error at http://rubyforge.org/forum/?group_id=2693"
9
+ PWDPAD = "IfYouAreReadingThisYouHaveTooMuchFreeTime"
10
+ MAX_HTTP_THREADS = 8
11
+
12
+ def real_connect
13
+ data, resp, cookies, forward = get(URL)
14
+ old_url = URL
15
+ until forward.nil?
16
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
17
+ end
18
+
19
+ postdata = "PPSX=%s&PwdPad=%s&login=%s&passwd=%s&LoginOptions=2&PPFT=%s" % [
20
+ CGI.escape(data.split("><").grep(/PPSX/).first[/=\S+$/][2..-3]),
21
+ PWDPAD[0...(PWDPAD.length-@password.length)],
22
+ CGI.escape(login),
23
+ CGI.escape(password),
24
+ CGI.escape(data.split("><").grep(/PPFT/).first[/=\S+$/][2..-3])
25
+ ]
26
+
27
+ form_url = data.split("><").grep(/form/).first.split[5][8..-2]
28
+ data, resp, cookies, forward = post(form_url, postdata, cookies)
29
+
30
+ if data.index("The e-mail address or password is incorrect")
31
+ raise AuthenticationError, "Username and password do not match"
32
+ elsif data != ""
33
+ raise AuthenticationError, "Required field must not be blank"
34
+ elsif cookies == ""
35
+ raise ConnectionError, PROTOCOL_ERROR
36
+ end
37
+
38
+ old_url = form_url
39
+
40
+ until forward.nil?
41
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
42
+ end
43
+
44
+ data, resp, cookies, forward = get("http://mail.live.com/mail", cookies)
45
+ until forward.nil?
46
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
47
+ end
48
+
49
+ @domain = URI.parse(old_url).host
50
+ @cookies = cookies
51
+ rescue AuthenticationError => m
52
+ if @attempt == 1
53
+ retry
54
+ else
55
+ raise m
56
+ end
57
+ end
58
+
59
+ def contacts(options = {})
60
+ if connected?
61
+ url = URI.parse(contact_list_url)
62
+ data, resp, cookies, forward = get( contact_list_url, @cookies )
63
+
64
+ if resp.code_type != Net::HTTPOK
65
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
66
+ end
67
+
68
+ @contacts = []
69
+ build_contacts = []
70
+ go = true
71
+ index = 0
72
+
73
+ while(go) do
74
+ go = false
75
+ url = URI.parse(get_contact_list_url(index))
76
+ http = open_http(url)
77
+ resp, data = http.get(get_contact_list_url(index), "Cookie" => @cookies)
78
+
79
+ email_match_text_beginning = Regexp.escape("http://m.mail.live.com/?rru=compose&amp;to=")
80
+ email_match_text_end = Regexp.escape("&amp;")
81
+
82
+ raw_html = resp.body.grep(/(?:e|dn)lk[0-9]+/)
83
+ raw_html.delete_at 0
84
+ raw_html.inject do |memo, row|
85
+ c_info = row.match(/(e|dn)lk([0-9])+/)
86
+
87
+ # Same contact, or different?
88
+ build_contacts << [] if memo != c_info[2]
89
+
90
+ # Grab info
91
+ case c_info[1]
92
+ when "e" # Email
93
+ build_contacts.last[1] = row.match(/#{email_match_text_beginning}(.*)#{email_match_text_end}/)[1]
94
+ when "dn" # Name
95
+ build_contacts.last[0] = row.match(/<a[^>]*>(.+)<\/a>/)[1]
96
+ end
97
+
98
+ # Set memo to contact id
99
+ c_info[2]
100
+ end
101
+
102
+ go = resp.body.include?("Next page")
103
+ index += 1
104
+ end
105
+
106
+ build_contacts.each do |contact|
107
+ unless contact[1].nil?
108
+ # Only return contacts with email addresses
109
+ contact[1] = CGI::unescape(contact[1])
110
+ @contacts << contact
111
+ end
112
+ end
113
+ return @contacts
114
+ end
115
+ end
116
+
117
+ def get_contact_list_url(index)
118
+ "http://mpeople.live.com/default.aspx?pg=#{index}"
119
+ end
120
+
121
+ private
122
+
123
+ TYPES[:hotmail] = Hotmail
124
+ end
125
+ end
@@ -0,0 +1,130 @@
1
+ require 'rexml/document'
2
+
3
+ class Contacts
4
+ class Plaxo < Base
5
+ URL = "http://www.plaxo.com/"
6
+ LOGIN_URL = "https://www.plaxo.com/signin"
7
+ ADDRESS_BOOK_URL = "http://www.plaxo.com/po3/?module=ab&operation=viewFull&mode=normal"
8
+ CONTACT_LIST_URL = "http://www.plaxo.com/axis/soap/contact?_action=getContacts&_format=xml"
9
+ PROTOCOL_ERROR = "Plaxo has changed its protocols, please upgrade this library first. If that does not work, dive into the code and submit a patch at http://github.com/cardmagic/contacts"
10
+
11
+ def real_connect
12
+
13
+ end # real_connect
14
+
15
+ def contacts
16
+ getdata = "&authInfo.authByEmail.email=%s" % CGI.escape(login)
17
+ getdata += "&authInfo.authByEmail.password=%s" % CGI.escape(password)
18
+ data, resp, cookies, forward = get(CONTACT_LIST_URL + getdata)
19
+
20
+ if resp.code_type != Net::HTTPOK
21
+ raise ConnectionError, PROTOCOL_ERROR
22
+ end
23
+
24
+ parse data
25
+ end # contacts
26
+
27
+ private
28
+ def parse(data, options={})
29
+ doc = REXML::Document.new(data)
30
+ code = doc.elements['//response/code'].text
31
+
32
+ if code == '401'
33
+ raise AuthenticationError, "Username and password do not match"
34
+ elsif code == '200'
35
+ @contacts = []
36
+ doc.elements.each('//contact') do |cont|
37
+ name = if cont.elements['fullName']
38
+ cont.elements['fullName'].text
39
+ elsif cont.elements['displayName']
40
+ cont.elements['displayName'].text
41
+ end
42
+ email = if cont.elements['email1']
43
+ cont.elements['email1'].text
44
+ end
45
+ if name || email
46
+ @contacts << [name, email]
47
+ end
48
+ end
49
+ @contacts
50
+ else
51
+ raise ConnectionError, PROTOCOL_ERROR
52
+ end
53
+
54
+ end # parse
55
+
56
+ end # Plaxo
57
+
58
+ TYPES[:plaxo] = Plaxo
59
+
60
+ end # Contacts
61
+
62
+
63
+ # sample contacts responses
64
+ =begin
65
+ Bad email
66
+ =========
67
+ <?xml version="1.0" encoding="utf-8" ?>
68
+ <ns1:GetContactsResponse xmlns:ns1="Plaxo">
69
+ <response>
70
+ <code>401</code>
71
+ <subCode>1</subCode>
72
+ <message>User not found.</message>
73
+ </response>
74
+ </ns1:GetContactsResponse>
75
+
76
+
77
+ Bad password
78
+ ============
79
+ <?xml version="1.0" encoding="utf-8" ?>
80
+ <ns1:GetContactsResponse xmlns:ns1="Plaxo">
81
+ <response>
82
+ <code>401</code>
83
+ <subCode>4</subCode>
84
+ <message>Bad password or security token.</message>
85
+ </response>
86
+ </ns1:GetContactsResponse>
87
+
88
+
89
+ Success
90
+ =======
91
+ <?xml version="1.0" encoding="utf-8" ?>
92
+ <ns1:GetContactsResponse xmlns:ns1="Plaxo">
93
+
94
+ <response>
95
+ <code>200</code>
96
+ <message>OK</message>
97
+ <userId>77311236242</userId>
98
+ </response>
99
+
100
+ <contacts>
101
+
102
+ <contact>
103
+ <itemId>61312569</itemId>
104
+ <displayName>Joe Blow1</displayName>
105
+ <fullName>Joe Blow1</fullName>
106
+ <firstName>Joe</firstName>
107
+ <lastName>Blow1</lastName>
108
+ <homeEmail1>joeblow1@mailinator.com</homeEmail1>
109
+ <email1>joeblow1@mailinator.com</email1>
110
+ <folderId>5291351</folderId>
111
+ </contact>
112
+
113
+ <contact>
114
+ <itemId>61313159</itemId>
115
+ <displayName>Joe Blow2</displayName>
116
+ <fullName>Joe Blow2</fullName>
117
+ <firstName>Joe</firstName>
118
+ <lastName>Blow2</lastName>
119
+ <homeEmail1>joeblow2@mailinator.com</homeEmail1>
120
+ <email1>joeblow2@mailinator.com</email1>
121
+ <folderId>5291351</folderId>
122
+ </contact>
123
+
124
+ </contacts>
125
+
126
+ <totalCount>2</totalCount>
127
+ <editCounter>3</editCounter>
128
+
129
+ </ns1:GetContactsResponse>
130
+ =end
@@ -0,0 +1,82 @@
1
+ require 'csv'
2
+
3
+ class Contacts
4
+ class Yahoo < Base
5
+ URL = "http://mail.yahoo.com/"
6
+ LOGIN_URL = "https://login.yahoo.com/config/login"
7
+ ADDRESS_BOOK_URL = "http://address.mail.yahoo.com/?1&VPC=import_export"
8
+ CONTACT_LIST_URL = "http://address.yahoo.com/index.php?VPC=import_export&A=B&submit[action_export_yahoo]=Export%20Now"
9
+ PROTOCOL_ERROR = "Yahoo has changed its protocols, please upgrade this library first. If that does not work, dive into the code and submit a patch at http://github.com/cardmagic/contacts"
10
+
11
+ def real_connect
12
+ postdata = ".tries=2&.src=ym&.md5=&.hash=&.js=&.last=&promo=&.intl=us&.bypass="
13
+ postdata += "&.partner=&.u=4eo6isd23l8r3&.v=0&.challenge=gsMsEcoZP7km3N3NeI4mX"
14
+ postdata += "kGB7zMV&.yplus=&.emailCode=&pkg=&stepid=&.ev=&hasMsgr=1&.chkP=Y&."
15
+ postdata += "done=#{CGI.escape(URL)}&login=#{CGI.escape(login)}&passwd=#{CGI.escape(password)}"
16
+
17
+ data, resp, cookies, forward = post(LOGIN_URL, postdata)
18
+
19
+ if data.index("Invalid ID or password") || data.index("This ID is not yet taken")
20
+ raise AuthenticationError, "Username and password do not match"
21
+ elsif data.index("Sign in") && data.index("to Yahoo!")
22
+ raise AuthenticationError, "Required field must not be blank"
23
+ elsif !data.match(/uncompressed\/chunked/)
24
+ raise ConnectionError, PROTOCOL_ERROR
25
+ elsif cookies == ""
26
+ raise ConnectionError, PROTOCOL_ERROR
27
+ end
28
+
29
+ data, resp, cookies, forward = get(forward, cookies, LOGIN_URL)
30
+
31
+ if resp.code_type != Net::HTTPOK
32
+ raise ConnectionError, PROTOCOL_ERROR
33
+ end
34
+
35
+ @cookies = cookies
36
+ end
37
+
38
+ def contacts
39
+ return @contacts if @contacts
40
+ if connected?
41
+ # first, get the addressbook site with the new crumb parameter
42
+ url = URI.parse(address_book_url)
43
+ http = open_http(url)
44
+ resp, data = http.get("#{url.path}?#{url.query}",
45
+ "Cookie" => @cookies
46
+ )
47
+
48
+ if resp.code_type != Net::HTTPOK
49
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
50
+ end
51
+
52
+ crumb = data.to_s[/id="crumb2" value="(.*?)"/][19...-1]
53
+
54
+ # now proceed with the new ".crumb" parameter to get the csv data
55
+ url = URI.parse("#{contact_list_url}&.crumb=#{crumb}")
56
+ http = open_http(url)
57
+ resp, data = http.get("#{url.path}?#{url.query}",
58
+ "Cookie" => @cookies
59
+ )
60
+
61
+ if resp.code_type != Net::HTTPOK
62
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
63
+ end
64
+
65
+ parse data
66
+ end
67
+ end
68
+
69
+ private
70
+
71
+ def parse(data, options={})
72
+ data = CSV.parse(data)
73
+ col_names = data.shift
74
+ @contacts = data.map do |person|
75
+ [[person[0], person[1], person[2]].delete_if{|i|i.empty?}.join(" "), person[4]] unless person[4].empty?
76
+ end.compact
77
+ end
78
+
79
+ end
80
+
81
+ TYPES[:yahoo] = Yahoo
82
+ end
@@ -0,0 +1,40 @@
1
+ gmail:
2
+ username: <changeme>
3
+ password: <changeme>
4
+ contacts:
5
+ -
6
+ name: "FirstName1 LastName1"
7
+ email_address: "firstname1@example.com"
8
+ -
9
+ name: "FirstName2 LastName2"
10
+ email_address: "firstname2@example.com"
11
+ yahoo:
12
+ username: <changeme>
13
+ password: <changeme>
14
+ contacts:
15
+ -
16
+ name: "FirstName1 LastName1"
17
+ email_address: "firstname1@example.com"
18
+ -
19
+ name: "FirstName2 LastName2"
20
+ email_address: "firstname2@example.com"
21
+ hotmail:
22
+ username: <changeme>
23
+ password: <changeme>
24
+ contacts:
25
+ -
26
+ name: "FirstName1 LastName1"
27
+ email_address: "firstname1@example.com"
28
+ -
29
+ name: "FirstName2 LastName2"
30
+ email_address: "firstname2@example.com"
31
+ aol:
32
+ username: <changeme>
33
+ password: <changeme>
34
+ contacts:
35
+ -
36
+ name: "FirstName1 LastName1"
37
+ email_address: "firstname1@example.com"
38
+ -
39
+ name: "FirstName2 LastName2"
40
+ email_address: "firstname2@example.com"
@@ -0,0 +1,30 @@
1
+ dir = File.dirname(__FILE__)
2
+ $LOAD_PATH.unshift(dir + "/../lib/")
3
+ require 'test/unit'
4
+ require 'contacts'
5
+
6
+ class ContactImporterTestCase < Test::Unit::TestCase
7
+ # Add more helper methods to be used by all tests here...
8
+ def default_test
9
+ assert true
10
+ end
11
+ end
12
+
13
+ class TestAccounts
14
+ def self.[](type)
15
+ load[type]
16
+ end
17
+
18
+ def self.load(file = File.dirname(__FILE__) + "/accounts.yml")
19
+ raise "/test/accounts.yml file not found, please create, see /test/example_accounts.yml for information" unless File.exist?(file)
20
+
21
+ accounts = {}
22
+ YAML::load(File.open(file)).each do |type, contents|
23
+ contacts = contents["contacts"].collect {|contact| [contact["name"], contact["email_address"]]}
24
+ accounts[type.to_sym] = Account.new(type.to_sym, contents["username"], contents["password"], contacts)
25
+ end
26
+ accounts
27
+ end
28
+
29
+ Account = Struct.new :type, :username, :password, :contacts
30
+ end
@@ -0,0 +1,4 @@
1
+ dir = File.dirname(__FILE__)
2
+ Dir["#{dir}/**/*_test.rb"].each do |file|
3
+ require file
4
+ end
@@ -0,0 +1,39 @@
1
+ dir = File.dirname(__FILE__)
2
+ require "#{dir}/../test_helper"
3
+ require 'contacts'
4
+
5
+ class GmailContactImporterTest < ContactImporterTestCase
6
+ def setup
7
+ super
8
+ @account = TestAccounts[:gmail]
9
+ end
10
+
11
+ def test_successful_login
12
+ Contacts.new(:gmail, @account.username, @account.password)
13
+ end
14
+
15
+ def test_importer_fails_with_invalid_password
16
+ assert_raise(Contacts::AuthenticationError) do
17
+ Contacts.new(:gmail, @account.username, "wrong_password")
18
+ end
19
+ end
20
+
21
+ def test_importer_fails_with_blank_password
22
+ assert_raise(Contacts::AuthenticationError) do
23
+ Contacts.new(:gmail, @account.username, "")
24
+ end
25
+ end
26
+
27
+ def test_importer_fails_with_blank_username
28
+ assert_raise(Contacts::AuthenticationError) do
29
+ Contacts.new(:gmail, "", @account.password)
30
+ end
31
+ end
32
+
33
+ def test_fetch_contacts
34
+ contacts = Contacts.new(:gmail, @account.username, @account.password).contacts
35
+ @account.contacts.each do |contact|
36
+ assert contacts.include?(contact), "Could not find: #{contact.inspect} in #{contacts.inspect}"
37
+ end
38
+ end
39
+ end
@@ -0,0 +1,27 @@
1
+ dir = File.dirname(__FILE__)
2
+ require "#{dir}/../test_helper"
3
+ require 'contacts'
4
+
5
+ class HotmailContactImporterTest < ContactImporterTestCase
6
+ def setup
7
+ super
8
+ @account = TestAccounts[:hotmail]
9
+ end
10
+
11
+ def test_successful_login
12
+ Contacts.new(:hotmail, @account.username, @account.password)
13
+ end
14
+
15
+ def test_importer_fails_with_invalid_password
16
+ assert_raise(Contacts::AuthenticationError) do
17
+ Contacts.new(:hotmail, @account.username,"wrong_password")
18
+ end
19
+ end
20
+
21
+ def test_fetch_contacts
22
+ contacts = Contacts.new(:hotmail, @account.username, @account.password).contacts
23
+ @account.contacts.each do |contact|
24
+ assert contacts.include?(contact), "Could not find: #{contact.inspect} in #{contacts.inspect}"
25
+ end
26
+ end
27
+ end
@@ -0,0 +1,23 @@
1
+ dir = File.dirname(__FILE__)
2
+ require "#{dir}/../test_helper"
3
+
4
+ class TestAccountsTest < ContactImporterTestCase
5
+ def test_test_accounts_loads_data_from_example_accounts_file
6
+ account = TestAccounts.load(File.dirname(__FILE__) + "/../example_accounts.yml")[:gmail]
7
+
8
+ assert_equal :gmail, account.type
9
+ assert_equal "<changeme>", account.username
10
+ assert_equal "<changeme>", account.password
11
+ assert_equal [["FirstName1 LastName1", "firstname1@example.com"], ["FirstName2 LastName2", "firstname2@example.com"]], account.contacts
12
+ end
13
+
14
+ def test_test_accounts_blows_up_if_file_doesnt_exist
15
+ assert_raise(RuntimeError) do
16
+ TestAccounts.load("file_that_does_not_exist.yml")
17
+ end
18
+ end
19
+
20
+ def test_we_can_load_from_account_file
21
+ assert_not_nil TestAccounts[:gmail].username
22
+ end
23
+ end
@@ -0,0 +1,32 @@
1
+ dir = File.dirname(__FILE__)
2
+ require "#{dir}/../test_helper"
3
+ require 'contacts'
4
+
5
+ class YahooContactImporterTest < ContactImporterTestCase
6
+ def setup
7
+ super
8
+ @account = TestAccounts[:yahoo]
9
+ end
10
+
11
+ def test_successful_login
12
+ Contacts.new(:yahoo, @account.username, @account.password)
13
+ end
14
+
15
+ def test_importer_fails_with_invalid_password
16
+ assert_raise(Contacts::AuthenticationError) do
17
+ Contacts.new(:yahoo, @account.username,"wrong_password")
18
+ end
19
+ # run the "successful" login test to ensure we reset yahoo's failed login lockout counter
20
+ # See http://www.pivotaltracker.com/story/show/138210
21
+ assert_nothing_raised do
22
+ Contacts.new(:yahoo, @account.username, @account.password)
23
+ end
24
+ end
25
+
26
+ def test_fetch_contacts
27
+ contacts = Contacts.new(:yahoo, @account.username, @account.password).contacts
28
+ @account.contacts.each do |contact|
29
+ assert contacts.include?(contact), "Could not find: #{contact.inspect} in #{contacts.inspect}"
30
+ end
31
+ end
32
+ end
metadata ADDED
@@ -0,0 +1,91 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: adamhunter-contacts
3
+ version: !ruby/object:Gem::Version
4
+ version: 1.1.16
5
+ platform: ruby
6
+ authors:
7
+ - Lucas Carlson
8
+ autorequire: contacts
9
+ bindir: bin
10
+ cert_chain: []
11
+
12
+ date: 2009-10-12 00:00:00 -04:00
13
+ default_executable:
14
+ dependencies:
15
+ - !ruby/object:Gem::Dependency
16
+ name: json
17
+ type: :runtime
18
+ version_requirement:
19
+ version_requirements: !ruby/object:Gem::Requirement
20
+ requirements:
21
+ - - ">="
22
+ - !ruby/object:Gem::Version
23
+ version: 0.4.1
24
+ version:
25
+ - !ruby/object:Gem::Dependency
26
+ name: gdata
27
+ type: :runtime
28
+ version_requirement:
29
+ version_requirements: !ruby/object:Gem::Requirement
30
+ requirements:
31
+ - - ">="
32
+ - !ruby/object:Gem::Version
33
+ version: 1.1.0
34
+ version:
35
+ description: " Ridiculously easy contact list information from various providers including Yahoo, Gmail, and Hotmail\n"
36
+ email: lucas@rufy.com
37
+ executables: []
38
+
39
+ extensions: []
40
+
41
+ extra_rdoc_files: []
42
+
43
+ files:
44
+ - lib/contacts/base.rb
45
+ - lib/contacts/gdatamail.rb
46
+ - lib/contacts/gmail.rb
47
+ - lib/contacts/hotmail.rb
48
+ - lib/contacts/plaxo.rb
49
+ - lib/contacts/yahoo.rb
50
+ - lib/contacts.rb
51
+ - test/example_accounts.yml
52
+ - test/test_helper.rb
53
+ - test/test_suite.rb
54
+ - test/unit/gmail_contact_importer_test.rb
55
+ - test/unit/hotmail_contact_importer_test.rb
56
+ - test/unit/test_accounts_test.rb
57
+ - test/unit/yahoo_csv_contact_importer_test.rb
58
+ - LICENSE
59
+ - Rakefile
60
+ - README
61
+ - examples/grab_contacts.rb
62
+ has_rdoc: true
63
+ homepage: http://rubyforge.org/projects/contacts
64
+ licenses: []
65
+
66
+ post_install_message:
67
+ rdoc_options: []
68
+
69
+ require_paths:
70
+ - lib
71
+ required_ruby_version: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: "0"
76
+ version:
77
+ required_rubygems_version: !ruby/object:Gem::Requirement
78
+ requirements:
79
+ - - ">="
80
+ - !ruby/object:Gem::Version
81
+ version: "0"
82
+ version:
83
+ requirements:
84
+ - A json parser, the gdata ruby gem
85
+ rubyforge_project:
86
+ rubygems_version: 1.3.5
87
+ signing_key:
88
+ specification_version: 3
89
+ summary: Ridiculously easy contact list information from various providers including Yahoo, Gmail, and Hotmail
90
+ test_files: []
91
+