当用作依赖项时,我的混合项目未加载 config/* 宏
My mix project is not loading config/* macros when used as a dependency
几天前,我看到了来自 José Valim Mocks and explicit contracts and decided to implement those patterns into my own project wikix 的 post。
这些是我的配置/*.exs files。
如您所见(在 test.exs 和 prod.exs 中)我定义了几个环境变量 "http_client" 和 "user_agent"。这样,当我测试模块时我使用模拟,当模块投入生产时我使用真正的 http 客户端。
问题是:当我尝试在我的 phoenix 应用程序中使用 wikix 作为依赖项时,出于某种原因 mix 忽略了我的 wikix/config/prod.exs 文件,因此从未定义 http 客户端(在 wikix 中)。
在我的 phoenix 应用程序中,我得到了 nil.HTTPClient,而不是 Wikix.HTTPClient。
这是我的 phoenix/mix.exs 文件
defmodule GraapyApi.Mixfile do
use Mix.Project
def project do
[app: :graapy_api,
version: "0.0.1",
elixir: "~> 1.0",
elixirc_paths: elixirc_paths(Mix.env),
compilers: [:phoenix, :gettext] ++ Mix.compilers,
build_embedded: Mix.env == :prod,
start_permanent: Mix.env == :prod,
aliases: aliases, deps: deps
]
end
# Configuration for the OTP application.
#
# Type `mix help compile.app` for more information.
def application do
[mod: {GraapyApi, []},
applications: [:phoenix, :cowboy, :logger, :gettext,
:phoenix_ecto, :mongodb_ecto, :wikix]]
end
# Specifies which paths to compile per environment.
defp elixirc_paths(:test), do: ["lib", "web", "test/support"]
defp elixirc_paths(_), do: ["lib", "web"]
# Specifies your project dependencies.
#
# Type `mix help deps` for examples and options.
defp deps do
[
{:phoenix, "~> 1.1.4"},
{:mongodb_ecto, ">= 0.0.0"},
{:phoenix_ecto, "~> 2.0"},
{:gettext, "~> 0.9"},
{:cowboy, "~> 1.0"},
{:dogma, "~> 0.1", only: :dev},
{:credo, "~> 0.3", only: [:dev, :test]},
{:wikix, github: "graapy/wikix"}
]
end
# Aliases are shortcut or tasks specific to the current project.
# For example, to create, migrate and run the seeds file at once:
#
# $ mix ecto.setup
#
# See the documentation for `Mix` for more info on aliases.
defp aliases do
["ecto.setup": ["ecto.create", "ecto.migrate", "run priv/repo/seeds.exs"]
我无法完全解读您的问题,但我认为 this question 的答案可能正是您要找的。具体这部分:
You can read keyword lists stored in a *.exs file, using
Mix.Config.read(path).
这是预期的行为。父项目(您的 phoenix 应用程序)未使用 deps 的配置。如果依赖项需要配置你需要在父应用程序中指定它,所以你需要复制:
config :wikix, http_client: Wikix.HTTPClient
config :wikix, user_agent: [{"User-agent", "tugorez tugorez@gmail.com"}]
到您的 Phoenix 应用。
提供默认值通常是个好主意,因此例如在 Wikix
应用程序中,您可以这样设置:
defmodule Wikix.SomeModule do
@httpclient Wikix.HTTPClient
call(first_arg, second_arg, httpclient \ @httclient) do
...
您现在可以阅读配置并将客户端作为最后一个参数传递,就像在有关模拟的文章中一样。
但您也可以确保您的 Phoenix 应用程序无需配置任何内容,并且在调用函数时将使用默认客户端而不指定最后一个参数。
您的 Phoenix 应用程序可能甚至不应该知道 Wikix
"in memory client",因此进行此类默认设置是有意义的。
非常感谢大家。 tkowal,我接受了你关于默认值的想法,我最终实现了这个:
defmodule Wikix.SomeModule do
@http_client Application.get_env( :wikix, :http_client, Wikix.HTTPClient)
...
如果我没记错,Application.get_env 中的第三个参数是默认值,以防第二个参数为 nil 。
几天前,我看到了来自 José Valim Mocks and explicit contracts and decided to implement those patterns into my own project wikix 的 post。
这些是我的配置/*.exs files。 如您所见(在 test.exs 和 prod.exs 中)我定义了几个环境变量 "http_client" 和 "user_agent"。这样,当我测试模块时我使用模拟,当模块投入生产时我使用真正的 http 客户端。
问题是:当我尝试在我的 phoenix 应用程序中使用 wikix 作为依赖项时,出于某种原因 mix 忽略了我的 wikix/config/prod.exs 文件,因此从未定义 http 客户端(在 wikix 中)。
在我的 phoenix 应用程序中,我得到了 nil.HTTPClient,而不是 Wikix.HTTPClient。
这是我的 phoenix/mix.exs 文件
defmodule GraapyApi.Mixfile do
use Mix.Project
def project do
[app: :graapy_api,
version: "0.0.1",
elixir: "~> 1.0",
elixirc_paths: elixirc_paths(Mix.env),
compilers: [:phoenix, :gettext] ++ Mix.compilers,
build_embedded: Mix.env == :prod,
start_permanent: Mix.env == :prod,
aliases: aliases, deps: deps
]
end
# Configuration for the OTP application.
#
# Type `mix help compile.app` for more information.
def application do
[mod: {GraapyApi, []},
applications: [:phoenix, :cowboy, :logger, :gettext,
:phoenix_ecto, :mongodb_ecto, :wikix]]
end
# Specifies which paths to compile per environment.
defp elixirc_paths(:test), do: ["lib", "web", "test/support"]
defp elixirc_paths(_), do: ["lib", "web"]
# Specifies your project dependencies.
#
# Type `mix help deps` for examples and options.
defp deps do
[
{:phoenix, "~> 1.1.4"},
{:mongodb_ecto, ">= 0.0.0"},
{:phoenix_ecto, "~> 2.0"},
{:gettext, "~> 0.9"},
{:cowboy, "~> 1.0"},
{:dogma, "~> 0.1", only: :dev},
{:credo, "~> 0.3", only: [:dev, :test]},
{:wikix, github: "graapy/wikix"}
]
end
# Aliases are shortcut or tasks specific to the current project.
# For example, to create, migrate and run the seeds file at once:
#
# $ mix ecto.setup
#
# See the documentation for `Mix` for more info on aliases.
defp aliases do
["ecto.setup": ["ecto.create", "ecto.migrate", "run priv/repo/seeds.exs"]
我无法完全解读您的问题,但我认为 this question 的答案可能正是您要找的。具体这部分:
You can read keyword lists stored in a *.exs file, using Mix.Config.read(path).
这是预期的行为。父项目(您的 phoenix 应用程序)未使用 deps 的配置。如果依赖项需要配置你需要在父应用程序中指定它,所以你需要复制:
config :wikix, http_client: Wikix.HTTPClient
config :wikix, user_agent: [{"User-agent", "tugorez tugorez@gmail.com"}]
到您的 Phoenix 应用。
提供默认值通常是个好主意,因此例如在 Wikix
应用程序中,您可以这样设置:
defmodule Wikix.SomeModule do
@httpclient Wikix.HTTPClient
call(first_arg, second_arg, httpclient \ @httclient) do
...
您现在可以阅读配置并将客户端作为最后一个参数传递,就像在有关模拟的文章中一样。
但您也可以确保您的 Phoenix 应用程序无需配置任何内容,并且在调用函数时将使用默认客户端而不指定最后一个参数。
您的 Phoenix 应用程序可能甚至不应该知道 Wikix
"in memory client",因此进行此类默认设置是有意义的。
非常感谢大家。 tkowal,我接受了你关于默认值的想法,我最终实现了这个:
defmodule Wikix.SomeModule do
@http_client Application.get_env( :wikix, :http_client, Wikix.HTTPClient)
...
如果我没记错,Application.get_env 中的第三个参数是默认值,以防第二个参数为 nil 。