class HTTP::Response

def build_http_rb_response_body_from_webmock_response(webmock_response)

def build_http_rb_response_body_from_webmock_response(webmock_response)
  if HTTP::VERSION < "1.0.0"
    Body.new(Streamer.new(webmock_response.body))
  elsif HTTP::VERSION < "3.0.0"
    Body.new(Streamer.new(webmock_response.body), webmock_response.body.encoding)
  else
    Body.new(
      Streamer.new(webmock_response.body, encoding: webmock_response.body.encoding),
      encoding: webmock_response.body.encoding
    )
  end
end

def from_webmock(request, webmock_response, request_signature = nil)

def from_webmock(request, webmock_response, request_signature = nil)
  status  = Status.new(webmock_response.status.first)
  headers = webmock_response.headers || {}
  uri     = normalize_uri(request_signature && request_signature.uri)
  # HTTP.rb 3.0+ uses a keyword argument to pass the encoding, but 1.x
  # and 2.x use a positional argument, and 0.x don't support supplying
  # the encoding.
  body = build_http_rb_response_body_from_webmock_response(webmock_response)
  return new(status, "1.1", headers, body, uri) if HTTP::VERSION < "1.0.0"
  # 5.0.0 had a breaking change to require request instead of uri.
  if HTTP::VERSION < '5.0.0'
    return new({
      status: status,
      version: "1.1",
      headers: headers,
      body: body,
      uri: uri
    })
  end
  new({
    status: status,
    version: "1.1",
    headers: headers,
    body: body,
    request: request,
  })
end

def normalize_uri(uri)

def normalize_uri(uri)
  return unless uri
  uri = Addressable::URI.parse uri
  uri.port = nil if uri.default_port && uri.port == uri.default_port
  uri
end

def reset_body_to_allow_it_to_be_streamed!(webmock_response)

def reset_body_to_allow_it_to_be_streamed!(webmock_response)
  @body = self.class.build_http_rb_response_body_from_webmock_response(webmock_response)
end

def to_webmock

def to_webmock
  webmock_response = ::WebMock::Response.new
  webmock_response.status  = [status.to_i, reason]
  webmock_response.body    = body.to_s
  # This call is used to reset the body of the response to enable it to be streamed if necessary.
  # The `body.to_s` call above reads the body, which allows WebMock to trigger any registered callbacks.
  # However, once the body is read to_s, it cannot be streamed again and attempting to do so
  # will raise a "HTTP::StateError: body has already been consumed" error.
  # To avoid this error, we replace the original body with a new one.
  # The new body has its @stream attribute set to new Streamer, instead of the original Connection.
  # Unfortunately, it's not possible to reset the original body to its initial streaming state.
  # Therefore, this replacement is the best workaround currently available.
  reset_body_to_allow_it_to_be_streamed!(webmock_response)
  webmock_response.headers = headers.to_h
  webmock_response
end