我正在关注Phoenix Guide on Plugs创建我自己的Module Plug
,从会话中加载当前用户。使用插件模块时未分配@user
,但在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: ....
答案 0 :(得分:15)
Plug.Conn.assign
返回已修改的连接。由于Elixir中的所有数据都是不可变的,因此无法修改旧数据。在你的情况下,你丢弃了assign
的结果,并且conn仍然指向旧的连接。你可能想要这样的东西:
conn = assign(conn, :user, user)
这将重新绑定conn
变量以指向修改后的连接结构。当然,如果assign(conn, :user, user)
将成为函数中的最后一个表达式,它也会起作用,因为它的结果将被返回。