defmodule Bar do
@after_compile __MODULE__
defmacro __before_compile__(_env) do
IO.inspect("#{__MODULE__}.__before_compile__")
end
def __after_compile__(_env, _bytecode) do
IO.inspect("#{__MODULE__}.__after_compile__")
end
end
defmodule Foo do
require Bar
@before_compile Bar
@after_compile __MODULE__
def __after_compile__(_env, _bytecode) do
IO.inspect("#{__MODULE__}.__after_compile__")
end
end
Bar is compiled, you see its output of Bar.__after_compile__
Foo is going to get compiled, it calls Bar.__before_compile__, you see the output Bar.__before_compile__, because __MODULE__ is Bar in that context. You wan’t to check env.mod to see the callers module.
Foo gets actuially compiled, its after compile gets called and you see its Foo.__after_compile__.
Thank you! Can I ask something more as you seem to have good knowledge of macros?
defmodule Baz do
defmacro __using__(_opts), do: Module.register_attribute(__MODULE__, :attr, [])
end
defmodule MyApp do
use Baz
end
== Compilation error in file test.exs ==
** (ArgumentError) could not call Module.register_attribute/3 because the module Baz is already compiled
Why does it say Baz is already compiled, when I’m executing __using__? As to my understanding, __using__ is run on compile time, so the file should not be compiled yet?
Baz is already compiled when used in MyApp. MyApp is not yet compiled. In macros it’s important to be aware if you‘re working in the context of the macro (__MODULE__ is the module with the macro definition) or if you‘re working on the AST of the macros result, which is effectively becoming part of the module calling the macro (__MODULE__ refers to the module calling the macro).
So if I understand right, if a module A calls __using__ of a module B, then __using__ macro of B will be executed after B is compiled.
Then if I try to register an attribute on another module:
defmodule Bar do
end
defmodule Baz do
defmacro __using__(_), do: Module.register_attribute(Bar, :attr, [])
end
defmodule MyApp do
use Baz
end
But again:
** (ArgumentError) could not call Module.register_attribute/3 because the module Bar is already compiled
So Bar is already compiled, but how can I tell that it should not compile before registering the attributes in the __using__ macro of Baz?
I tried require MyApp into Bar, which I thought means “wait for MyApp to be compiled before the compilation of Bar”, then I am sure that Bar will not be compiled yet, but obviously got it all wrong again.
In your example there’s no compile time relationship between MyApp(/Baz) and Bar, so they’re compiled in parallel. I’m not sure though why exactly it doesn’t work with require MyApp added.
But I guess more important would be to know why you want to do that. Why should calling use Baz register a module attribute on a third module?
These were indeed all examples with Foo/Bar/Baz placeholders because I did not want to bloat the problem with application-specific details. But here is what I try to do:
I want my library to receive options at compile time, e.g. the user can pass option otp_app: :my_app to TheLibrary somehow. Often this is done with use but I don’t need to inject anything in a user’s module.
Then in some module of the library, I need to use those options in a __before_compile__/1 macro (because based on those options, I want to generate some functions into that module in the library).
I could not find a way for __before_compile__ to get access to the option; keep having the blah-blah-blah-already-compiled.
I do not know what is the best way for the user to pass the option to the library.
As the option is already compile time only you can just as easily use the app env for configuration and the value will be available everywhere without intruducing compile time dependencies just to pass along configuration values.
Again, you need to look into the callers environment. Macros get passed __CALLER__ as a magic variable, so you probably want to Module.register_attribute(__CALLER__.mod, :attr, []) or in this case even more appropriate, as you do not make it accumulating or persistent:
defmacro __using__(_opts) do
quote do
@attr []
end
end
Or instead of the empty list use any other default value you prefer.
Beeing aware of the current context you are in is important for macros.
Perhaps give Metaprogramming Elixir by @chrismccord a read? I only skimmed about the first quarter of it, but it helped me to understand macros enough for my usecase.
defmodule A do
defmacro __using__(opts) do
otp_app = opts[:otp_app] # this gets set to :foo
quote do # note 'use' is called by B, so the next two lines live in B.
@otp_app unquote(otp_app) # becomes @otp_app :foo
@before_compile A
end
end
defmacro __before_compile__(_) do
__CALLER__.module
|> IO.inspect(label: "module")
|> Module.get_attribute(:otp_app)
|> IO.inspect(label: "otp app")
quote do end
end
end
defmodule B do
use A, otp_app: :foo
end
compile-time output:
module: B
otp app: :foo
If it’s clearer, this is effectively equivalent:
defmodule A do
defmacro __using__(opts) do
otp_app = opts[:otp_app]
quote do
@otp_app unquote(otp_app)
end
end
defmacro __before_compile__(_) do
__CALLER__.module
|> IO.inspect(label: "module")
|> Module.get_attribute(:otp_app)
|> IO.inspect(label: "otp app")
quote do end
end
end
defmodule B do
use A, otp_app: :foo
@before_compile A
end