Merge pull request #117 from elixirkoans/split-display
Extract smaller modules from display.
This commit is contained in:
@@ -1,13 +0,0 @@
|
|||||||
defmodule Colours do
|
|
||||||
alias IO.ANSI
|
|
||||||
|
|
||||||
def red(str), do: colourize(ANSI.red, str)
|
|
||||||
|
|
||||||
def cyan(str), do: colourize(ANSI.cyan, str)
|
|
||||||
|
|
||||||
def green(str), do: colourize(ANSI.green, str)
|
|
||||||
|
|
||||||
defp colourize(color, message) do
|
|
||||||
Enum.join([color, message, ANSI.reset], "")
|
|
||||||
end
|
|
||||||
end
|
|
@@ -1,75 +1,39 @@
|
|||||||
defmodule Display do
|
defmodule Display do
|
||||||
alias IO.ANSI
|
alias IO.ANSI
|
||||||
|
alias Display.ProgressBar
|
||||||
@no_value :ex_unit_no_meaningful_value
|
alias Display.Intro
|
||||||
@progress_bar_length 30
|
alias Display.Failure
|
||||||
|
alias Display.Notifications
|
||||||
|
|
||||||
def invalid_koan(koan, modules) do
|
def invalid_koan(koan, modules) do
|
||||||
koans_names = module_names(modules)
|
Notifications.invalid_koan(koan, modules)
|
||||||
IO.puts("Did not find koan #{name(koan)} in " <> koans_names )
|
|> IO.puts
|
||||||
end
|
end
|
||||||
|
|
||||||
defp module_names(modules) do
|
|
||||||
modules
|
|
||||||
|> Enum.map(&Atom.to_string/1)
|
|
||||||
|> Enum.map(&name/1)
|
|
||||||
|> Enum.join(", ")
|
|
||||||
|> Colours.red
|
|
||||||
end
|
|
||||||
|
|
||||||
defp name("Elixir." <> module), do: module
|
|
||||||
defp name(module), do: name(Atom.to_string(module))
|
|
||||||
|
|
||||||
def show_failure(failure, module, name) do
|
def show_failure(failure, module, name) do
|
||||||
IO.puts(format(failure, module, name))
|
format(failure, module, name)
|
||||||
|
|> IO.puts
|
||||||
end
|
end
|
||||||
|
|
||||||
def format(failure, module, name) do
|
def format(failure, module, name) do
|
||||||
"""
|
"""
|
||||||
#{intro(module, Tracker.visited)}
|
#{Intro.intro(module, Tracker.visited)}
|
||||||
Now meditate upon #{format_module(module)}
|
Now meditate upon #{format_module(module)}
|
||||||
#{progress_bar(Tracker.summarize)}
|
#{ProgressBar.progress_bar(Tracker.summarize)}
|
||||||
----------------------------------------
|
----------------------------------------
|
||||||
#{name}
|
#{name}
|
||||||
#{format_failure(failure)}
|
#{Failure.format_failure(failure)}
|
||||||
"""
|
"""
|
||||||
end
|
end
|
||||||
|
|
||||||
defp intro(module, modules) do
|
def show_compile_error(error) do
|
||||||
if not module in modules do
|
Failure.show_compile_error(error)
|
||||||
show_intro(module.intro)
|
|
||||||
end
|
|
||||||
end
|
|
||||||
|
|
||||||
defp show_intro(""), do: nil
|
|
||||||
defp show_intro(message) do
|
|
||||||
message <> "\n"
|
|
||||||
|> Colours.green
|
|
||||||
|> IO.puts
|
|> IO.puts
|
||||||
end
|
end
|
||||||
|
|
||||||
def progress_bar(%{current: current, total: total}) do
|
|
||||||
arrow = caluculate_progress(current, total) |> build_arrow
|
|
||||||
|
|
||||||
"|" <> String.ljust(arrow, @progress_bar_length) <> "| #{current} of #{total}"
|
|
||||||
end
|
|
||||||
|
|
||||||
defp caluculate_progress(current, total) do
|
|
||||||
round( (current/total) * @progress_bar_length)
|
|
||||||
end
|
|
||||||
|
|
||||||
defp build_arrow(0), do: ""
|
|
||||||
defp build_arrow(length) do
|
|
||||||
String.duplicate("=", length-1) <> ">"
|
|
||||||
end
|
|
||||||
|
|
||||||
def show_compile_error(error) do
|
|
||||||
IO.puts("")
|
|
||||||
format_error(error) |> IO.puts
|
|
||||||
end
|
|
||||||
|
|
||||||
def congratulate do
|
def congratulate do
|
||||||
IO.puts(Colours.green("\nYou have learned much. You must find your own path now."))
|
Notifications.congratulate
|
||||||
|
|> IO.puts
|
||||||
end
|
end
|
||||||
|
|
||||||
def clear_screen do
|
def clear_screen do
|
||||||
@@ -79,34 +43,6 @@ defmodule Display do
|
|||||||
end
|
end
|
||||||
end
|
end
|
||||||
|
|
||||||
defp format_failure(%{error: %ExUnit.AssertionError{expr: @no_value, message: message}, file: file, line: line}) do
|
|
||||||
"""
|
|
||||||
#{Colours.cyan("Assertion failed in #{file}:#{line}")}
|
|
||||||
#{Colours.red(message)}
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
defp format_failure(%{error: %ExUnit.AssertionError{expr: expr}, file: file, line: line}) do
|
|
||||||
format_assertion_error(expr, file, line)
|
|
||||||
end
|
|
||||||
defp format_failure(%{error: error, file: file, line: line}) do
|
|
||||||
"""
|
|
||||||
#{Colours.cyan("Error in #{file}:#{line}")}
|
|
||||||
#{format_error(error)}
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
defp format_assertion_error(error, file, line) do
|
|
||||||
"""
|
|
||||||
#{Colours.cyan("Assertion failed in #{file}:#{line}")}
|
|
||||||
#{Colours.red(Macro.to_string(error))}
|
|
||||||
"""
|
|
||||||
end
|
|
||||||
|
|
||||||
defp format_error(error) do
|
|
||||||
trace = System.stacktrace |> Enum.take(2)
|
|
||||||
Colours.red(Exception.format(:error, error, trace))
|
|
||||||
end
|
|
||||||
|
|
||||||
defp format_module(module) do
|
defp format_module(module) do
|
||||||
Module.split(module) |> List.last
|
Module.split(module) |> List.last
|
||||||
end
|
end
|
||||||
|
36
lib/display/colours.ex
Normal file
36
lib/display/colours.ex
Normal file
@@ -0,0 +1,36 @@
|
|||||||
|
defmodule Display.Paint do
|
||||||
|
def red(str), do: painter().red(str)
|
||||||
|
|
||||||
|
def cyan(str), do: painter().cyan(str)
|
||||||
|
|
||||||
|
def green(str), do: painter().green(str)
|
||||||
|
|
||||||
|
defp painter do
|
||||||
|
case Mix.env do
|
||||||
|
:test -> Display.Uncoloured
|
||||||
|
_ -> Display.Colours
|
||||||
|
end
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
defmodule Display.Colours do
|
||||||
|
alias IO.ANSI
|
||||||
|
|
||||||
|
def red(str), do: colourize(ANSI.red, str)
|
||||||
|
|
||||||
|
def cyan(str), do: colourize(ANSI.cyan, str)
|
||||||
|
|
||||||
|
def green(str), do: colourize(ANSI.green, str)
|
||||||
|
|
||||||
|
defp colourize(color, message) do
|
||||||
|
Enum.join([color, message, ANSI.reset], "")
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
defmodule Display.Uncoloured do
|
||||||
|
def red(str), do: str
|
||||||
|
|
||||||
|
def cyan(str), do: str
|
||||||
|
|
||||||
|
def green(str), do: str
|
||||||
|
end
|
37
lib/display/failure.ex
Normal file
37
lib/display/failure.ex
Normal file
@@ -0,0 +1,37 @@
|
|||||||
|
defmodule Display.Failure do
|
||||||
|
alias Display.Paint
|
||||||
|
|
||||||
|
@no_value :ex_unit_no_meaningful_value
|
||||||
|
|
||||||
|
def format_failure(%{error: %ExUnit.AssertionError{expr: @no_value, message: message}, file: file, line: line}) do
|
||||||
|
"""
|
||||||
|
#{Paint.cyan("Assertion failed in #{file}:#{line}")}
|
||||||
|
#{Paint.red(message)}
|
||||||
|
"""
|
||||||
|
end
|
||||||
|
def format_failure(%{error: %ExUnit.AssertionError{expr: expr}, file: file, line: line}) do
|
||||||
|
format_assertion_error(expr, file, line)
|
||||||
|
end
|
||||||
|
def format_failure(%{error: error, file: file, line: line}) do
|
||||||
|
"""
|
||||||
|
#{Paint.cyan("Error in #{file}:#{line}")}
|
||||||
|
#{format_error(error)}
|
||||||
|
"""
|
||||||
|
end
|
||||||
|
|
||||||
|
defp format_assertion_error(error, file, line) do
|
||||||
|
"""
|
||||||
|
#{Paint.cyan("Assertion failed in #{file}:#{line}")}
|
||||||
|
#{Paint.red(Macro.to_string(error))}
|
||||||
|
"""
|
||||||
|
end
|
||||||
|
|
||||||
|
defp format_error(error) do
|
||||||
|
trace = System.stacktrace |> Enum.take(2)
|
||||||
|
Paint.red(Exception.format(:error, error, trace))
|
||||||
|
end
|
||||||
|
|
||||||
|
def show_compile_error(error) do
|
||||||
|
format_error(error)
|
||||||
|
end
|
||||||
|
end
|
16
lib/display/intro.ex
Normal file
16
lib/display/intro.ex
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
defmodule Display.Intro do
|
||||||
|
alias Display.Paint
|
||||||
|
|
||||||
|
def intro(module, modules) do
|
||||||
|
if not module in modules do
|
||||||
|
show_intro(module.intro)
|
||||||
|
else
|
||||||
|
""
|
||||||
|
end
|
||||||
|
end
|
||||||
|
|
||||||
|
def show_intro(message) do
|
||||||
|
message <> "\n"
|
||||||
|
|> Paint.green
|
||||||
|
end
|
||||||
|
end
|
23
lib/display/notification.ex
Normal file
23
lib/display/notification.ex
Normal file
@@ -0,0 +1,23 @@
|
|||||||
|
defmodule Display.Notifications do
|
||||||
|
alias Display.Paint
|
||||||
|
|
||||||
|
def congratulate do
|
||||||
|
Paint.green("\nYou have learned much. You must find your own path now.")
|
||||||
|
end
|
||||||
|
|
||||||
|
def invalid_koan(koan, modules) do
|
||||||
|
koans_names = module_names(modules)
|
||||||
|
"Did not find koan #{name(koan)} in " <> koans_names
|
||||||
|
end
|
||||||
|
|
||||||
|
defp module_names(modules) do
|
||||||
|
modules
|
||||||
|
|> Enum.map(&Atom.to_string/1)
|
||||||
|
|> Enum.map(&name/1)
|
||||||
|
|> Enum.join(", ")
|
||||||
|
|> Paint.red
|
||||||
|
end
|
||||||
|
|
||||||
|
defp name("Elixir." <> module), do: module
|
||||||
|
defp name(module), do: name(Atom.to_string(module))
|
||||||
|
end
|
19
lib/display/progress_bar.ex
Normal file
19
lib/display/progress_bar.ex
Normal file
@@ -0,0 +1,19 @@
|
|||||||
|
defmodule Display.ProgressBar do
|
||||||
|
|
||||||
|
@progress_bar_length 30
|
||||||
|
|
||||||
|
def progress_bar(%{current: current, total: total}) do
|
||||||
|
arrow = caluculate_progress(current, total) |> build_arrow
|
||||||
|
|
||||||
|
"|" <> String.ljust(arrow, @progress_bar_length) <> "| #{current} of #{total}"
|
||||||
|
end
|
||||||
|
|
||||||
|
defp caluculate_progress(current, total) do
|
||||||
|
round( (current/total) * @progress_bar_length)
|
||||||
|
end
|
||||||
|
|
||||||
|
defp build_arrow(0), do: ""
|
||||||
|
defp build_arrow(length) do
|
||||||
|
String.duplicate("=", length-1) <> ">"
|
||||||
|
end
|
||||||
|
end
|
24
test/display/failure_test.exs
Normal file
24
test/display/failure_test.exs
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
defmodule FailureTests do
|
||||||
|
use ExUnit.Case
|
||||||
|
alias Display.Failure
|
||||||
|
|
||||||
|
test "assertion failure with proper expression" do
|
||||||
|
error = error(%ExUnit.AssertionError{expr: "hi"})
|
||||||
|
|
||||||
|
assert Failure.format_failure(error) == "Assertion failed in some_file.ex:42\n\"hi\"\n"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "assertion failure with message" do
|
||||||
|
error = error(%ExUnit.AssertionError{expr: :ex_unit_no_meaningful_value, message: "hola"})
|
||||||
|
|
||||||
|
assert Failure.format_failure(error) == "Assertion failed in some_file.ex:42\nhola\n"
|
||||||
|
end
|
||||||
|
|
||||||
|
defp error(error) do
|
||||||
|
%{
|
||||||
|
error: error,
|
||||||
|
file: "some_file.ex",
|
||||||
|
line: 42
|
||||||
|
}
|
||||||
|
end
|
||||||
|
end
|
13
test/display/intro_test.exs
Normal file
13
test/display/intro_test.exs
Normal file
@@ -0,0 +1,13 @@
|
|||||||
|
defmodule IntroTest do
|
||||||
|
use ExUnit.Case
|
||||||
|
|
||||||
|
alias Display.Intro
|
||||||
|
|
||||||
|
test "module not visited yet" do
|
||||||
|
assert Intro.intro(SampleKoan, []) == "There is something\n\n"
|
||||||
|
end
|
||||||
|
|
||||||
|
test "module has been visited" do
|
||||||
|
assert Intro.intro(SampleKoan, [SampleKoan]) == ""
|
||||||
|
end
|
||||||
|
end
|
9
test/display/notification_test.exs
Normal file
9
test/display/notification_test.exs
Normal file
@@ -0,0 +1,9 @@
|
|||||||
|
defmodule NotificationTest do
|
||||||
|
use ExUnit.Case
|
||||||
|
alias Display.Notifications
|
||||||
|
|
||||||
|
test "shows possible koans when a koan can not be found" do
|
||||||
|
message = Notifications.invalid_koan(SampleKoan, [PassingKoan])
|
||||||
|
assert message == "Did not find koan SampleKoan in PassingKoan"
|
||||||
|
end
|
||||||
|
end
|
@@ -1,19 +1,21 @@
|
|||||||
defmodule DisplayTest do
|
defmodule ProgressBarTest do
|
||||||
use ExUnit.Case
|
use ExUnit.Case
|
||||||
|
|
||||||
|
alias Display.ProgressBar
|
||||||
|
|
||||||
test "empty bar" do
|
test "empty bar" do
|
||||||
bar = Display.progress_bar(%{total: 12, current: 0})
|
bar = ProgressBar.progress_bar(%{total: 12, current: 0})
|
||||||
assert bar == "| | 0 of 12"
|
assert bar == "| | 0 of 12"
|
||||||
end
|
end
|
||||||
|
|
||||||
test "puts counter on the right until half the koans are complete" do
|
test "puts counter on the right until half the koans are complete" do
|
||||||
bar = Display.progress_bar(%{total: 12, current: 3})
|
bar = ProgressBar.progress_bar(%{total: 12, current: 3})
|
||||||
assert bar == "|=======> | 3 of 12"
|
assert bar == "|=======> | 3 of 12"
|
||||||
end
|
end
|
||||||
|
|
||||||
|
|
||||||
test "full bar" do
|
test "full bar" do
|
||||||
bar = Display.progress_bar(%{total: 12, current: 12})
|
bar = ProgressBar.progress_bar(%{total: 12, current: 12})
|
||||||
assert bar == "|=============================>| 12 of 12"
|
assert bar == "|=============================>| 12 of 12"
|
||||||
end
|
end
|
||||||
end
|
end
|
Reference in New Issue
Block a user