1

背景

我有一个应用程序,它是一个带有牛仔的网络服务器并使用 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

我究竟做错了什么?

4

1 回答 1

3

路线在Example.Route.Dispatch和上不匹配Example.Route.Check。要解决此问题,您需要进行两项更改:

defmodule Example.Route.Dispatch do
  use Plug.Router

  plug(:dispatch)

  get "/*glob", do: send_resp(conn, 200, "Info dispatched")
end

defmodule Example.Route.Check do
  use Plug.Router

  plug :match
  plug :dispatch

  get "/*glob" do
    send_resp(conn, 200, "ok")
  end
end

或者,您可以这样做:https ://github.com/Fl4m3Ph03n1x/plug-pipeline-problem/pull/1

要了解更多关于它为什么不匹配的信息,您可以在“catch all”匹配中使用Plug.Router.match_path/1 。

于 2018-11-20T10:38:31.640 回答