module Doorkeeper::OAuth::Helpers::URIChecker

def self.as_uri(url)

def self.as_uri(url)
  URI.parse(url)
end

def self.hypertext_scheme?(uri)

def self.hypertext_scheme?(uri)
  %w[http https].include?(uri.scheme)
end

def self.iff_host?(uri)

def self.iff_host?(uri)
  !(hypertext_scheme?(uri) && uri.host.nil?)
end

def self.loopback_uri?(uri)

def self.loopback_uri?(uri)
  IPAddr.new(uri.host).loopback?
rescue IPAddr::Error
  false
end

def self.matches?(url, client_url)

def self.matches?(url, client_url)
  url = as_uri(url)
  client_url = as_uri(client_url)
  unless client_url.query.nil?
    return false unless query_matches?(url.query, client_url.query)
    # Clear out queries so rest of URI can be tested. This allows query
    # params to be in the request but order not mattering.
    client_url.query = nil
  end
  # RFC8252, Paragraph 7.3
  # @see https://tools.ietf.org/html/rfc8252#section-7.3
  if loopback_uri?(url) && loopback_uri?(client_url)
    url.port = nil
    client_url.port = nil
  end
  url.query = nil
  url == client_url
end

def self.oob_uri?(uri)

def self.oob_uri?(uri)
  NonStandard::IETF_WG_OAUTH2_OOB_METHODS.include?(uri)
end

def self.query_matches?(query, client_query)

def self.query_matches?(query, client_query)
  return true if client_query.blank? && query.blank?
  return false if client_query.nil? || query.nil?
  # Will return true independent of query order
  client_query.split("&").sort == query.split("&").sort
end

def self.valid?(url)

def self.valid?(url)
  return true if oob_uri?(url)
  uri = as_uri(url)
  valid_scheme?(uri) && iff_host?(uri) && uri.fragment.nil? && uri.opaque.nil?
rescue URI::InvalidURIError
  false
end

def self.valid_for_authorization?(url, client_url)

def self.valid_for_authorization?(url, client_url)
  valid?(url) && client_url.split.any? { |other_url| matches?(url, other_url) }
end

def self.valid_scheme?(uri)

def self.valid_scheme?(uri)
  return false if uri.scheme.nil?
  %w[localhost].include?(uri.scheme) == false
end