class Capybara::RackTest::Browser

def app

def app
  driver.app
end

def base_href

def base_href
  find(:css, 'head > base').first&.[](:href).to_s
end

def base_relative_uri_for(uri)

def base_relative_uri_for(uri)
  base_uri = URI.parse(base_href)
  current_uri = URI.parse(safe_last_request&.url.to_s).tap do |c|
    c.path.sub!(%r{/[^/]*$}, '/') unless uri.path.empty?
    c.path = '/' if c.path.empty?
  end
  if [current_uri, base_uri].any?(&:absolute?)
    current_uri.merge(base_uri)
  else
    base_uri.path = current_uri.path if base_uri.path.empty?
    base_uri
  end
end

def build_rack_mock_session

def build_rack_mock_session
  reset_host! unless current_host
  Rack::MockSession.new(app, current_host)
end

def build_uri(path)

def build_uri(path)
  uri = URI.parse(path)
  base_uri = base_relative_uri_for(uri)
  uri.path = base_uri.path + uri.path unless uri.absolute? || uri.path.start_with?('/')
  if base_uri.absolute?
    base_uri.merge(uri)
  else
    uri.scheme ||= @current_scheme
    uri.host ||= @current_host
    uri.port ||= @current_port unless uri.default_port == @current_port
    uri
  end
end

def current_url

def current_url
  uri = build_uri(last_request.url)
  uri.fragment = @current_fragment if @current_fragment
  uri.to_s
rescue Rack::Test::Error
  ''
end

def dom

def dom
  @dom ||= Capybara::HTML(html)
end

def find(format, selector)

def find(format, selector)
  if format == :css
    dom.css(selector, Capybara::RackTest::CSSHandlers.new)
  else
    dom.xpath(selector)
  end.map { |node| Capybara::RackTest::Node.new(self, node) }
end

def follow(method, path, **attributes)

def follow(method, path, **attributes)
  return if fragment_or_script?(path)
  process_and_follow_redirects(method, path, attributes, 'HTTP_REFERER' => referer_url)
end

def fragment_or_script?(path)

def fragment_or_script?(path)
  path.gsub(/^#{Regexp.escape(request_path)}/, '').start_with?('#') || path.downcase.start_with?('javascript:')
end

def html

def html
  last_response.body
rescue Rack::Test::Error
  ''
end

def initialize(driver)

def initialize(driver)
  @driver = driver
  @current_fragment = nil
end

def last_request

def last_request
  raise Rack::Test::Error if @new_visit_request
  super
end

def last_response

def last_response
  raise Rack::Test::Error if @new_visit_request
  super
end

def options

def options
  driver.options
end

def process(method, path, attributes = {}, env = {})

def process(method, path, attributes = {}, env = {})
  method = method.downcase
  new_uri = build_uri(path)
  @current_scheme, @current_host, @current_port = new_uri.select(:scheme, :host, :port)
  @current_fragment = new_uri.fragment || @current_fragment
  reset_cache!
  @new_visit_request = false
  send(method, new_uri.to_s, attributes, env.merge(options[:headers] || {}))
end

def process_and_follow_redirects(method, path, attributes = {}, env = {})

def process_and_follow_redirects(method, path, attributes = {}, env = {})
  @current_fragment = build_uri(path).fragment
  process(method, path, attributes, env)
  return unless driver.follow_redirects?
  driver.redirect_limit.times do
    if last_response.redirect?
      if [307, 308].include? last_response.status
        process(last_request.request_method, last_response['Location'], last_request.params, env)
      else
        process(:get, last_response['Location'], {}, env)
      end
    end
  end
  if last_response.redirect? # rubocop:disable Style/GuardClause
    raise Capybara::InfiniteRedirectError, "redirected more than #{driver.redirect_limit} times, check for infinite redirects."
  end
end

def referer_url

def referer_url
  build_uri(last_request.url).to_s
rescue Rack::Test::Error
  ''
end

def refresh

def refresh
  reset_cache!
  request(last_request.fullpath, last_request.env)
end

def request_path

def request_path
  last_request.path
rescue Rack::Test::Error
  '/'
end

def reset_cache!

def reset_cache!
  @dom = nil
end

def reset_host!

def reset_host!
  uri = URI.parse(driver.session_options.app_host || driver.session_options.default_host)
  @current_scheme, @current_host, @current_port = uri.select(:scheme, :host, :port)
end

def safe_last_request

def safe_last_request
  last_request
rescue Rack::Test::Error
  nil
end

def submit(method, path, attributes, content_type: nil)

def submit(method, path, attributes, content_type: nil)
  path = request_path if path.nil? || path.empty?
  uri = build_uri(path)
  uri.query = '' if method.to_s.casecmp('get').zero?
  env = { 'HTTP_REFERER' => referer_url }
  env['CONTENT_TYPE'] = content_type if content_type
  process_and_follow_redirects(
    method,
    uri.to_s,
    attributes,
    env
  )
end

def title

def title
  dom.title
end

def visit(path, **attributes)

def visit(path, **attributes)
  @new_visit_request = true
  reset_cache!
  reset_host!
  process_and_follow_redirects(:get, path, attributes)
end