指定模块名称后如何访问与Config相关的键

Bla*_*man 0 elixir phoenix-framework

在phoenix中查看配置文件时,如果我创建这样的配置:

config :myapp,
  http: 4000
Run Code Online (Sandbox Code Playgroud)

我可以像这样在我的代码中引用该键:

Application.fetch_env!(:myapp, :http)
Run Code Online (Sandbox Code Playgroud)

有时配置似乎特定于某个模块,例如下面的MyApp.Endpoint。

这只是将配置分为几类吗?如果我也需要如何在代码中引用下面的http端口?

config :myapp, MyApp.Endpoint,
  http: [port: 4000],
  debug_errors: true
Run Code Online (Sandbox Code Playgroud)

Mát*_*áté 6

在这种情况下,您从调用中返回的fetch_env!/2关键字列表。您可以像使用地图一样访问它:

Application.fetch_env!(:myapp, :http)[:port]
Run Code Online (Sandbox Code Playgroud)

或者将其转换为 Map,然后这样做:

Application.fetch_env!(:myapp, :http) |> Map.new |> Map.get(:port)
Run Code Online (Sandbox Code Playgroud)

编辑 - 基于 @BrettBeatty 的评论

Keyword.get/2您也可以使用该功能:

:myapp |> Application.get_env(:http) |> Keyword.get(:port)
Run Code Online (Sandbox Code Playgroud)


7st*_*tud 5

有时配置似乎特定于某个模块,例如下面的MyApp.Endpoint。

这只是将配置分为几类吗?

tldr;

是。使用配置文件,您可以创建以下内容:

[
  {:ecto_repos, [MyApp.Repo]},
  {.Repo,
   [
     username: "postgres",
     password: "postgres",
     database: "myapp_dev",
     hostname: "localhost",
     pool_size: 10
   ]},

  {:http, [port: 4000]},   <===**** HERE ****
  {:included_applications, []},

  {MyAppWeb.Endpoint,     <====**** HERE ****
   [
     url: [host: "localhost"],
     secret_key_base: "rchq5yMDPqqEBzFR+wIoqpc+kNquiyNDYUp/K8aF2Yj6POl/gjfj0H0rljE06LI5",
     render_errors: [view: MyAppWeb.ErrorView, accepts: ["html", "json"]],
     pubsub: [name: Rum.PubSub, adapter: Phoenix.PubSub.PG2],
     http: [port: 4000],
     debug_errors: true,
     code_reloader: true,
     check_origin: false,
     watchers: [
       node: [
         "node_modules/webpack/bin/webpack.js",
         "--mode",
         "development",
         "--watch-stdin",
         {:cd, "/Users/7stud/phoenix_apps/book/myapp/assets"}
       ]
     ],
     live_reload: [
       patterns: [~r/priv\/static\/.*(js|css|png|jpeg|jpg|gif|svg)$/,
        ~r/priv\/gettext\/.*(po)$/, ~r/lib\/myapp_web\/views\/.*(ex)$/,
        ~r/lib\/myapp_web\/templates\/.*(eex)$/]
     ]
   ]},
Run Code Online (Sandbox Code Playgroud)

如果我也需要如何在代码中引用下面的http端口?

config :myapp, MyApp.Endpoint,
  http: [port: 4000],
  debug_errors: true
Run Code Online (Sandbox Code Playgroud)

如果查看该配置部分并删除换行符,则会得到:

config :myapp, MyApp.Endpoint, http: [port: 4000], debug_errors: true
Run Code Online (Sandbox Code Playgroud)

而且,如果加上一些括号,您将得到:

config(:myapp, MyApp.Endpoint, http: [port: 4000], debug_errors: true)
Run Code Online (Sandbox Code Playgroud)

那是一个函数调用,其中第一个参数是:myapp,第二个参数是MyApp.Endpoint,然后是这一部分:

http: [port: 4000], debug_errors: true
Run Code Online (Sandbox Code Playgroud)

是一个Keyword list。离题片刻...在长生不老药中,您可以定义如下函数:

  def go(x, y, z) do
    IO.puts x
    IO.puts y
    IO.inspect z
  end
Run Code Online (Sandbox Code Playgroud)

并这样称呼他们:

My.go(10, 20, a: 1, b: 2)
Run Code Online (Sandbox Code Playgroud)

将输出:

10
20
[a: 1, b: 2]
Run Code Online (Sandbox Code Playgroud)

这表明,如果函数调用中的最后一个参数具有特定的语法,则最后一个参数都将被收集到一个称为a的东西中,Keyword list并作为一个参数传递给函数。因此,即使看起来您正在My.go()使用四个参数调用,您实际上也确实在使用三个参数调用它。

结果,此函数调用:

config(:myapp, MyApp.Endpoint, http: [port: 4000], debug_errors: true)
Run Code Online (Sandbox Code Playgroud)

正在调用config/3函数定义。

config/3在配置文件中未定义函数时,如何调用该函数?这行:

use Mix.Config
Run Code Online (Sandbox Code Playgroud)

config/3函数定义注入文件中。这是长生不老药的问题:发生了很多不可思议的事情,这使得很难弄清函数定义的来源。

至此,您知道config/3在配置文件中正在调用一个函数:

config :myapp, MyApp.Endpoint,
  http: [port: 4000],
  debug_errors: true
Run Code Online (Sandbox Code Playgroud)

此外,use Mix.Config配置文件顶部的行是config/3可能在该模块中定义的提示,因此您可以细读Mix.Config文档。果然,config/3文档中描述了一个功能。config/2在文档中还描述了一个函数,在这里称为:

config :myapp,
  http: 4000
Run Code Online (Sandbox Code Playgroud)

如果您阅读了该文档,config/2那么似乎很容易理解它是如何工作的:使用设置键和值时config/2,您可以通过调用以下方法来检索与键关联的特定值:

Application.get_env(:myapp, key)  
Run Code Online (Sandbox Code Playgroud)

例如,

Application.get_env(:myapp, :http)
#=> 4000
Run Code Online (Sandbox Code Playgroud)

但是,当您使用设置键和值时config/3,您将无法检索与键相关联的特定值,相反,您最终将检索键/值对的整个列表:

keyword_list = Application.get_env(:myapp, module)
Run Code Online (Sandbox Code Playgroud)

config/3 创建一个嵌套结构,其中模块是键,其值是由您在配置文件中指定的键/值对组成的关键字列表,例如:

  http: [port: 4000],
  debug_errors: true
Run Code Online (Sandbox Code Playgroud)

解决问题的另一种方法是考虑嵌套地图:

%{
  myapp: %{
    http: 4000,  #<== config/2
    my_app_endpoint: %{http: [port: 4000, debug_errors: true]}  #<== config/3
  }
Run Code Online (Sandbox Code Playgroud)

...并且Application.get_env()仅允许您指定两个键,例如:

Application.get_env(:myapp, :http)  #=> 4000
Run Code Online (Sandbox Code Playgroud)

要么:

Application.get_env(:myapp, :my_app_endpoint)  #=> %{http: [port: 4000, debug_errors: true]}
Run Code Online (Sandbox Code Playgroud)

检索keyword_list后:

keyword_list = Application.get_env(:myapp, module)
Run Code Online (Sandbox Code Playgroud)

那么您必须使用您对elixir的一般知识来从关键字列表中检索与特定键相关联的值:

all = Application.get_all_env(:myapp)
IO.inspect all  #=> lots of stuff (which is not a keyword list)

keyword_list = Application.get_env(:myapp, MyApp.Endpoint)
IO.inspect keyword_list #=> a keyword list with lots of key/value pairs 
http = Keyword.get(keyword_list, :http)
IO.inspect http  #=> [port: 4000]
port = Keyword.get(http_keyword_list, :port)
IO.inspect port  #=>4000
Run Code Online (Sandbox Code Playgroud)