背景
我有一个应用程序,它是一个带有牛仔的网络服务器并使用 Plugs。由于这个应用程序是继承的,除非我们重新制作整个应用程序,否则使用 Phoenix 是不可能的,这不会发生。
我的目标是不要将所有内容都放在一个巨大的文件中,而是要有几个插头并通过管道连接它们。
代码
假设我有一个主路由器插件,如下所示:
defmodule MyApp.Web.Router do
use Plug.Router
plug(:match)
forward "/check", to: MyApp.Route.Check
forward "/dispatch", to: MyApp.Plug.Dispatch
end
所以在这里我有两件事。endpoint 的 Route ,/check
如下所示:
defmodule MyApp.Route.Check do
use Plug.Router
plug(:dispatch)
get "/", do: send_resp(conn, 200, "ok")
end
一个 Plug 管道/dispatch
看起来像这样:
defmodule MyApp.Plug.Dispatch do
use Plug.Builder
plug(Plug.Parsers, parsers: [:urlencoded]) #parses parameters
plug(MyApp.Plug.Metrics) #exposes /metrics path
plug(Cipher.ValidatePlug) #typical URL validation
plug(MyApp.Route.Dispatch) #forwards to dispatch Route
end
此管道解析参数,通知度量服务,验证请求,然后将其发送到正确的路由器,如下所示:
defmodule MyApp.Route.Dispatch do
use Plug.Router
plug(:dispatch)
get "/", do: send_resp(conn, 200, "Info dispatched")
end
问题
这里的问题是没有任何效果。从字面上看,如果我启动应用程序并尝试访问最愚蠢的端点(/check
),代码就会出现错误:
17:44:03.330 [error] #PID<0.402.0> running MyApp.Web.Router (connection #PID<0.401.0>, stream id 1) terminated
Server: localhost:4003 (http)
Request: GET /check
** (exit) an exception was raised:
** (Plug.Conn.NotSentError) a response was neither set nor sent from the connection
(plug_cowboy) lib/plug/cowboy/handler.ex:37: Plug.Cowboy.Handler.maybe_send/2
(plug_cowboy) lib/plug/cowboy/handler.ex:13: Plug.Cowboy.Handler.init/2
(...)
我现在一整天都在阅读文档,这就是我所能得到的。该应用程序非常简单,几乎是插件的 hello world:
https://elixirschool.com/en/lessons/specifics/plug/
但MyApp.Web.Router
不是他们使用的那个。
在这里可以看到一个 MWE:
https://github.com/Fl4m3Ph03n1x/plug-pipeline-problem
我究竟做错了什么?