Plug.Conn.assign не работает при вызове из Connect Pip Plug

Я следую Phoenix Guide on Plugs, чтобы создать свой собственный Module Plug, который загружает текущего пользователя из сеанса. @user не назначается при использовании модуля Plug, но отлично работает, когда я называю его частной функцией в router.ex.

Это мой web/router:

defmodule MyApp.Router do
  use MyApp.Web, :router

  pipeline :browser do
    plug :accepts, ["html"]
    plug :fetch_session
    plug :fetch_flash
    plug :protect_from_forgery
    plug MyApp.Plugs.User
  end

  # Scopes and Routes...

end

Это мой модуль (в web/plugs/user.ex):

defmodule MyApp.Plugs.User do
  import Plug.Conn

  def init(default), do: default

  def call(conn, default) do
    user = %{
      id:       get_session(conn, :user_id),
      username: get_session(conn, :username)
    }

    assign(conn, :user, user)

    IO.inspect conn
  end
end

Я попытался проверить его, чтобы убедиться, что это действительно назначено, но это не так:

%Plug.Conn{adapter: {Plug.Adapters.Cowboy.Conn, :...}, assigns: %{},
 before_send: [#Function<1.75757453/1 in Plug.CSRFProtection.call/2>,
  #Function<1.30521811/1 in Phoenix.Controller.fetch_flash/2>,
  #Function<0.39713025/1 in Plug.Session.before_send/2>,
  #Function<1.7608073/1 in Plug.Logger.call/2>,
  #Function<0.72065156/1 in Phoenix.LiveReloader.before_send_inject_reloader/1>],
 body_params: %{},
 cookies: ....

Ответ 1

Plug.Conn.assign возвращает измененное соединение. Поскольку все данные в Elixir неизменяемы, невозможно изменить старый. В вашем случае вы отбрасываете результат assign, а conn все еще указывает на старое соединение. Вероятно, вы хотите что-то вроде:

conn = assign(conn, :user, user)

Это приведет к восстановлению переменной conn, чтобы указать на измененную структуру соединения. Конечно, это также сработает, если assign(conn, :user, user) будет последним выражением в вашей функции, так как его результат будет возвращен.