diff options
author | Tomasz Chojnacki <tomaszchojnacki2001@gmail.com> | 2023-04-05 21:07:23 +0200 |
---|---|---|
committer | Tomasz Chojnacki <tomaszchojnacki2001@gmail.com> | 2023-04-05 21:07:23 +0200 |
commit | 4ecfe6624434f15f3ac00142f7a3c4d78bfab3a1 (patch) | |
tree | 53e1059b58dc5cdaa2059ba07a3da1b2437c5d91 /aoc-2020-gleam/src/util | |
parent | 8367d45862c59889e1b9824bab2e11645c09521f (diff) | |
download | gleam_aoc2020-4ecfe6624434f15f3ac00142f7a3c4d78bfab3a1.tar.gz gleam_aoc2020-4ecfe6624434f15f3ac00142f7a3c4d78bfab3a1.zip |
Finish day 10
Diffstat (limited to 'aoc-2020-gleam/src/util')
-rw-r--r-- | aoc-2020-gleam/src/util/cache.gleam | 53 |
1 files changed, 53 insertions, 0 deletions
diff --git a/aoc-2020-gleam/src/util/cache.gleam b/aoc-2020-gleam/src/util/cache.gleam new file mode 100644 index 0000000..e1eefd2 --- /dev/null +++ b/aoc-2020-gleam/src/util/cache.gleam @@ -0,0 +1,53 @@ +import gleam/map.{Map} +import gleam/otp/actor.{Continue, Next, Stop} +import gleam/erlang/process.{Normal, Subject} + +const timeout = 1000 + +type Message(k, v) { + Shutdown + Get(key: k, client: Subject(Result(v, Nil))) + Set(key: k, value: v) +} + +type Server(k, v) = + Subject(Message(k, v)) + +fn handle_message(message: Message(k, v), dict: Map(k, v)) -> Next(Map(k, v)) { + case message { + Shutdown -> Stop(Normal) + Get(key, client) -> { + process.send(client, map.get(dict, key)) + Continue(dict) + } + Set(key, value) -> Continue(map.insert(dict, key, value)) + } +} + +pub opaque type Cache(k, v) { + Cache(server: Server(k, v)) +} + +pub fn create(apply fun: fn(Cache(k, v)) -> t) -> t { + let assert Ok(server) = actor.start(map.new(), handle_message) + let result = fun(Cache(server)) + process.send(server, Shutdown) + result +} + +pub fn set(in cache: Cache(k, v), for key: k, insert value: v) -> Nil { + process.send(cache.server, Set(key, value)) +} + +pub fn get(from cache: Cache(k, v), fetch key: k) -> Result(v, Nil) { + process.call(cache.server, fn(c) { Get(key, c) }, timeout) +} + +pub fn memoize(with cache: Cache(k, v), this key: k, apply fun: fn() -> v) -> v { + let result = case get(from: cache, fetch: key) { + Ok(value) -> value + Error(Nil) -> fun() + } + set(in: cache, for: key, insert: result) + result +} |