Merge pull request #29 from ukutaht/process_koans

Koans about processes.
This commit is contained in:
Uku Taht
2016-03-08 14:06:35 +02:00
7 changed files with 215 additions and 16 deletions

View File

@@ -1,4 +1,6 @@
defmodule BlankAssertions do
require ExUnit.Assertions
defmacro assert(expr) do
if contains_blank?(expr) do
code = Macro.escape(expr)
@@ -25,6 +27,19 @@ defmodule BlankAssertions do
end
end
defmacro assert_receive(expr) do
if contains_blank?(expr) do
code = Macro.escape(expr)
quote do
raise ExUnit.AssertionError, expr: unquote(code)
end
else
quote do
ExUnit.Assertions.assert_receive(unquote(expr), 100)
end
end
end
def assert(value, opts) do
ExUnit.Assertions.assert(value, opts)
end
@@ -33,6 +48,12 @@ defmodule BlankAssertions do
ExUnit.Assertions.refute(value, opts)
end
defmacro flunk(message \\ "Flunked!") do
quote do
assert false, message: unquote(message)
end
end
defp contains_blank?(expr) do
{_, blank} = Macro.prewalk(expr, false, &blank?/2)
blank

View File

@@ -7,7 +7,7 @@ defmodule Display do
IO.puts("Now meditate upon #{display_module(module)}")
IO.puts("---------------------------------------")
IO.puts(format_cyan(display_failed_assertion(module, expr)))
IO.puts(format_cyan(last_failure_location))
IO.puts(display_koan(name))
IO.puts(format_red(Macro.to_string(expr)))
end
@@ -23,8 +23,20 @@ defmodule Display do
end
end
def display_failed_assertion(module, expr) do
"Assertion failed in #{source_file(module)}:#{line_number(expr)}"
def last_failure_location do
{file, line} = System.stacktrace
|> Enum.drop_while(&in_ex_unit?/1)
|> List.first
|> extract_file_and_line
"Assertion failed in #{file}:#{line}"
end
defp in_ex_unit?({ExUnit.Assertions, _, _, _}), do: true
defp in_ex_unit?(_), do: false
defp extract_file_and_line({_, _, _, [file: file, line: line]}) do
{file, line}
end
def format_compile_error(error) do
@@ -32,16 +44,6 @@ defmodule Display do
IO.puts(format_red(Exception.format(:error, error, trace)))
end
defp line_number({_, [line: line], _}) do
line
end
defp source_file(module) do
module.__info__(:compile)
|> Dict.get(:source)
|> Path.relative_to(@current_dir)
end
defp format_red(str) do
Enum.join([ANSI.red, str, ANSI.reset], "")
end

View File

@@ -15,10 +15,10 @@ defmodule Koans do
end
end
defmacro __using__(_) do
defmacro __using__(opts) do
quote do
import Koans
require ExUnit.Assertions
import Koans
import BlankAssertions
end
end

123
lib/koans/10_processes.ex Normal file
View File

@@ -0,0 +1,123 @@
defmodule Processes do
use Koans
koan "tests run in a process!" do
assert Process.alive?(self)
end
koan "can spew out information about a process" do
information = Process.info(self)
assert information[:status] == :running
end
koan "process can send messages to itself" do
send self(), "hola!"
assert_receive "hola!"
end
koan "a common pattern is to include the sender in the message" do
pid = spawn(fn -> receive do
{:hello, sender} -> send sender, :how_are_you?
_ -> assert false
end
end)
send pid, {:hello, self()}
assert_receive :how_are_you?
end
koan "you don't have to wait forever for messages" do
parent = self()
spawn(fn -> receive do
_anything -> flunk "I really wasn't expecting messages"
after
10 -> send parent, {:waited_too_long, "I am inpatient"}
end
end)
assert_receive {:waited_too_long, "I am inpatient"}
end
koan "killing a process will terminate it" do
pid = spawn(fn -> Process.exit(self(), :kill) end)
:timer.sleep(500)
refute Process.alive?(pid)
end
koan "killing a process kills it for good" do
pid = spawn(fn -> receive do
end
end)
assert Process.alive?(pid)
Process.exit(pid, :kill)
refute Process.alive?(pid)
end
koan "can trap a signal in a child process" do
parent = self()
pid = spawn(fn ->
Process.flag(:trap_exit, true)
send parent, :ready
receive do
{:EXIT, _pid, reason} -> send parent, {:exited, reason}
end
end)
wait()
Process.exit(pid, :random_reason)
assert_receive {:exited, :random_reason}
refute Process.alive?(pid)
end
koan "quitting normally has no effect" do
pid = spawn(fn -> receive do
end
end)
Process.exit(pid, :normal)
assert Process.alive?(pid)
end
koan "quititing your own process normally does terminate it though" do
pid = spawn(fn -> receive do
:bye -> Process.exit(self(), :normal)
end
end)
assert Process.alive?(pid)
send pid, :bye
:timer.sleep(100)
refute Process.alive?(pid)
end
koan "linked processes are informed about exit signals of children when trapping those signals" do
parent = self()
spawn(fn ->
Process.flag(:trap_exit, true)
spawn_link(fn -> Process.exit(self(), :normal) end)
receive do
{:EXIT, _pid ,reason} -> send parent, {:exited, reason}
end
end)
assert_receive {:exited, :normal}
end
koan "monitoring processes are informed via messages without having trapping" do
parent = self()
spawn(fn ->
spawn_monitor(fn -> Process.exit(self(), :normal) end)
receive do
{:DOWN, _ref, :process, _pid, reason} -> send parent, {:exited, reason}
end
end)
assert_receive {:exited, :normal}
end
def wait do
receive do
:ready -> true
end
end
end

49
lib/koans/11_task.ex Normal file
View File

@@ -0,0 +1,49 @@
defmodule Tasks do
use Koans
koan "Tasks can be used for asynchronous computations with results" do
task = Task.async(fn -> 3 *3 end)
do_other_stuff()
assert Task.await(task) + 1 == 10
end
koan "if you don't need a result, use start_link/1" do
{:ok, pid} = Task.start_link(fn -> 1+1 end)
end
koan "yield returns nil if the task isn't done yet" do
handle = Task.async(fn ->
:timer.sleep(100)
3 * 3
end)
assert Task.yield(handle, 10) == nil
assert Task.await(handle) == 9
end
koan "tasks can be aborted with shutdown" do
handle = Task.async(fn ->
:timer.sleep(100)
3 * 3
end)
refute Task.shutdown(handle)
end
koan "shutdown will give you an answer if it has it" do
handle = Task.async(fn -> 3 * 3 end)
:timer.sleep(10)
assert Task.shutdown(handle) == {:ok, 9}
end
koan "you can yield to multiple tasks at once and extract the results" do
squares = [1,2,3,4]
|> Enum.map(fn(number) -> Task.async(fn -> number * number end) end)
|> Task.yield_many(100)
|> Enum.map(fn({_task,{:ok, result}}) -> result end)
assert squares == [1,4,9,16]
end
def do_other_stuff do
:timer.sleep(50)
end
end

View File

@@ -3,6 +3,8 @@ defmodule Mix.Tasks.Meditate do
alias Options
def run(args) do
ExUnit.start
Application.ensure_all_started(:ex_unit)
Application.ensure_all_started(:elixir_koans)
Code.compiler_options(ignore_module_conflict: true)
Watcher.start

View File

@@ -8,7 +8,9 @@ defmodule Runner do
Enums,
Arithmetic,
Structs,
PatternMatching
PatternMatching,
Processes,
Tasks
]
def run do