# frozen_string_literal: true ## # The Uri handles rubygems source URIs. # class Gem::Uri def initialize(source_uri) @parsed_uri = parse(source_uri) end def redacted return self unless valid_uri? if token? || oauth_basic? with_redacted_user elsif password? with_redacted_password else self end end def to_s @parsed_uri.to_s end def redact_credentials_from(text) return text unless valid_uri? && password? text.sub(password, 'REDACTED') end def method_missing(method_name, *args, &blk) if @parsed_uri.respond_to?(method_name) @parsed_uri.send(method_name, *args, &blk) else super end end def respond_to_missing?(method_name, include_private = false) @parsed_uri.respond_to?(method_name, include_private) || super end private ## # Parses the #uri, raising if it's invalid def parse!(uri) require "uri" raise URI::InvalidURIError unless uri # Always escape URI's to deal with potential spaces and such # It should also be considered that source_uri may already be # a valid URI with escaped characters. e.g. "{DESede}" is encoded # as "%7BDESede%7D". If this is escaped again the percentage # symbols will be escaped. begin URI.parse(uri) rescue URI::InvalidURIError URI.parse(URI::DEFAULT_PARSER.escape(uri)) end end ## # Parses the #uri, returning the original uri if it's invalid def parse(uri) return uri unless uri.is_a?(String) parse!(uri) rescue URI::InvalidURIError uri end def with_redacted_user clone.tap {|uri| uri.user = 'REDACTED' } end def with_redacted_password clone.tap {|uri| uri.password = 'REDACTED' } end def valid_uri? !@parsed_uri.is_a?(String) end def password? !!password end def oauth_basic? password == 'x-oauth-basic' end def token? !user.nil? && password.nil? end end