当用作依赖项时,我的混合项目未加载 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 。