Skip to content

Commit

Permalink
fix: Client library credentials provide correct self-signed JWT and e…
Browse files Browse the repository at this point in the history
…xternal account behavior when loading from a file path or JSON data
  • Loading branch information
dazuma committed Feb 4, 2024
1 parent 0fc7a5d commit 44f10e9
Show file tree
Hide file tree
Showing 2 changed files with 243 additions and 265 deletions.
50 changes: 23 additions & 27 deletions lib/googleauth/credentials.rb
Original file line number Diff line number Diff line change
Expand Up @@ -356,8 +356,9 @@ def self.lookup_local_constant name
#
def initialize keyfile, options = {}
verify_keyfile_provided! keyfile
@project_id = options["project_id"] || options["project"]
@quota_project_id = options["quota_project_id"]
options = symbolize_hash_keys options
@project_id = options[:project_id] || options[:project]
@quota_project_id = options[:quota_project_id]
case keyfile
when Google::Auth::BaseClient
update_from_signet keyfile
Expand Down Expand Up @@ -484,45 +485,40 @@ def verify_keyfile_exists! keyfile
end

# Initializes the Signet client.
def init_client keyfile, connection_options = {}
client_opts = client_options keyfile
Signet::OAuth2::Client.new(client_opts)
.configure_connection(connection_options)
def init_client hash, options = {}
options = update_client_options options
io = StringIO.new JSON.generate hash
options.merge! json_key_io: io
Google::Auth::DefaultCredentials.make_creds options
end

# returns a new Hash with string keys instead of symbol keys.
def stringify_hash_keys hash
hash.to_h.transform_keys(&:to_s)
end

# rubocop:disable Metrics/AbcSize
# returns a new Hash with symbol keys instead of string keys.
def symbolize_hash_keys hash
hash.to_h.transform_keys(&:to_sym)
end

def update_client_options options
options = options.dup

def client_options options
# Keyfile options have higher priority over constructor defaults
options["token_credential_uri"] ||= self.class.token_credential_uri
options["audience"] ||= self.class.audience
options["scope"] ||= self.class.scope
options["target_audience"] ||= self.class.target_audience
# options have higher priority over constructor defaults
options[:token_credential_uri] ||= self.class.token_credential_uri
options[:audience] ||= self.class.audience
options[:scope] ||= self.class.scope
options[:target_audience] ||= self.class.target_audience

if !Array(options["scope"]).empty? && options["target_audience"]
if !Array(options[:scope]).empty? && options[:target_audience]
raise ArgumentError, "Cannot specify both scope and target_audience"
end
options.delete :scope unless options[:target_audience].nil?

needs_scope = options["target_audience"].nil?
# client options for initializing signet client
{
token_credential_uri: options["token_credential_uri"],
audience: options["audience"],
scope: (needs_scope ? Array(options["scope"]) : nil),
target_audience: options["target_audience"],
issuer: options["client_email"],
signing_key: OpenSSL::PKey::RSA.new(options["private_key"]),
universe_domain: options["universe_domain"] || "googleapis.com"
}
options
end

# rubocop:enable Metrics/AbcSize

def update_from_signet client
@project_id ||= client.project_id if client.respond_to? :project_id
@quota_project_id ||= client.quota_project_id if client.respond_to? :quota_project_id
Expand Down
Loading

0 comments on commit 44f10e9

Please sign in to comment.