From 731d95e4f096110685d3acc75d5129502a0ce161 Mon Sep 17 00:00:00 2001 From: en30 Date: Tue, 1 Feb 2022 20:32:51 +0900 Subject: [PATCH] Skip XML.decode!(body) if body is empty (#956) MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * Skip XML.decode!(body) if body is empty Google Cloud Storage has almost S3 compatible XML API, but some responses have `content-type: text/html; charset=UTF-8` and empty body. This change prevents XML decode error and enables work fine with Google Cloud Storage. * Handle Google Cloud Storage XML API response * Rename S3.encode to S3.decode * Improve a test case description Co-authored-by: Jonatan Kłosko * Make S3.decode/1 let any content type pass * mix format Co-authored-by: Jonatan Kłosko --- lib/livebook/file_system/s3.ex | 18 +++++++++--------- test/livebook/file_system/s3_test.exs | 20 ++++++++++++++++++++ 2 files changed, 29 insertions(+), 9 deletions(-) diff --git a/lib/livebook/file_system/s3.ex b/lib/livebook/file_system/s3.ex index 2329ca7bf..5aaabfbdc 100644 --- a/lib/livebook/file_system/s3.ex +++ b/lib/livebook/file_system/s3.ex @@ -200,7 +200,7 @@ defimpl Livebook.FileSystem, for: Livebook.FileSystem.S3 do query = %{"list-type" => "2", "prefix" => prefix, "delimiter" => delimiter} - case request(file_system, :get, "/", query: query) |> encode() do + case request(file_system, :get, "/", query: query) |> decode() do {:ok, 200, _headers, %{"ListBucketResult" => result}} -> bucket = result["Name"] file_keys = result |> xml_get_list("Contents") |> Enum.map(& &1["Key"]) @@ -224,7 +224,7 @@ defimpl Livebook.FileSystem, for: Livebook.FileSystem.S3 do query = %{"list-type" => "2", "max-keys" => "0"} - case request(file_system, :get, "/", query: query) |> encode() do + case request(file_system, :get, "/", query: query) |> decode() do {:ok, 200, _headers, %{"ListBucketResult" => %{"Name" => bucket}}} -> {:ok, bucket} @@ -242,7 +242,7 @@ defimpl Livebook.FileSystem, for: Livebook.FileSystem.S3 do end defp put_object(file_system, key, content) do - case request(file_system, :put, "/" <> encode_key(key), body: content) |> encode() do + case request(file_system, :put, "/" <> encode_key(key), body: content) |> decode() do {:ok, 200, _headers, _body} -> :ok other -> request_response_to_error(other) end @@ -268,7 +268,7 @@ defimpl Livebook.FileSystem, for: Livebook.FileSystem.S3 do headers = [{"x-amz-copy-source", copy_source}] case request(file_system, :put, "/" <> encode_key(destination_key), headers: headers) - |> encode() do + |> decode() do {:ok, 200, _headers, _body} -> :ok {:ok, 404, _headers, _body} -> FileSystem.Utils.posix_error(:enoent) other -> request_response_to_error(other) @@ -276,7 +276,7 @@ defimpl Livebook.FileSystem, for: Livebook.FileSystem.S3 do end defp delete_object(file_system, key) do - case request(file_system, :delete, "/" <> encode_key(key)) |> encode() do + case request(file_system, :delete, "/" <> encode_key(key)) |> decode() do {:ok, 204, _headers, _body} -> :ok {:ok, 404, _headers, _body} -> :ok other -> request_response_to_error(other) @@ -296,7 +296,7 @@ defimpl Livebook.FileSystem, for: Livebook.FileSystem.S3 do headers = [{"Content-MD5", body_md5}] case request(file_system, :post, "/", query: %{"delete" => ""}, headers: headers, body: body) - |> encode() do + |> decode() do {:ok, 200, _headers, %{"Error" => errors}} -> {:error, format_errors(errors)} {:ok, 200, _headers, _body} -> :ok other -> request_response_to_error(other) @@ -394,17 +394,17 @@ defimpl Livebook.FileSystem, for: Livebook.FileSystem.S3 do host |> String.split(".") |> Enum.reverse() |> Enum.at(2, "auto") end - defp encode({:ok, status, headers, body}) do + defp decode({:ok, status, headers, body}) do case HTTP.fetch_content_type(headers) do {:ok, content_type} when content_type in ["text/xml", "application/xml"] -> {:ok, status, headers, XML.decode!(body)} - :error -> + _ -> {:ok, status, headers, body} end end - defp encode(other), do: other + defp decode(other), do: other defp xml_get_list(xml_map, key) do case xml_map do diff --git a/test/livebook/file_system/s3_test.exs b/test/livebook/file_system/s3_test.exs index fef94d4a4..5a2d145ec 100644 --- a/test/livebook/file_system/s3_test.exs +++ b/test/livebook/file_system/s3_test.exs @@ -254,6 +254,26 @@ defmodule Livebook.FileSystem.S3Test do assert :ok = FileSystem.write(file_system, file_path, content) end + + # Google Cloud Storage XML API returns this type of response. + test "returns success when the status is 200 even if the content type is text/html", %{ + bypass: bypass + } do + content = "content" + + Bypass.expect_once(bypass, "PUT", "/mybucket/dir/file.txt", fn conn -> + assert {:ok, ^content, conn} = Plug.Conn.read_body(conn) + + conn + |> Plug.Conn.put_resp_content_type("text/html; charset=UTF-8") + |> Plug.Conn.resp(200, "") + end) + + file_system = S3.new(bucket_url(bypass.port), "key", "secret") + file_path = "/dir/file.txt" + + assert :ok = FileSystem.write(file_system, file_path, content) + end end describe "FileSystem.create_dir/2" do