forked from dojo-toulouse/elixir-koans
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathabout_tuples.exs
executable file
·87 lines (70 loc) · 2.75 KB
/
about_tuples.exs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
#!/usr/bin/env elixir
ExUnit.start
defmodule About_Tuples do
use ExUnit.Case
use Koans
think "Getting tuples size is a kernel feature" do
a_tuple = {:foo, :bar}
assert tuple_size(a_tuple) == __?
end
think "Can reach tuple element with index" do
a_tuple = {:foo, :bar}
assert elem(a_tuple, 0) == __?
end
defmacrop erlang_elem(tuple, index) do
# Note: Elixir provides access to erlang primitives with atom :erlang
quote do: :erlang.element(unquote(index), unquote(tuple))
end
test "In erlang tuples and lists start at index 1, wich is different in elixir" do
a_tuple = {:foo, :bar}
foo_index = 0
assert elem(a_tuple, foo_index) == erlang_elem(a_tuple, foo_index + __?)
end
think "Can set a tuple element" do
a_tuple = {:foo, :bar}
baz_tuple = put_elem(a_tuple, 0, :baz)
# Note: think about immutability
assert elem(a_tuple, 0) == __?
assert elem(baz_tuple, 0) == __?
end
think "Setting a tuple element that not exists raise an argument error" do
a_tuple = {:foo, :bar}
assert_raise ArgumentError, fn -> put_elem(a_tuple, __?, :baz) end
end
think "Can insert a tuple element" do
a_tuple = {:foo, :bar}
baz_tuple = Tuple.insert_at(a_tuple, 2, :baz)
assert elem(baz_tuple, 2) == __?
end
think "Inserting a tuple element raise an argument error if index is invalid" do
a_tuple = {:foo, :bar}
assert_raise ArgumentError, fn -> Tuple.insert_at(a_tuple, __?, :baz) end
end
think "The building blocks of Elixir are tuples of three elements" do
# Note: quote gives the representation of a block
quoted_block = quote do
a = 1 + 2
assert a == 3
end
assert is_tuple(quoted_block)
assert tuple_size(quoted_block) == __?
end
test "We can use tuples to define blocks" do
# Note: unquote is the reverse of quote
# It gives a block from its representation
unquoted_block = unquote {:"{}", [], [1, 2, 3]}
# Note: it's an hungarian notation
# the atom :"{}" represent the function
# the list [] contains metadatas like the line and module where code is defined
# the list [1, 2, 3] are arguments passed to the function
# For more infos see Macros and quote/unquote functions
assert unquoted_block == __?
end
think "Are tuples enumerables ?" do
invalid_argument = ArgumentError
undefined_protocol = Protocol.UndefinedError
assert_raise __?, fn -> Enum.empty?({1, 2, 3}) end
# Note:
# Do you smell the underlying machinery that make list and tuple types differents ?
end
end