Skip to content

Commit

Permalink
feat!: Make HTTP::Feature work like rack middleware
Browse files Browse the repository at this point in the history
  • Loading branch information
John Doe committed Oct 6, 2023
1 parent 65276d7 commit 33ef7dd
Show file tree
Hide file tree
Showing 3 changed files with 37 additions and 37 deletions.
42 changes: 16 additions & 26 deletions lib/http/client.rb
Original file line number Diff line number Diff line change
Expand Up @@ -32,7 +32,7 @@ def request(verb, uri, opts = {})
return res unless opts.follow

Redirector.new(opts.follow).perform(req, res) do |request|
perform(wrap_request(request, opts), opts)
perform(request, opts)
end
end

Expand All @@ -51,8 +51,6 @@ def build_request(verb, uri, opts = {})
:headers => headers,
:body => body
)

wrap_request(req, opts)
end

# @!method persistent?
Expand All @@ -62,6 +60,21 @@ def build_request(verb, uri, opts = {})

# Perform a single (no follow) HTTP request
def perform(req, options)
app = ->(r) { perform__(r, options) }
options.features.values.inject(app) do |app, feature|
->(r) { feature.call(r, &app) }
end.call(req)
end

def close
@connection&.close
@connection = nil
@state = :clean
end

private

def perform__(req, options)
verify_connection!(req.uri)

@state = :dirty
Expand All @@ -73,18 +86,9 @@ def perform(req, options)
@connection.send_request(req)
@connection.read_headers!
end
rescue Error => e
options.features.each_value do |feature|
feature.on_error(req, e)
end
raise
end
res = build_response(req, options)

res = options.features.inject(res) do |response, (_name, feature)|
feature.wrap_response(response)
end

@connection.finish_response if req.verb == :head
@state = :clean

Expand All @@ -94,20 +98,6 @@ def perform(req, options)
raise
end

def close
@connection&.close
@connection = nil
@state = :clean
end

private

def wrap_request(req, opts)
opts.features.inject(req) do |request, (_name, feature)|
feature.wrap_request(request)
end
end

def build_response(req, options)
Response.new(
:status => @connection.status_code,
Expand Down
8 changes: 8 additions & 0 deletions lib/http/feature.rb
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,14 @@ def initialize(opts = {})
@opts = opts
end

def call(req, &app)
req = wrap_request(req)
wrap_response(app.call(req))
rescue StandardError => ex
on_error(req, ex)
raise
end

def wrap_request(request)
request
end
Expand Down
24 changes: 13 additions & 11 deletions spec/lib/http/client_spec.rb
Original file line number Diff line number Diff line change
Expand Up @@ -13,7 +13,7 @@

before do
stubbed_client = Class.new(HTTP::Client) do
def perform(request, options)
def perform__(request, options)
stubbed = stubs[HTTP::URI::NORMALIZER.call(request.uri).to_s]
stubbed ? stubbed.call(request) : super(request, options)
end
Expand Down Expand Up @@ -142,11 +142,19 @@ def simple_response(body, status = 200)
** INFO **
> GET http://example.com/
** INFO **
< 302 Found
** INFO **
> GET http://example.com/1
** INFO **
< 302 Found
** INFO **
> GET http://example.com/2
** INFO **
< 302 Found
** INFO **
> GET http://example.com/3
** INFO **
< 200 OK
OUTPUT
end
end
Expand Down Expand Up @@ -300,19 +308,13 @@ def simple_response(body, status = 200)
let(:client) { described_class.new :headers => headers, :features => {:auto_deflate => {}}, :body => "foo" }

it "deletes Content-Length header" do
expect(client).to receive(:perform) do |req, _|
expect(req["Content-Length"]).to eq nil
end

client.request(:get, "http://example.com/")
res = client.request(:get, "http://example.com/")
expect(res.request["Content-Length"]).to eq nil
end

it "sets Content-Encoding header" do
expect(client).to receive(:perform) do |req, _|
expect(req["Content-Encoding"]).to eq "gzip"
end

client.request(:get, "http://example.com/")
res = client.request(:get, "http://example.com/")
expect(res.request["Content-Encoding"]).to eq "gzip"
end

context "and there is no body" do
Expand Down

0 comments on commit 33ef7dd

Please sign in to comment.