glennfu-contacts 1.2.1

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,47 @@
1
+ == Welcome to Contacts
2
+
3
+ Contacts is a universal interface to grab contact list information from various providers including Hotmail, AOL, Gmail, Plaxo and Yahoo.
4
+
5
+ == Download
6
+
7
+ * gem install contacts
8
+ * http://github.com/cardmagic/contacts
9
+ * git clone git://github.com/cardmagic/contacts.git
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
+ * Glenn Sidney from Glenn Fu (mailto:glenn@glennfu.com) - http://glennfu.com
43
+ * Brian McQuay from Onomojo (mailto:brian@onomojo.com) - http://onomojo.com
44
+ * Adam Hunter (mailto:adamhunter@me.com) - http://adamhunter.me/
45
+
46
+ This library is released under the terms of the BSD.
47
+
data/Rakefile ADDED
@@ -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.1')
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,153 @@
1
+ class Hash
2
+ def to_query_string
3
+ u = ERB::Util.method(:u)
4
+ map { |k, v|
5
+ u.call(k) + "=" + u.call(v)
6
+ }.join("&")
7
+ end
8
+ end
9
+
10
+ class Contacts
11
+ require 'hpricot'
12
+ require 'csv'
13
+ class Aol < Base
14
+ URL = "http://www.aol.com/"
15
+ LOGIN_URL = "https://my.screenname.aol.com/_cqr/login/login.psp"
16
+ LOGIN_REFERER_URL = "http://webmail.aol.com/"
17
+ LOGIN_REFERER_PATH = "sitedomain=sns.webmail.aol.com&lang=en&locale=us&authLev=0&uitype=mini&loginId=&redirType=js&xchk=false"
18
+ AOL_NUM = "29970-343" # this seems to change each time they change the protocol
19
+
20
+ CONTACT_LIST_URL = "http://webmail.aol.com/#{AOL_NUM}/aim-2/en-us/Lite/ContactList.aspx?folder=Inbox&showUserFolders=False"
21
+ CONTACT_LIST_CSV_URL = "http://webmail.aol.com/#{AOL_NUM}/aim-2/en-us/Lite/ABExport.aspx?command=all"
22
+ PROTOCOL_ERROR = "AOL 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"
23
+
24
+ def real_connect
25
+
26
+ postdata = {
27
+ "loginId" => login,
28
+ "password" => password,
29
+ "rememberMe" => "on",
30
+ "_sns_fg_color_" => "",
31
+ "_sns_err_color_" => "",
32
+ "_sns_link_color_" => "",
33
+ "_sns_width_" => "",
34
+ "_sns_height_" => "",
35
+ "offerId" => "mail-second-en-us",
36
+ "_sns_bg_color_" => "",
37
+ "sitedomain" => "sns.webmail.aol.com",
38
+ "regPromoCode" => "",
39
+ "mcState" => "initialized",
40
+ "uitype" => "std",
41
+ "siteId" => "",
42
+ "lang" => "en",
43
+ "locale" => "us",
44
+ "authLev" => "0",
45
+ "siteState" => "",
46
+ "isSiteStateEncoded" => "false",
47
+ "use_aam" => "0",
48
+ "seamless" => "novl",
49
+ "aolsubmit" => CGI.escape("Sign In"),
50
+ "idType" => "SN",
51
+ "usrd" => "",
52
+ "doSSL" => "",
53
+ "redirType" => "",
54
+ "xchk" => "false"
55
+ }
56
+
57
+ # Get this cookie and stick it in the form to confirm to Aol that your cookies work
58
+ data, resp, cookies, forward = get(URL)
59
+ postdata["stips"] = cookie_hash_from_string(cookies)["stips"]
60
+ postdata["tst"] = cookie_hash_from_string(cookies)["tst"]
61
+
62
+ data, resp, cookies, forward, old_url = get(LOGIN_REFERER_URL, cookies) + [URL]
63
+ until forward.nil?
64
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
65
+ end
66
+
67
+ data, resp, cookies, forward, old_url = get("#{LOGIN_URL}?#{LOGIN_REFERER_PATH}", cookies) + [LOGIN_REFERER_URL]
68
+ until forward.nil?
69
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
70
+ end
71
+
72
+ doc = Hpricot(data)
73
+ (doc/:input).each do |input|
74
+ postdata["usrd"] = input.attributes["value"] if input.attributes["name"] == "usrd"
75
+ end
76
+ # parse data for <input name="usrd" value="2726212" type="hidden"> and add it to the postdata
77
+
78
+ postdata["SNS_SC"] = cookie_hash_from_string(cookies)["SNS_SC"]
79
+ postdata["SNS_LDC"] = cookie_hash_from_string(cookies)["SNS_LDC"]
80
+ postdata["LTState"] = cookie_hash_from_string(cookies)["LTState"]
81
+ # raise data.inspect
82
+
83
+ data, resp, cookies, forward, old_url = post(LOGIN_URL, postdata.to_query_string, cookies, LOGIN_REFERER_URL) + [LOGIN_REFERER_URL]
84
+
85
+ until forward.nil?
86
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
87
+ end
88
+
89
+ if data.index("Invalid Screen Name or Password.")
90
+ raise AuthenticationError, "Username and password do not match"
91
+ elsif data.index("Required field must not be blank")
92
+ raise AuthenticationError, "Login and password must not be blank"
93
+ elsif data.index("errormsg_0_logincaptcha")
94
+ raise AuthenticationError, "Captcha error"
95
+ elsif data.index("Invalid request")
96
+ raise ConnectionError, PROTOCOL_ERROR
97
+ elsif cookies == ""
98
+ raise ConnectionError, PROTOCOL_ERROR
99
+ end
100
+
101
+ @cookies = cookies
102
+ end
103
+
104
+ def contacts
105
+ postdata = {
106
+ "file" => 'contacts',
107
+ "fileType" => 'csv'
108
+ }
109
+
110
+ return @contacts if @contacts
111
+ if connected?
112
+ data, resp, cookies, forward, old_url = get(CONTACT_LIST_URL, @cookies, CONTACT_LIST_URL) + [CONTACT_LIST_URL]
113
+
114
+ until forward.nil?
115
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
116
+ end
117
+
118
+ if resp.code_type != Net::HTTPOK
119
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
120
+ end
121
+
122
+ # parse data and grab <input name="user" value="8QzMPIAKs2" type="hidden">
123
+ doc = Hpricot(data)
124
+ (doc/:input).each do |input|
125
+ postdata["user"] = input.attributes["value"] if input.attributes["name"] == "user"
126
+ end
127
+
128
+ data, resp, cookies, forward, old_url = get(CONTACT_LIST_CSV_URL, @cookies, CONTACT_LIST_URL) + [CONTACT_LIST_URL]
129
+
130
+ until forward.nil?
131
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
132
+ end
133
+
134
+ if data.include?("error.gif")
135
+ raise AuthenticationError, "Account invalid"
136
+ end
137
+
138
+ parse data
139
+ end
140
+ end
141
+ private
142
+
143
+ def parse(data, options={})
144
+ data = CSV::Reader.parse(data)
145
+ col_names = data.shift
146
+ @contacts = data.map do |person|
147
+ ["#{person[0]} #{person[1]}", person[4]] if person[4] && !person[4].empty?
148
+ end.compact
149
+ end
150
+ end
151
+
152
+ TYPES[:aol] = Aol
153
+ end
@@ -0,0 +1,215 @@
1
+ require "cgi"
2
+ require "net/http"
3
+ require "net/https"
4
+ require "uri"
5
+ require "zlib"
6
+ require "stringio"
7
+ require "thread"
8
+ require "erb"
9
+
10
+ class Contacts
11
+ TYPES = {}
12
+ VERSION = "1.2.1"
13
+
14
+ class Base
15
+ def initialize(login, password)
16
+ @login = login
17
+ @password = password
18
+ @connections = {}
19
+ connect
20
+ end
21
+
22
+ def connect
23
+ raise AuthenticationError, "Login and password must not be nil, login: #{@login.inspect}, password: #{@password.inspect}" if @login.nil? || @login.empty? || @password.nil? || @password.empty?
24
+ real_connect
25
+ end
26
+
27
+ def connected?
28
+ @cookies && !@cookies.empty?
29
+ end
30
+
31
+ def contacts(options = {})
32
+ return @contacts if @contacts
33
+ if connected?
34
+ url = URI.parse(contact_list_url)
35
+ http = open_http(url)
36
+ resp, data = http.get("#{url.path}?#{url.query}",
37
+ "Cookie" => @cookies
38
+ )
39
+
40
+ if resp.code_type != Net::HTTPOK
41
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
42
+ end
43
+
44
+ parse(data, options)
45
+ end
46
+ end
47
+
48
+ def login
49
+ @attempt ||= 0
50
+ @attempt += 1
51
+
52
+ if @attempt == 1
53
+ @login
54
+ else
55
+ if @login.include?("@#{domain}")
56
+ @login.sub("@#{domain}","")
57
+ else
58
+ "#{@login}@#{domain}"
59
+ end
60
+ end
61
+ end
62
+
63
+ def password
64
+ @password
65
+ end
66
+
67
+ private
68
+
69
+ def domain
70
+ @d ||= URI.parse(self.class.const_get(:URL)).host.sub(/^www\./,'')
71
+ end
72
+
73
+ def contact_list_url
74
+ self.class.const_get(:CONTACT_LIST_URL)
75
+ end
76
+
77
+ def address_book_url
78
+ self.class.const_get(:ADDRESS_BOOK_URL)
79
+ end
80
+
81
+ def open_http(url)
82
+ c = @connections[Thread.current.object_id] ||= {}
83
+ http = c["#{url.host}:#{url.port}"]
84
+ unless http
85
+ http = Net::HTTP.new(url.host, url.port)
86
+ if url.port == 443
87
+ http.use_ssl = true
88
+ http.verify_mode = OpenSSL::SSL::VERIFY_NONE
89
+ end
90
+ c["#{url.host}:#{url.port}"] = http
91
+ end
92
+ http.start unless http.started?
93
+ http
94
+ end
95
+
96
+ def cookie_hash_from_string(cookie_string)
97
+ cookie_string.split(";").map{|i|i.split("=", 2).map{|j|j.strip}}.inject({}){|h,i|h[i[0]]=i[1];h}
98
+ end
99
+
100
+ def parse_cookies(data, existing="")
101
+ return existing if data.nil?
102
+
103
+ cookies = cookie_hash_from_string(existing)
104
+
105
+ data.gsub!(/ ?[\w]+=EXPIRED;/,'')
106
+ data.gsub!(/ ?expires=(.*?, .*?)[;,$]/i, ';')
107
+ data.gsub!(/ ?(domain|path)=[\S]*?[;,$]/i,';')
108
+ data.gsub!(/[,;]?\s*(secure|httponly)/i,'')
109
+ data.gsub!(/(;\s*){2,}/,', ')
110
+ data.gsub!(/(,\s*){2,}/,', ')
111
+ data.sub!(/^,\s*/,'')
112
+ data.sub!(/\s*,$/,'')
113
+
114
+ data.split(", ").map{|t|t.to_s.split(";").first}.each do |data|
115
+ k, v = data.split("=", 2).map{|j|j.strip}
116
+ if cookies[k] && v.empty?
117
+ cookies.delete(k)
118
+ elsif v && !v.empty?
119
+ cookies[k] = v
120
+ end
121
+ end
122
+
123
+ cookies.map{|k,v| "#{k}=#{v}"}.join("; ")
124
+ end
125
+
126
+ def remove_cookie(cookie, cookies)
127
+ parse_cookies("#{cookie}=", cookies)
128
+ end
129
+
130
+ def post(url, postdata, cookies="", referer="")
131
+ url = URI.parse(url)
132
+ http = open_http(url)
133
+ resp, data = http.post(url.path, postdata,
134
+ "User-Agent" => "Mozilla/5.0 (Macintosh; U; Intel Mac OS X; en-US; rv:1.8.1) Gecko/20061010 Firefox/2.0",
135
+ "Accept-Encoding" => "gzip",
136
+ "Cookie" => cookies,
137
+ "Referer" => referer,
138
+ "Content-Type" => 'application/x-www-form-urlencoded'
139
+ )
140
+ data = uncompress(resp, data)
141
+ cookies = parse_cookies(resp.response['set-cookie'], cookies)
142
+ forward = resp.response['Location']
143
+ forward ||= (data =~ /<meta.*?url='([^']+)'/ ? CGI.unescapeHTML($1) : nil)
144
+ if (not forward.nil?) && URI.parse(forward).host.nil?
145
+ forward = url.scheme.to_s + "://" + url.host.to_s + forward
146
+ end
147
+ return data, resp, cookies, forward
148
+ end
149
+
150
+ def get(url, cookies="", referer="")
151
+ url = URI.parse(url)
152
+ http = open_http(url)
153
+ resp, data = http.get("#{url.path}?#{url.query}",
154
+ "User-Agent" => "Mozilla/5.0 (Macintosh; U; Intel Mac OS X; en-US; rv:1.8.1) Gecko/20061010 Firefox/2.0",
155
+ "Accept-Encoding" => "gzip",
156
+ "Cookie" => cookies,
157
+ "Referer" => referer
158
+ )
159
+ data = uncompress(resp, data)
160
+ cookies = parse_cookies(resp.response['set-cookie'], cookies)
161
+ forward = resp.response['Location']
162
+ if (not forward.nil?) && URI.parse(forward).host.nil?
163
+ forward = url.scheme.to_s + "://" + url.host.to_s + forward
164
+ end
165
+ return data, resp, cookies, forward
166
+ end
167
+
168
+ def uncompress(resp, data)
169
+ case resp.response['content-encoding']
170
+ when 'gzip'
171
+ gz = Zlib::GzipReader.new(StringIO.new(data))
172
+ data = gz.read
173
+ gz.close
174
+ resp.response['content-encoding'] = nil
175
+ # FIXME: Not sure what Hotmail was feeding me with their 'deflate',
176
+ # but the headers definitely were not right
177
+ when 'deflate'
178
+ data = Zlib::Inflate.inflate(data)
179
+ resp.response['content-encoding'] = nil
180
+ end
181
+
182
+ data
183
+ end
184
+ end
185
+
186
+ class ContactsError < StandardError
187
+ end
188
+
189
+ class AuthenticationError < ContactsError
190
+ end
191
+
192
+ class ConnectionError < ContactsError
193
+ end
194
+
195
+ class TypeNotFound < ContactsError
196
+ end
197
+
198
+ def self.new(type, login, password)
199
+ if TYPES.include?(type.to_s.intern)
200
+ TYPES[type.to_s.intern].new(login, password)
201
+ else
202
+ raise TypeNotFound, "#{type.inspect} is not a valid type, please choose one of the following: #{TYPES.keys.inspect}"
203
+ end
204
+ end
205
+
206
+ def self.guess(login, password)
207
+ TYPES.inject([]) do |a, t|
208
+ begin
209
+ a + t[1].new(login, password).contacts
210
+ rescue AuthenticationError
211
+ a
212
+ end
213
+ end.uniq
214
+ end
215
+ end
@@ -0,0 +1,35 @@
1
+ require 'gdata'
2
+
3
+ class Contacts
4
+ class Gmail < Base
5
+
6
+ CONTACTS_SCOPE = 'http://www.google.com/m8/feeds/'
7
+ CONTACTS_FEED = CONTACTS_SCOPE + 'contacts/default/full/?max-results=1000'
8
+
9
+ def contacts
10
+ return @contacts if @contacts
11
+ end
12
+
13
+ def real_connect
14
+ @client = GData::Client::Contacts.new
15
+ @client.clientlogin(@login, @password)
16
+
17
+ feed = @client.get(CONTACTS_FEED).to_xml
18
+
19
+ @contacts = feed.elements.to_a('entry').collect do |entry|
20
+ title, email = entry.elements['title'].text, nil
21
+ entry.elements.each('gd:email') do |e|
22
+ email = e.attribute('address').value if e.attribute('primary')
23
+ end
24
+ [title, email] unless email.nil?
25
+ end
26
+ @contacts.compact!
27
+ rescue GData::Client::AuthorizationError => e
28
+ raise AuthenticationError, "Username or password are incorrect"
29
+ end
30
+
31
+ private
32
+
33
+ TYPES[:gmail] = Gmail
34
+ end
35
+ end
@@ -0,0 +1,124 @@
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
+ old_url = form_url
31
+ until cookies =~ /; PPAuth=/ || forward.nil?
32
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
33
+ end
34
+
35
+ if data.index("The e-mail address or password is incorrect")
36
+ raise AuthenticationError, "Username and password do not match"
37
+ elsif data != ""
38
+ raise AuthenticationError, "Required field must not be blank"
39
+ elsif cookies == ""
40
+ raise ConnectionError, PROTOCOL_ERROR
41
+ end
42
+
43
+ data, resp, cookies, forward = get("http://mail.live.com/mail", cookies)
44
+ until forward.nil?
45
+ data, resp, cookies, forward, old_url = get(forward, cookies, old_url) + [forward]
46
+ end
47
+
48
+ @domain = URI.parse(old_url).host
49
+ @cookies = cookies
50
+ rescue AuthenticationError => m
51
+ if @attempt == 1
52
+ retry
53
+ else
54
+ raise m
55
+ end
56
+ end
57
+
58
+ def contacts(options = {})
59
+ if connected?
60
+ url = URI.parse(contact_list_url)
61
+ data, resp, cookies, forward = get( contact_list_url, @cookies )
62
+
63
+ if resp.code_type != Net::HTTPOK
64
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
65
+ end
66
+
67
+ @contacts = []
68
+ build_contacts = []
69
+ go = true
70
+ index = 0
71
+
72
+ while(go) do
73
+ go = false
74
+ url = URI.parse(get_contact_list_url(index))
75
+ http = open_http(url)
76
+ resp, data = http.get(get_contact_list_url(index), "Cookie" => @cookies)
77
+
78
+ email_match_text_beginning = Regexp.escape("http://m.mail.live.com/?rru=compose&amp;to=")
79
+ email_match_text_end = Regexp.escape("&amp;")
80
+
81
+ raw_html = resp.body.grep(/(?:e|dn)lk[0-9]+/)
82
+ raw_html.delete_at 0
83
+ raw_html.inject do |memo, row|
84
+ c_info = row.match(/(e|dn)lk([0-9])+/)
85
+
86
+ # Same contact, or different?
87
+ build_contacts << [] if memo != c_info[2]
88
+
89
+ # Grab info
90
+ case c_info[1]
91
+ when "e" # Email
92
+ build_contacts.last[1] = row.match(/#{email_match_text_beginning}(.*)#{email_match_text_end}/)[1]
93
+ when "dn" # Name
94
+ build_contacts.last[0] = row.match(/<a[^>]*>(.+)<\/a>/)[1]
95
+ end
96
+
97
+ # Set memo to contact id
98
+ c_info[2]
99
+ end
100
+
101
+ go = resp.body.include?("Next page")
102
+ index += 1
103
+ end
104
+
105
+ build_contacts.each do |contact|
106
+ unless contact[1].nil?
107
+ # Only return contacts with email addresses
108
+ contact[1] = CGI::unescape(contact[1])
109
+ @contacts << contact
110
+ end
111
+ end
112
+ return @contacts
113
+ end
114
+ end
115
+
116
+ def get_contact_list_url(index)
117
+ "http://mpeople.live.com/default.aspx?pg=#{index}"
118
+ end
119
+
120
+ private
121
+
122
+ TYPES[:hotmail] = Hotmail
123
+ end
124
+ end
@@ -0,0 +1,16 @@
1
+ if !Object.const_defined?('ActiveSupport')
2
+ require 'json'
3
+ end
4
+
5
+ class Contacts
6
+ def self.parse_json( string )
7
+ if Object.const_defined?('ActiveSupport') and
8
+ ActiveSupport.const_defined?('JSON')
9
+ ActiveSupport::JSON.decode( string )
10
+ elsif Object.const_defined?('JSON')
11
+ JSON.parse( string )
12
+ else
13
+ raise 'Contacts requires JSON or Rails (with ActiveSupport::JSON)'
14
+ end
15
+ end
16
+ 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,109 @@
1
+ class Contacts
2
+ class Yahoo < Base
3
+ URL = "http://mail.yahoo.com/"
4
+ LOGIN_URL = "https://login.yahoo.com/config/login"
5
+ ADDRESS_BOOK_URL = "http://address.mail.yahoo.com/?.rand=430244936"
6
+ CONTACT_LIST_URL = "http://address.mail.yahoo.com/?_src=&_crumb=crumb&sortfield=3&bucket=1&scroll=1&VPC=social_list&.r=time"
7
+ 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"
8
+
9
+ def real_connect
10
+ postdata = ".tries=2&.src=ym&.md5=&.hash=&.js=&.last=&promo=&.intl=us&.bypass="
11
+ postdata += "&.partner=&.u=4eo6isd23l8r3&.v=0&.challenge=gsMsEcoZP7km3N3NeI4mX"
12
+ postdata += "kGB7zMV&.yplus=&.emailCode=&pkg=&stepid=&.ev=&hasMsgr=1&.chkP=Y&."
13
+ postdata += "done=#{CGI.escape(URL)}&login=#{CGI.escape(login)}&passwd=#{CGI.escape(password)}"
14
+
15
+ data, resp, cookies, forward = post(LOGIN_URL, postdata)
16
+
17
+ if data.index("Invalid ID or password") || data.index("This ID is not yet taken")
18
+ raise AuthenticationError, "Username and password do not match"
19
+ elsif data.index("Sign in") && data.index("to Yahoo!")
20
+ raise AuthenticationError, "Required field must not be blank"
21
+ elsif !data.match(/uncompressed\/chunked/)
22
+ raise ConnectionError, PROTOCOL_ERROR
23
+ elsif cookies == ""
24
+ raise ConnectionError, PROTOCOL_ERROR
25
+ end
26
+
27
+ data, resp, cookies, forward = get(forward, cookies, LOGIN_URL)
28
+
29
+ if resp.code_type != Net::HTTPOK
30
+ raise ConnectionError, PROTOCOL_ERROR
31
+ end
32
+
33
+ @cookies = cookies
34
+ end
35
+
36
+ def contacts
37
+ return @contacts if @contacts
38
+ if connected?
39
+ # first, get the addressbook site with the new crumb parameter
40
+ url = URI.parse(address_book_url)
41
+ http = open_http(url)
42
+ resp, data = http.get("#{url.path}?#{url.query}",
43
+ "Cookie" => @cookies
44
+ )
45
+
46
+ if resp.code_type != Net::HTTPOK
47
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
48
+ end
49
+
50
+ crumb = data.to_s[/dotCrumb: '(.*?)'/][13...-1]
51
+
52
+ # now proceed with the new ".crumb" parameter to get the csv data
53
+ url = URI.parse(contact_list_url.sub("_crumb=crumb","_crumb=#{crumb}").sub("time", Time.now.to_f.to_s.sub(".","")[0...-2]))
54
+ http = open_http(url)
55
+ resp, more_data = http.get("#{url.path}?#{url.query}",
56
+ "Cookie" => @cookies,
57
+ "X-Requested-With" => "XMLHttpRequest",
58
+ "Referer" => address_book_url
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
+
67
+ parse more_data
68
+
69
+ if more_data =~ /"TotalABContacts":(\d+)/
70
+ total = $1.to_i
71
+ ((total / 50)).times do |i|
72
+ # now proceed with the new ".crumb" parameter to get the csv data
73
+ url = URI.parse(contact_list_url.sub("bucket=1","bucket=#{i+2}").sub("_crumb=crumb","_crumb=#{crumb}").sub("time", Time.now.to_f.to_s.sub(".","")[0...-2]))
74
+ http = open_http(url)
75
+ resp, more_data = http.get("#{url.path}?#{url.query}",
76
+ "Cookie" => @cookies,
77
+ "X-Requested-With" => "XMLHttpRequest",
78
+ "Referer" => address_book_url
79
+ )
80
+
81
+ if resp.code_type != Net::HTTPOK
82
+ raise ConnectionError, self.class.const_get(:PROTOCOL_ERROR)
83
+ end
84
+
85
+ parse more_data
86
+ end
87
+ end
88
+
89
+ @contacts
90
+ end
91
+ end
92
+
93
+ private
94
+
95
+ def parse(data, options={})
96
+ @contacts ||= []
97
+ if data =~ /var InitialContacts = (\[.*?\])/
98
+ @contacts += Contacts.parse_json($1).select{|contact|!contact["email"].to_s.empty?}.map{|contact|[contact["contactName"], contact["email"]]}
99
+ elsif data =~ /^\{"response":/
100
+ @contacts += Contacts.parse_json(data)["response"]["ResultSet"]["Contacts"].to_a.select{|contact|!contact["email"].to_s.empty?}.map{|contact|[contact["contactName"], contact["email"]]}
101
+ else
102
+ @contacts
103
+ end
104
+ end
105
+
106
+ end
107
+
108
+ TYPES[:yahoo] = Yahoo
109
+ end
data/lib/contacts.rb ADDED
@@ -0,0 +1,11 @@
1
+ $:.unshift(File.dirname(__FILE__)+"/contacts/")
2
+
3
+ require 'rubygems'
4
+
5
+ require 'base'
6
+ require 'gmail'
7
+ require 'hotmail'
8
+ require 'yahoo'
9
+ require 'plaxo'
10
+ require 'aol'
11
+ require 'json_picker'
metadata ADDED
@@ -0,0 +1,85 @@
1
+ --- !ruby/object:Gem::Specification
2
+ name: glennfu-contacts
3
+ version: !ruby/object:Gem::Version
4
+ version: 1.2.1
5
+ platform: ruby
6
+ authors:
7
+ - Glenn Sidney
8
+ autorequire:
9
+ bindir: bin
10
+ cert_chain: []
11
+
12
+ date: 2009-05-06 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: 1.1.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.1
34
+ version:
35
+ description: A universal interface to grab contact list information from various providers including Yahoo, Gmail, Hotmail, and Plaxo.
36
+ email: glenn@glennfu.com
37
+ executables: []
38
+
39
+ extensions: []
40
+
41
+ extra_rdoc_files: []
42
+
43
+ files:
44
+ - LICENSE
45
+ - Rakefile
46
+ - README
47
+ - examples/grab_contacts.rb
48
+ - lib/contacts.rb
49
+ - lib/contacts/base.rb
50
+ - lib/contacts/json_picker.rb
51
+ - lib/contacts/gmail.rb
52
+ - lib/contacts/aol.rb
53
+ - lib/contacts/hotmail.rb
54
+ - lib/contacts/plaxo.rb
55
+ - lib/contacts/yahoo.rb
56
+ has_rdoc: true
57
+ homepage: http://github.com/glennfu/contacts
58
+ licenses: []
59
+
60
+ post_install_message:
61
+ rdoc_options: []
62
+
63
+ require_paths:
64
+ - lib
65
+ required_ruby_version: !ruby/object:Gem::Requirement
66
+ requirements:
67
+ - - ">="
68
+ - !ruby/object:Gem::Version
69
+ version: "0"
70
+ version:
71
+ required_rubygems_version: !ruby/object:Gem::Requirement
72
+ requirements:
73
+ - - ">="
74
+ - !ruby/object:Gem::Version
75
+ version: "0"
76
+ version:
77
+ requirements: []
78
+
79
+ rubyforge_project:
80
+ rubygems_version: 1.3.5
81
+ signing_key:
82
+ specification_version: 3
83
+ summary: A universal interface to grab contact list information from various providers including Yahoo, Gmail, Hotmail, and Plaxo.
84
+ test_files: []
85
+