defmodule ServerProcess do def start(callback_module) do spawn(fn -> initial_state = callback_module.init() loop(callback_module, initial_state) end) end def loop(callback_module, current_state) do receive do {:call, request, caller} -> {response, new_state} = callback_module.handle_call( request, current_state ) send(caller, {:response, response}) loop(callback_module, new_state) {:cast, request} -> new_state = callback_module.handle_cast( request, current_state ) loop(callback_module, new_state) end end def call(server_pid, request) do send(server_pid, {:call, request, self()}) receive do {:response, response} -> response end end def cast(server_pid, request) do send(server_pid, {:cast, request}) end end defmodule TodoServer do def start do ServerProcess.start(TodoServer) end def init, do: TodoList.new() def entries(server_pid, date) do ServerProcess.call(server_pid, {:entries, date}) end def add_entry(server_pid, entry) do ServerProcess.cast(server_pid, {:add_entry, entry}) end def delete_entry(server_pid, entry_id) do ServerProcess.cast(server_pid, {:delete_entry, entry_id}) end def update_entry(server_pid, entry_id, updater_fun) do ServerProcess.cast(server_pid, {:update_entry, entry_id, updater_fun}) end def handle_cast({:add_entry, entry}, state) do TodoList.add_entry(state, entry) end def handle_cast({:delete_entry, entry_id}, state) do TodoList.delete_entry(state, entry_id) end def handle_cast({:update_entry, entry_id, updater_fun}, state) do TodoList.update_entry(state, entry_id, updater_fun) end def handle_call({:entries, date}, state) do entries = TodoList.entries(state, date) {entries, state} end end defmodule TodoList do defstruct auto_id: 1, entries: %{} def new(), do: %TodoList{} def add_entry(todo_list, entry) do entry = Map.put(entry, :id, todo_list.auto_id) new_entries = Map.put( todo_list.entries, todo_list.auto_id, entry ) %TodoList{todo_list | entries: new_entries, auto_id: todo_list.auto_id + 1} end def entries(todo_list, date) do todo_list.entries |> Stream.filter(fn {_, entry} -> entry.date == date end) |> Enum.map(fn {_, entry} -> entry end) end def update_entry(todo_list, %{} = new_entry) do update_entry(todo_list, new_entry.id, fn _ -> new_entry end) end def update_entry(todo_list, entry_id, updater_fun) do case Map.fetch(todo_list.entries, entry_id) do :error -> todo_list {:ok, old_entry} -> old_entry_id = old_entry.id # Make sure that the result of the updater is a map and the # id remains unchanged. new_entry = %{id: ^old_entry_id} = updater_fun.(old_entry) new_entries = Map.put(todo_list.entries, new_entry.id, new_entry) %TodoList{todo_list | entries: new_entries} end end def delete_entry(todo_list, entry_id) do %TodoList{todo_list | entries: Map.delete(todo_list.entries, entry_id)} end end