140 lines
3.0 KiB
Elixir
140 lines
3.0 KiB
Elixir
defmodule TodoServer do
|
|
use GenServer
|
|
|
|
@impl GenServer
|
|
def init(%_{} = todo_list) do
|
|
{:ok, todo_list}
|
|
end
|
|
|
|
@impl GenServer
|
|
def init(entries) do
|
|
{:ok, TodoList.new(entries)}
|
|
end
|
|
|
|
@impl GenServer
|
|
def handle_call({:entries, date}, _from, todo_list) do
|
|
entries = TodoList.entries(todo_list, date)
|
|
|
|
{:reply, entries, todo_list}
|
|
end
|
|
|
|
@impl GenServer
|
|
def handle_cast({:add, entry}, todo_list) do
|
|
new_todo_list = TodoList.add(todo_list, entry)
|
|
|
|
{:no_reply, new_todo_list}
|
|
end
|
|
|
|
@impl GenServer
|
|
def handle_cast({:update, id, update_fun}, todo_list) do
|
|
new_todo_list = TodoList.update(todo_list, id, update_fun)
|
|
|
|
{:no_reply, new_todo_list}
|
|
end
|
|
|
|
@impl GenServer
|
|
def handle_cast({:delete, id}, todo_list) do
|
|
new_todo_list = TodoList.delete(todo_list, id)
|
|
|
|
{:no_reply, new_todo_list}
|
|
end
|
|
|
|
def start(entries \\ [])
|
|
|
|
def start(%_{} = todo_list) do
|
|
GenServer.start(__MODULE__, todo_list)
|
|
end
|
|
|
|
def start(entries) do
|
|
GenServer.start(__MODULE__, entries)
|
|
end
|
|
|
|
def add(pid, entry) do
|
|
GenServer.cast(pid, {:add, entry})
|
|
end
|
|
|
|
def entries(pid, date) do
|
|
GenServer.call(pid, {:entries, date})
|
|
end
|
|
|
|
def update(pid, id, update_fun) do
|
|
GenServer.cast(pid, {:update, id, update_fun})
|
|
end
|
|
|
|
def delete(pid, id) do
|
|
GenServer.cast(pid, {:delete, id})
|
|
end
|
|
end
|
|
|
|
defmodule TodoList do
|
|
defstruct entries: %{}, next_id: 1
|
|
|
|
def new(entries \\ []) do
|
|
Enum.reduce(entries, %__MODULE__{}, &add(&2, &1))
|
|
end
|
|
|
|
def add(
|
|
%__MODULE__{entries: entries, next_id: next_id} = todo_list,
|
|
%{date: _, title: _} = entry
|
|
) do
|
|
new_entry = Map.put(entry, :id, next_id)
|
|
|
|
new_entries = Map.put(entries, next_id, new_entry)
|
|
|
|
%__MODULE__{todo_list | entries: new_entries, next_id: next_id + 1}
|
|
end
|
|
|
|
def entries(%__MODULE__{} = todo_list, date) do
|
|
todo_list.entries
|
|
|> Map.filter(fn {_, entry} -> entry.date == date end)
|
|
|> Map.values()
|
|
end
|
|
|
|
def update(%__MODULE__{entries: entries} = todo_list, id, update_fun)
|
|
when is_function(update_fun, 1) do
|
|
case Map.fetch(entries, id) do
|
|
:error ->
|
|
todo_list
|
|
|
|
{:ok, entry} ->
|
|
new_entry = update_fun.(entry)
|
|
new_entries = Map.put(entries, id, new_entry)
|
|
%__MODULE__{todo_list | entries: new_entries}
|
|
end
|
|
end
|
|
|
|
def delete(%__MODULE__{entries: entries} = todo_list, id) when is_number(id) do
|
|
new_entries = Map.delete(entries, id)
|
|
|
|
%__MODULE__{todo_list | entries: new_entries}
|
|
end
|
|
end
|
|
|
|
defmodule TodoList.CSVImporter do
|
|
def import(path) do
|
|
path
|
|
|> File.stream!()
|
|
|> Stream.map(&parse_line/1)
|
|
|> TodoList.new()
|
|
end
|
|
|
|
defp parse_line(line) do
|
|
line
|
|
|> String.trim()
|
|
|> String.split(",")
|
|
|> create_entry()
|
|
end
|
|
|
|
defp create_entry([date, title]) do
|
|
final_date = parse_date(date)
|
|
|
|
%{date: final_date, title: title}
|
|
end
|
|
|
|
defp parse_date(string) do
|
|
[year, month, day] = string |> String.split("-") |> Enum.map(&String.to_integer/1)
|
|
|
|
Date.new!(year, month, day)
|
|
end
|
|
end
|