1
0
Fork 0
mirror of https://github.com/omniauth/omniauth.git synced 2022-11-09 12:31:49 -05:00

Google OAuth strategy with spec

This commit is contained in:
mpd 2010-12-29 12:52:31 -08:00
parent 0992d32818
commit 0eff08e5e7
3 changed files with 82 additions and 0 deletions

View file

@ -21,5 +21,6 @@ module OmniAuth
autoload :Goodreads, 'omniauth/strategies/goodreads'
autoload :Yahoo, 'omniauth/strategies/yahoo'
autoload :TypePad, 'omniauth/strategies/type_pad'
autoload :Google, 'omniauth/strategies/google'
end
end

View file

@ -0,0 +1,76 @@
require 'omniauth/oauth'
require 'multi_json'
module OmniAuth
module Strategies
#
# Authenticate to Google via OAuth and retrieve basic
# user information.
#
# Usage:
#
# use OmniAuth::Strategies::Google, 'consumerkey', 'consumersecret'
#
class Google < OmniAuth::Strategies::OAuth
def initialize(app, consumer_key = nil, consumer_secret = nil, options = {}, &block)
client_options = {
:site => 'https://www.google.com',
:request_token_path => '/accounts/OAuthGetRequestToken',
:access_token_path => '/accounts/OAuthGetAccessToken',
:authorize_path => '/accounts/OAuthAuthorizeToken'
}
super(app, :google, consumer_key, consumer_secret, client_options, options)
end
def auth_hash
ui = user_info
OmniAuth::Utils.deep_merge(super, {
'uid' => ui['uid'],
'user_info' => ui,
'extra' => {'user_hash' => user_hash}
})
end
def user_info
email = user_hash['feed']['id']['$t']
name = user_hash['feed']['author'].first['name']['$t']
name = email if name.strip == '(unknown)'
{
'email' => email,
'uid' => email,
'name' => name
}
end
def user_hash
# Google is very strict about keeping authorization and
# authentication separated.
# They give no endpoint to get a user's profile directly that I can
# find. We *can* get their name and email out of the contacts feed,
# however. It will fail in the extremely rare case of a user who has
# a Google Account but has never even signed up for Gmail. This has
# not been seen in the field.
@user_hash ||= MultiJson.decode(@access_token.get("http://www.google.com/m8/feeds/contacts/default/full?max-results=1&alt=json").body)
end
# Monkeypatch OmniAuth to pass the scope in the consumer.get_request_token call
def request_phase
request_token = consumer.get_request_token({:oauth_callback => callback_url}, {:scope => "http://www.google.com/m8/feeds"})
(session['oauth']||={})[name.to_s] = {'callback_confirmed' => request_token.callback_confirmed?, 'request_token' => request_token.token, 'request_secret' => request_token.secret}
r = Rack::Response.new
if request_token.callback_confirmed?
r.redirect(request_token.authorize_url)
else
r.redirect(request_token.authorize_url(:oauth_callback => callback_url))
end
r.finish
end
end
end
end

View file

@ -0,0 +1,5 @@
require File.expand_path(File.dirname(__FILE__) + '/../../spec_helper')
describe OmniAuth::Strategies::Google do
it_should_behave_like 'an oauth strategy'
end