Page MenuHomePhorge

No OneTemporary

Size
4 KB
Referenced Files
None
Subscribers
None
diff --git a/lib/tesla/adapter/httpc.ex b/lib/tesla/adapter/httpc.ex
index 1f7dff5..4dea458 100644
--- a/lib/tesla/adapter/httpc.ex
+++ b/lib/tesla/adapter/httpc.ex
@@ -1,63 +1,59 @@
defmodule Tesla.Adapter.Httpc do
@moduledoc """
Adapter for `:httpc`
**NOTE** Tesla overrides default autoredirect value with false to ensure
consistency between adapters
"""
+ import Tesla.Adapter.Shared, only: [stream_to_fun: 1, next_chunk: 1]
+
@override_defaults autoredirect: false
@http_opts ~w(timeout connect_timeout ssl essl autoredirect proxy_auth version relaxed url_encode)a
def call(env, opts) do
opts = Keyword.merge(@override_defaults, opts || [])
with {:ok, {status, headers, body}} <- request(env, opts) do
format_response(env, status, headers, body)
end
end
defp format_response(env, {_, status, _}, headers, body) do
%{env | status: status,
headers: headers,
body: body}
end
defp request(env, opts) do
content_type = to_char_list(env.headers["content-type"] || "")
handle request(
env.method || :get,
Tesla.build_url(env.url, env.query) |> to_char_list,
Enum.into(env.headers, [], fn {k,v} -> {to_char_list(k), to_char_list(v)} end),
content_type,
env.body,
Keyword.split(opts ++ env.opts, @http_opts)
)
end
defp request(method, url, headers, _content_type, nil, {http_opts, opts}) do
:httpc.request(method, {url, headers}, http_opts, opts)
end
defp request(method, url, headers, content_type, %Stream{} = body, opts) do
- reductor = fn(item, _acc) -> {:suspend, item} end
- {_, _, fun} = Enumerable.reduce(body, {:suspend, nil}, reductor)
+ fun = stream_to_fun(body)
request(method, url, headers, content_type, fun, opts)
end
defp request(method, url, headers, content_type, body, opts) when is_function(body) do
body = {:chunkify, &next_chunk/1, body}
request(method, url, headers, content_type, body, opts)
end
defp request(method, url, headers, content_type, body, {http_opts, opts}) do
:httpc.request(method, {url, headers, content_type, body}, http_opts, opts)
end
- defp next_chunk(fun), do: parse_chunk fun.({:cont, nil})
-
- defp parse_chunk({:suspended, item, fun}), do: {:ok, item, fun}
- defp parse_chunk(_), do: :eof
-
defp handle({:error, {:failed_connect, _}}), do: {:error, :econnrefused}
defp handle(response), do: response
end
diff --git a/lib/tesla/adapter/ibrowse.ex b/lib/tesla/adapter/ibrowse.ex
index 09cd9a7..4ce08d4 100644
--- a/lib/tesla/adapter/ibrowse.ex
+++ b/lib/tesla/adapter/ibrowse.ex
@@ -1,25 +1,42 @@
if Code.ensure_loaded?(:ibrowse) do
defmodule Tesla.Adapter.Ibrowse do
+ import Tesla.Adapter.Shared, only: [stream_to_fun: 1, next_chunk: 1]
+
def call(env, opts) do
with {:ok, status, headers, body} <- request(env, opts || []) do
%{env | status: status,
headers: headers,
body: body}
end
end
defp request(env, opts) do
body = env.body || []
- handle :ibrowse.send_req(
+ handle request(
Tesla.build_url(env.url, env.query) |> to_char_list,
Enum.into(env.headers, []),
env.method,
body,
opts ++ env.opts
)
end
+ defp request(url, headers, method, %Stream{} = body, opts) do
+ fun = stream_to_fun(body)
+ request(url, headers, method, fun, opts)
+ end
+
+ defp request(url, headers, method, body, opts) when is_function(body) do
+ body = {&next_chunk/1, body}
+ opts = Keyword.put(opts, :transfer_encoding, :chunked)
+ request(url, headers, method, body, opts)
+ end
+
+ defp request(url, headers, method, body, opts) do
+ :ibrowse.send_req(url, headers, method, body, opts)
+ end
+
defp handle({:error, {:conn_failed, error}}), do: error
defp handle(response), do: response
end
end
diff --git a/lib/tesla/adapter/shared.ex b/lib/tesla/adapter/shared.ex
new file mode 100644
index 0000000..ebe7dc8
--- /dev/null
+++ b/lib/tesla/adapter/shared.ex
@@ -0,0 +1,13 @@
+defmodule Tesla.Adapter.Shared do
+ def stream_to_fun(stream) do
+ reductor = fn(item, _acc) -> {:suspend, item} end
+ {_, _, fun} = Enumerable.reduce(stream, {:suspend, nil}, reductor)
+
+ fun
+ end
+
+ def next_chunk(fun), do: parse_chunk fun.({:cont, nil})
+
+ defp parse_chunk({:suspended, item, fun}), do: {:ok, item, fun}
+ defp parse_chunk(_), do: :eof
+end
diff --git a/test/tesla/adapter/ibrowse_test.exs b/test/tesla/adapter/ibrowse_test.exs
index a663749..03e2457 100644
--- a/test/tesla/adapter/ibrowse_test.exs
+++ b/test/tesla/adapter/ibrowse_test.exs
@@ -1,10 +1,11 @@
defmodule IbrowseTest do
use ExUnit.Case
use Tesla.Adapter.TestCase.Basic, adapter: :ibrowse
+ use Tesla.Adapter.TestCase.StreamRequestBody, adapter: :ibrowse
use Tesla.Adapter.TestCase.SSL, adapter: :ibrowse
setup do
Application.ensure_started(:ibrowse)
:ok
end
end

File Metadata

Mime Type
text/x-diff
Expires
Sun, Jan 19, 11:31 AM (4 h, 9 m)
Storage Engine
blob
Storage Format
Raw Data
Storage Handle
55132
Default Alt Text
(4 KB)

Event Timeline