In :test
env I have the usual logger config 
# Print only warnings and errors during test
config :logger, level: :warning
which means I have a nice clean console if everything passes (and I use capture_log
and with_log
when necessary).
Something I would very much enjoy is if I could somehow get the :info
logs for failed tests only.
It has to be possible, right? Maybe some kind of custom logger that holds the :info
logs in memory and only prints them if current test fails?
If it is possible, it would be really nice to be able to write logger config like this:
My wish 
config :logger, level: [passed: :warning, failed: :info]
Here’s how I have my config/test.exs
set up:
config :logger, level: System.get_env("LOG_LEVEL", "warning") |> String.to_existing_atom()
Using this, you could run mix test
to get the standard warning/error logs, then you could run LOG_LEVEL=info mix test --failed
to get more detailed logs for your failed tests (could also use --trace
to make it easier to tell which logs apply to which tests, perhaps).
It’s not quite what you’re asking for, but could get you something close to the desired results with very little work involved.
I usually put a Logger.configure(level: level)
in the test in question.
2 Likes
Would that need to be cleared/reset at the end of the test?
These are useful and good techniques.
However, my wish is for a 0-effort solution that will also catch, say, a “flakey” test that fails sometimes and only when run as part of an entire async test suite. I do not want to have to manually change the log level and re-run tests. I want…magic!
1 Like
You could generally enable logs and use ExUnit.Case — ExUnit v1.18.3 to show them only in case of errors.
There is however no magic after the fact getting access to already dropped logs.
1 Like
In my case I have:
# config/test.exs
config :logger, :console,
level: String.to_atom(System.get_env("LOGGER_LEVEL", "none"))
# test/test_helper.exs
logs =
case System.get_env("TEST_LOGS", "all") do
level when level in ~w[all true] ->
true
level when level in ~w[emergency alert critical error warning notice info debug] ->
[level: String.to_existing_atom(level)]
"warn" ->
[level: :warning]
level when level in ~w[none disabled false] ->
false
end
ExUnit.start(
capture_log: logs
)
And I have 2 separate variable for logs in the terminal and in the error reports.
1 Like
Only marginally on topic here but this is one of the beautiful things about tests which are completely deterministic. When designed correctly you can retroactively re-enable (sometimes expensive) logging without changing the result. Obviously this is difficult to implement in practice, and not always possible 