Ale*_*hin 6 elixir phoenix-framework
有一些用Phoenix制作的API,API与JSON一起使用.
但是,当您测试它并发送JSON curl失败时,因为Phoenix不会将请求解析为JSON.您需要显式添加application/json标头curl.我想让它更强大,并告诉Phoenix总是将所有请求解析为JSON.
有没有办法迫使Phoenix总是将请求视为JSON并将其解析为JSON?
UPDATE
我尝试使用插件来设置@AbM建议的请求标头,并在Router中使用以下代码:
def set_format conn, format do Plug.Conn.put_private conn, :phoenix_format, format end
def put_req_header conn, {key, value} do Plug.Conn.put_req_header conn, key, value end
pipeline :api do
plug :put_req_header, {"accept", "application/json"}
plug :put_req_header, {"content-type", "application/json"}
plug :set_format, "json"
plug :accepts, ["json"]
end
Run Code Online (Sandbox Code Playgroud)
请求已使用CURL
curl -X POST http://localhost:4000/api/upload -d '{"key": "value"}'
Run Code Online (Sandbox Code Playgroud)
连接看起来像:
%Plug.Conn{adapter: {Plug.Adapters.Cowboy.Conn, :...}, assigns: %{},
before_send: [#Function<1.93474994/1 in Plug.Logger.call/2>,
#Function<0.119481890/1 in Phoenix.LiveReloader.before_send_inject_reloader/1>],
body_params: %{"{\"key\": \"value\"}" => nil},
cookies: %Plug.Conn.Unfetched{aspect: :cookies}, halted: false,
host: "localhost", method: "POST", owner: #PID<0.483.0>,
params: %{"{\"key\": \"value\"}" => nil},
path_info: ["api", "upload"], peer: {{127, 0, 0, 1}, 58408},
port: 4000,
private: %{App.Router => {[], %{}},
:phoenix_action => :upload,
:phoenix_controller => ApiController, :phoenix_endpoint => App.Endpoint,
:phoenix_format => "json", :phoenix_layout => {LayoutView, :app},
:phoenix_pipelines => [:api],
:phoenix_route => #Function<8.59735990/1 in App.Router.match_route/4>,
:phoenix_router => App.Router, :phoenix_view => ApiView,
:plug_session_fetch => #Function<1.89562996/1 in Plug.Session.fetch_session/1>},
query_params: %{},
query_string: "", remote_ip: {127, 0, 0, 1},
req_cookies: %Plug.Conn.Unfetched{aspect: :cookies},
req_headers: [{"user-agent", "curl/7.37.1"}, {"host", "localhost:4000"},
{"accept", "application/json"}, {"content-length", "16"},
{"content-type", "application/json"}],
request_path: "/api/upload", resp_body: nil, resp_cookies: %{},
resp_headers: [{"cache-control", "max-age=0, private, must-revalidate"},
{"x-request-id", "xxx"}], scheme: :http,
script_name: [],
secret_key_base: "xxx",
state: :unset, status: nil}
Run Code Online (Sandbox Code Playgroud)
如果我将-H "Content-Type: application/json"参数添加到CURL,它可以工作,没有它它不起作用.
小智 3
如果有人会用谷歌搜索并想要实现这种行为。
lib/%APP_NAME%/endpoint.ex
plug Plug.Head
# add custom plug
plug :set_format, "json"
Run Code Online (Sandbox Code Playgroud)
定义它:
defp set_format(conn, format) do
if Regex.match?(~r/^api.*/, conn.host) do
Plug.Conn.put_private conn, :phoenix_format, format
else
conn
end
end
Run Code Online (Sandbox Code Playgroud)
在这个例子中,我们有肮脏的黑客,它将强制子域的 JSON 格式api
不建议这样做,但由于 Phoenix 随时强制执行 HTML,此 hack 修复了荒谬的行为,例如: Elixir.Phoenix.Router.NoRouteError for pipeline :apito show 404.json 而不是 404.html
| 归档时间: |
|
| 查看次数: |
1522 次 |
| 最近记录: |