2015-02-12 03:23:34 +00:00
|
|
|
# Copyright 2015, Google Inc.
|
|
|
|
# All rights reserved.
|
|
|
|
#
|
|
|
|
# Redistribution and use in source and binary forms, with or without
|
|
|
|
# modification, are permitted provided that the following conditions are
|
|
|
|
# met:
|
|
|
|
#
|
|
|
|
# * Redistributions of source code must retain the above copyright
|
|
|
|
# notice, this list of conditions and the following disclaimer.
|
|
|
|
# * 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.
|
|
|
|
# * Neither the name of Google Inc. nor the names of its
|
|
|
|
# contributors may be used to endorse or promote products derived from
|
|
|
|
# this software without specific prior written permission.
|
|
|
|
#
|
|
|
|
# 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.
|
|
|
|
|
2019-02-27 16:06:41 +00:00
|
|
|
require "signet/oauth_2/client"
|
2015-02-12 03:23:34 +00:00
|
|
|
|
|
|
|
module Signet
|
|
|
|
# OAuth2 supports OAuth2 authentication.
|
|
|
|
module OAuth2
|
2016-01-05 21:58:50 +00:00
|
|
|
AUTH_METADATA_KEY = :authorization
|
2015-02-12 03:23:34 +00:00
|
|
|
# Signet::OAuth2::Client creates an OAuth2 client
|
|
|
|
#
|
|
|
|
# This reopens Client to add #apply and #apply! methods which update a
|
|
|
|
# hash with the fetched authentication token.
|
|
|
|
class Client
|
2019-03-15 19:34:54 +00:00
|
|
|
def configure_connection options
|
2019-01-02 22:42:42 +00:00
|
|
|
@connection_info =
|
|
|
|
options[:connection_builder] || options[:default_connection]
|
|
|
|
self
|
|
|
|
end
|
|
|
|
|
2015-02-12 03:23:34 +00:00
|
|
|
# Updates a_hash updated with the authentication token
|
2019-03-15 19:34:54 +00:00
|
|
|
def apply! a_hash, opts = {}
|
2015-02-12 03:23:34 +00:00
|
|
|
# fetch the access token there is currently not one, or if the client
|
|
|
|
# has expired
|
2020-04-08 00:19:29 +00:00
|
|
|
token_type = target_audience ? :id_token : :access_token
|
|
|
|
fetch_access_token! opts if send(token_type).nil? || expires_within?(60)
|
|
|
|
a_hash[AUTH_METADATA_KEY] = "Bearer #{send token_type}"
|
2015-02-12 03:23:34 +00:00
|
|
|
end
|
|
|
|
|
|
|
|
# Returns a clone of a_hash updated with the authentication token
|
2019-03-15 19:34:54 +00:00
|
|
|
def apply a_hash, opts = {}
|
2015-02-12 03:23:34 +00:00
|
|
|
a_copy = a_hash.clone
|
2019-03-15 19:34:54 +00:00
|
|
|
apply! a_copy, opts
|
2015-02-12 03:23:34 +00:00
|
|
|
a_copy
|
|
|
|
end
|
|
|
|
|
|
|
|
# Returns a reference to the #apply method, suitable for passing as
|
|
|
|
# a closure
|
|
|
|
def updater_proc
|
|
|
|
lambda(&method(:apply))
|
|
|
|
end
|
2015-10-14 20:53:37 +00:00
|
|
|
|
2019-03-15 19:34:54 +00:00
|
|
|
def on_refresh &block
|
2019-08-16 01:46:38 +00:00
|
|
|
@refresh_listeners = [] unless defined? @refresh_listeners
|
2015-10-14 20:53:37 +00:00
|
|
|
@refresh_listeners << block
|
|
|
|
end
|
|
|
|
|
2016-07-13 22:44:16 +00:00
|
|
|
alias orig_fetch_access_token! fetch_access_token!
|
2019-03-15 19:34:54 +00:00
|
|
|
def fetch_access_token! options = {}
|
2019-01-02 22:42:42 +00:00
|
|
|
unless options[:connection]
|
|
|
|
connection = build_default_connection
|
2019-03-15 19:34:54 +00:00
|
|
|
options = options.merge connection: connection if connection
|
2019-01-02 22:42:42 +00:00
|
|
|
end
|
2019-06-10 20:52:45 +00:00
|
|
|
info = retry_with_error do
|
|
|
|
orig_fetch_access_token! options
|
|
|
|
end
|
2015-10-14 20:53:37 +00:00
|
|
|
notify_refresh_listeners
|
|
|
|
info
|
|
|
|
end
|
|
|
|
|
|
|
|
def notify_refresh_listeners
|
2019-08-16 01:46:38 +00:00
|
|
|
listeners = defined?(@refresh_listeners) ? @refresh_listeners : []
|
2015-10-14 20:53:37 +00:00
|
|
|
listeners.each do |block|
|
2019-03-15 19:34:54 +00:00
|
|
|
block.call self
|
2015-10-14 20:53:37 +00:00
|
|
|
end
|
|
|
|
end
|
2016-12-07 15:56:14 +00:00
|
|
|
|
2019-01-02 22:42:42 +00:00
|
|
|
def build_default_connection
|
|
|
|
if !defined?(@connection_info)
|
|
|
|
nil
|
|
|
|
elsif @connection_info.respond_to? :call
|
|
|
|
@connection_info.call
|
|
|
|
else
|
|
|
|
@connection_info
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2019-03-15 19:34:54 +00:00
|
|
|
def retry_with_error max_retry_count = 5
|
2016-12-07 15:56:14 +00:00
|
|
|
retry_count = 0
|
|
|
|
|
|
|
|
begin
|
|
|
|
yield
|
2019-03-15 19:34:54 +00:00
|
|
|
rescue StandardError => e
|
|
|
|
raise e if e.is_a?(Signet::AuthorizationError) || e.is_a?(Signet::ParseError)
|
2016-12-07 15:56:14 +00:00
|
|
|
|
|
|
|
if retry_count < max_retry_count
|
|
|
|
retry_count += 1
|
|
|
|
sleep retry_count * 0.3
|
|
|
|
retry
|
|
|
|
else
|
|
|
|
msg = "Unexpected error: #{e.inspect}"
|
2019-03-15 19:34:54 +00:00
|
|
|
raise Signet::AuthorizationError, msg
|
2016-12-07 15:56:14 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2015-02-12 03:23:34 +00:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|